diff --git a/.env.example b/.env.example index 888cbe6b..bbab9bb8 100644 --- a/.env.example +++ b/.env.example @@ -1,7 +1,3 @@ -# The format: -# SPOTIFY_SECRETS=clintId1:clientSecret1,clientId2:clientSecret2 -SPOTIFY_SECRETS=$SPOTIFY_SECRETS - # 0 or 1 # 0 = disable # 1 = enable @@ -13,4 +9,4 @@ LASTFM_API_SECRET=$LASTFM_API_SECRET # Release channel. Can be: nightly, stable RELEASE_CHANNEL=$RELEASE_CHANNEL -HIDE_DONATIONS=$HIDE_DONATIONS +HIDE_DONATIONS=$HIDE_DONATIONS \ No newline at end of file diff --git a/.fvm/fvm_config.json b/.fvm/fvm_config.json index 7f89fed4..58b893ee 100644 --- a/.fvm/fvm_config.json +++ b/.fvm/fvm_config.json @@ -1,3 +1,3 @@ { - "flutterSdkVersion": "3.24.5" + "flutterSdkVersion": "3.35.2" } \ No newline at end of file diff --git a/.fvmrc b/.fvmrc index 679f8e11..cf986e39 100644 --- a/.fvmrc +++ b/.fvmrc @@ -1,4 +1,4 @@ { - "flutter": "3.24.5", + "flutter": "3.35.2", "flavors": {} } \ No newline at end of file diff --git a/.github/Dockerfile b/.github/Dockerfile deleted file mode 100644 index f6a9f538..00000000 --- a/.github/Dockerfile +++ /dev/null @@ -1,25 +0,0 @@ -ARG FLUTTER_VERSION - -FROM --platform=linux/arm64 krtirtho/flutter_distributor:${FLUTTER_VERSION} - -ARG BUILD_VERSION - -WORKDIR /app - -COPY . . - -RUN chown -R $(whoami) /app - -RUN rustup target add aarch64-unknown-linux-gnu - -RUN flutter pub get - -RUN alias dpkg-deb="dpkg-deb --Zxz" &&\ - flutter_distributor package --platform=linux --targets=deb --skip-clean - -RUN make tar VERSION=${BUILD_VERSION} ARCH=arm64 PKG_ARCH=aarch64 - -RUN mv build/spotube-linux-*-aarch64.tar.xz dist/ &&\ - mv dist/**/spotube-*-linux.deb dist/Spotube-linux-aarch64.deb - -CMD [ "sleep", "5000000" ] \ No newline at end of file diff --git a/.github/ISSUE_TEMPLATE/bug_report.yml b/.github/ISSUE_TEMPLATE/bug_report.yml index a9c57836..d4872798 100644 --- a/.github/ISSUE_TEMPLATE/bug_report.yml +++ b/.github/ISSUE_TEMPLATE/bug_report.yml @@ -9,7 +9,8 @@ body: attributes: label: Is there an existing issue for this? (Please read the description) description: | - PLEASE! Make sure to check if this issue is a duplicate. + 🚨 PLEASE! Make sure to check if this issue is a duplicate. 🚨 + Don't waste our time, we are working hard to make Spotube better for you. Try with multiple similar keywords, and check the closed issues too. @@ -50,7 +51,7 @@ body: value: |
Logs - + ``` ``` @@ -60,7 +61,7 @@ body: - type: input attributes: label: Operating System - description: The OS in which you used Spotube to face the issue. + description: The OS in which you used Spotube to face the issue. Use comma to separate multiple OS. placeholder: Android, Linux, macOS or Windows? Make sure to include the version too. validations: required: true @@ -96,7 +97,10 @@ body: - type: checkboxes attributes: label: Self grab - description: If you are a developer and want to work on this issue yourself, you can check this box and wait for maintainer response. Any contributions are welcome! + description: | + If you are a developer and want to work on this issue yourself, you can check this box and wait for maintainer response. Any contributions are welcome! + + This project is maintained by one person. So PRs are always welcome. This is the best way to get your issue fixed faster. options: - label: I'm ready to work on this issue! required: false diff --git a/.github/workflows/pr-lint.yml b/.github/workflows/pr-lint.yml index 23e5cc74..3e73be4d 100644 --- a/.github/workflows/pr-lint.yml +++ b/.github/workflows/pr-lint.yml @@ -4,7 +4,7 @@ on: pull_request: env: - FLUTTER_VERSION: 3.24.5 + FLUTTER_VERSION: 3.35.2 jobs: lint: @@ -21,14 +21,12 @@ jobs: run: | envsubst < .env.example > .env env: - SPOTIFY_SECRETS: xxx:xxx ENABLE_UPDATE_CHECK: true LASTFM_API_KEY: xxx LASTFM_API_SECRET: xxx RELEASE_CHANNEL: nightly HIDE_DONATIONS: 0 - - name: Configure repo run: | flutter pub get @@ -36,4 +34,4 @@ jobs: - name: Lint Dart files run: | - dart analyze --no-fatal-warnings \ No newline at end of file + dart analyze --no-fatal-warnings diff --git a/.github/workflows/spotube-publish-binary.yml b/.github/workflows/spotube-publish-binary.yml index 3a456bda..e682dbdd 100644 --- a/.github/workflows/spotube-publish-binary.yml +++ b/.github/workflows/spotube-publish-binary.yml @@ -4,7 +4,7 @@ on: inputs: version: description: Version to publish (x.x.x) - default: 3.8.3 + default: 4.0.0 required: true dry_run: description: Dry run @@ -12,10 +12,10 @@ on: type: boolean default: true jobs: - description: Jobs to run (flathub,aur,winget,chocolatey,playstore) + description: Jobs to run (flathub,aur,winget,chocolatey) required: true type: string - default: "flathub,aur,winget,chocolatey,playstore" + default: "flathub,aur,winget,chocolatey" jobs: flathub: @@ -55,7 +55,7 @@ jobs: - uses: dsaltares/fetch-gh-release-asset@master with: version: tags/v${{ inputs.version }} # mind the "v" prefix - file: spotube-linux-${{inputs.version}}-x86_64.tar.xz + file: spotube-linux-${{inputs.version}}-x86_64.tar.xz token: ${{ secrets.GITHUB_TOKEN }} - name: Update PKGBUILD versions @@ -111,27 +111,27 @@ jobs: steps: - name: Tagname (workflow dispatch) run: echo 'TAG_NAME=${{inputs.version}}' >> $GITHUB_ENV - - - uses: robinraju/release-downloader@main - with: - repository: KRTirtho/spotube - tag: v${{ env.TAG_NAME }} - tarBall: false - zipBall: false - out-file-path: dist - fileName: "Spotube-playstore-all-arch.aab" - - - name: Create service-account.json - run: | - echo "${{ secrets.GOOGLE_PLAY_SERVICE_ACCOUNT_BASE64 }}" | base64 -d > service-account.json - - - name: Upload Android Release to Play Store - if: ${{!inputs.dry_run}} - uses: r0adkll/upload-google-play@v1 - with: - serviceAccountJson: ./service-account.json - releaseFiles: ./dist/Spotube-playstore-all-arch.aab - packageName: oss.krtirtho.spotube - track: production - status: draft - releaseName: ${{ env.TAG_NAME }} + + # - uses: robinraju/release-downloader@main + # with: + # repository: KRTirtho/spotube + # tag: v${{ env.TAG_NAME }} + # tarBall: false + # zipBall: false + # out-file-path: dist + # fileName: "Spotube-playstore-all-arch.aab" + + # - name: Create service-account.json + # run: | + # echo "${{ secrets.GOOGLE_PLAY_SERVICE_ACCOUNT_BASE64 }}" | base64 -d > service-account.json + + # - name: Upload Android Release to Play Store + # if: ${{!inputs.dry_run}} + # uses: r0adkll/upload-google-play@v1 + # with: + # serviceAccountJson: ./service-account.json + # releaseFiles: ./dist/Spotube-playstore-all-arch.aab + # packageName: oss.krtirtho.spotube + # track: production + # status: draft + # releaseName: ${{ env.TAG_NAME }} diff --git a/.github/workflows/spotube-release-binary.yml b/.github/workflows/spotube-release-binary.yml index 89c2fedd..dfec7d44 100644 --- a/.github/workflows/spotube-release-binary.yml +++ b/.github/workflows/spotube-release-binary.yml @@ -20,7 +20,8 @@ on: description: Dry run without uploading to release env: - FLUTTER_VERSION: 3.24.5 + FLUTTER_VERSION: 3.35.2 + FLUTTER_CHANNEL: master permissions: contents: write @@ -30,97 +31,102 @@ jobs: strategy: matrix: include: - - os: ubuntu-latest + - os: ubuntu-22.04 platform: linux + arch: x86 files: | dist/Spotube-linux-x86_64.deb dist/Spotube-linux-x86_64.rpm + dist/Spotube-linux-x86_64.AppImage dist/spotube-linux-*-x86_64.tar.xz - - os: ubuntu-latest - platform: linux_arm + - os: ubuntu-22.04-arm + platform: linux + arch: arm64 files: | dist/Spotube-linux-aarch64.deb + dist/Spotube-linux-aarch64.AppImage dist/spotube-linux-*-aarch64.tar.xz - - os: ubuntu-latest + - os: ubuntu-22.04 platform: android + arch: all files: | build/Spotube-android-all-arch.apk - build/Spotube-playstore-all-arch.aab - os: windows-latest platform: windows + arch: x86 files: | dist/Spotube-windows-x86_64.nupkg dist/Spotube-windows-x86_64-setup.exe - - os: macos-latest + - os: macos-14 platform: ios + arch: all files: | Spotube-iOS.ipa - os: macos-14 platform: macos + arch: all files: | build/Spotube-macos-universal.dmg build/Spotube-macos-universal.pkg runs-on: ${{matrix.os}} steps: - uses: actions/checkout@v4 - - uses: subosito/flutter-action@v2.12.0 + - uses: subosito/flutter-action@v2.18.0 with: - cache: true - cache-key: ${{ runner.os }}-flutter-${{ hashFiles('**/pubspec.yaml') }} flutter-version: ${{ env.FLUTTER_VERSION }} + channel: ${{ env.FLUTTER_CHANNEL }} + cache: true + git-source: https://github.com/flutter/flutter.git + + - name: free disk space + if: ${{ matrix.platform == 'android' }} + run: | + sudo swapoff -a + sudo rm -f /swapfile + sudo apt clean + docker rmi $(docker image ls -aq) + df -h - name: Setup Java if: ${{matrix.platform == 'android'}} uses: actions/setup-java@v4 with: - distribution: 'zulu' - java-version: '17' - cache: 'gradle' + distribution: "zulu" + java-version: "17" + cache: "gradle" check-latest: true - - name: Set up QEMU - if: ${{matrix.platform == 'linux_arm'}} - uses: docker/setup-qemu-action@v3 - - name: Set up Docker Buildx - if: ${{matrix.platform == 'linux_arm'}} - uses: docker/setup-buildx-action@v3 + - name: Setup Rust toolchain - if: ${{matrix.platform != 'linux_arm'}} uses: dtolnay/rust-toolchain@stable with: toolchain: stable + - name: Install Xcode + if: ${{matrix.platform == 'ios'}} + uses: maxim-lobanov/setup-xcode@v1 + with: + xcode-version: "16.2" + - name: Install ${{matrix.platform}} dependencies run: | flutter pub get - dart cli/cli.dart install-dependencies --platform=${{matrix.platform}} + dart cli/cli.dart install-dependencies --platform=${{matrix.platform}} --arch=${{matrix.arch}} - name: Sign Apk if: ${{matrix.platform == 'android'}} run: | echo '${{ secrets.KEYSTORE }}' | base64 --decode > android/app/upload-keystore.jks echo '${{ secrets.KEY_PROPERTIES }}' > android/key.properties - - - name: Unessary hosted tools - if: ${{matrix.platform == 'linux_arm'}} - uses: jlumbroso/free-disk-space@main - with: - tool-cache: false - swap-storage: false - android: true - dotnet: true - haskell: true - large-packages: true - docker-images: true - + - name: Build ${{matrix.platform}} binaries - run: dart cli/cli.dart build ${{matrix.platform}} + run: dart cli/cli.dart build --arch=${{matrix.arch}} ${{matrix.platform}} env: CHANNEL: ${{inputs.channel}} DOTENV: ${{secrets.DOTENV_RELEASE}} - - - uses: actions/upload-artifact@v3 + + - uses: actions/upload-artifact@v4 with: if-no-files-found: error - name: Spotube-Release-Binaries + name: ${{matrix.platform}}-${{matrix.arch}} path: ${{matrix.files}} - name: Debug With SSH When fails @@ -130,14 +136,13 @@ jobs: limit-access-to-actor: true upload: - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 needs: - build_platform steps: - uses: actions/checkout@v4 - - uses: actions/download-artifact@v3 + - uses: actions/download-artifact@v4 with: - name: Spotube-Release-Binaries path: ./Spotube-Release-Binaries - name: Install dependencies @@ -146,18 +151,19 @@ jobs: - name: Generate Checksums run: | tree . - md5sum Spotube-Release-Binaries/* >> RELEASE.md5sum - sha256sum Spotube-Release-Binaries/* >> RELEASE.sha256sum + find Spotube-Release-Binaries -type f -exec md5sum {} \; >> RELEASE.md5sum + find Spotube-Release-Binaries -type f -exec sha256sum {} \; >> RELEASE.sha256sum + sed -i 's|Spotube-Release-Binaries/.*/\([^/]*\)$|\1|' RELEASE.sha256sum RELEASE.md5sum sed -i 's|Spotube-Release-Binaries/||' RELEASE.sha256sum RELEASE.md5sum - + - name: Extract pubspec version run: | echo "PUBSPEC_VERSION=$(grep -oP 'version:\s*\K[^+]+(?=\+)' pubspec.yaml)" >> $GITHUB_ENV - - uses: actions/upload-artifact@v3 + - uses: actions/upload-artifact@v4 with: if-no-files-found: error - name: Spotube-Release-Binaries + name: sums path: | RELEASE.md5sum RELEASE.sha256sum @@ -172,7 +178,7 @@ jobs: omitNameDuringUpdate: true omitPrereleaseDuringUpdate: true allowUpdates: true - artifacts: Spotube-Release-Binaries/*,RELEASE.sha256sum,RELEASE.md5sum + artifacts: Spotube-Release-Binaries/**/*,RELEASE.sha256sum,RELEASE.md5sum - name: Upload Release Binaries (nightly) if: ${{ !inputs.dry_run && inputs.channel == 'nightly' }} @@ -184,9 +190,15 @@ jobs: omitNameDuringUpdate: true omitPrereleaseDuringUpdate: true allowUpdates: true - artifacts: Spotube-Release-Binaries/*,RELEASE.sha256sum,RELEASE.md5sum + artifacts: Spotube-Release-Binaries/**/*,RELEASE.sha256sum,RELEASE.md5sum body: | Build Number: ${{github.run_number}} - + Nightly release includes newest features but may contain bugs It is preferred to use the stable version unless you know what you're doing + + - name: Debug With SSH When fails + if: ${{ failure() && inputs.debug && inputs.channel == 'nightly' }} + uses: mxschmitt/action-tmate@v3 + with: + limit-access-to-actor: true diff --git a/.gitignore b/.gitignore index f9bd15f8..544dbba8 100644 --- a/.gitignore +++ b/.gitignore @@ -9,6 +9,7 @@ .history .svn/ + # IntelliJ related *.iml *.ipr @@ -79,4 +80,7 @@ android/key.properties tm.json # FVM Version Cache -.fvm/ \ No newline at end of file +.fvm/ + +android/build +android/app/.cxx diff --git a/.metadata b/.metadata index 828f2c0a..e8b36fde 100644 --- a/.metadata +++ b/.metadata @@ -4,7 +4,7 @@ # This file should be version controlled and should not be manually edited. version: - revision: "300451adae589accbece3490f4396f10bdf15e6e" + revision: "d7b523b356d15fb81e7d340bbe52b47f93937323" channel: "stable" project_type: app @@ -13,11 +13,11 @@ project_type: app migration: platforms: - platform: root - create_revision: 300451adae589accbece3490f4396f10bdf15e6e - base_revision: 300451adae589accbece3490f4396f10bdf15e6e + create_revision: d7b523b356d15fb81e7d340bbe52b47f93937323 + base_revision: d7b523b356d15fb81e7d340bbe52b47f93937323 - platform: windows - create_revision: 300451adae589accbece3490f4396f10bdf15e6e - base_revision: 300451adae589accbece3490f4396f10bdf15e6e + create_revision: d7b523b356d15fb81e7d340bbe52b47f93937323 + base_revision: d7b523b356d15fb81e7d340bbe52b47f93937323 # User provided section diff --git a/.vscode/launch.json b/.vscode/launch.json index 7a1e8b9b..b81e2eee 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -17,6 +17,17 @@ "dev" ] }, + { + "name": "spotube (mobile-skia)", + "type": "dart", + "request": "launch", + "program": "lib/main.dart", + "args": [ + "--flavor", + "dev", + "--no-enable-impeller" + ] + }, { "name": "spotube (profile)", "type": "dart", @@ -30,6 +41,17 @@ "request": "launch", "program": "lib/main.dart", "flutterMode": "release" + }, + { + "name": "spotube (mobile) (release)", + "type": "dart", + "request": "launch", + "program": "lib/main.dart", + "flutterMode": "release", + "args": [ + "--flavor", + "dev" + ] } ], "compounds": [] diff --git a/.vscode/settings.json b/.vscode/settings.json index 11fae610..69c80bb3 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -5,14 +5,17 @@ "ambiguate", "Amoled", "Buildless", + "configurators", "danceability", "fuzzywuzzy", "gapless", "instrumentalness", + "isrc", "Mpris", "RGBO", "riverpod", "Scrobblenaut", + "shadcn", "skeletonizer", "songlink", "speechiness", @@ -27,5 +30,5 @@ "README.md": "LICENSE,CODE_OF_CONDUCT.md,CONTRIBUTING.md,SECURITY.md,CONTRIBUTION.md,CHANGELOG.md,PRIVACY_POLICY.md", "*.dart": "${capture}.g.dart,${capture}.freezed.dart" }, - "dart.flutterSdkPath": ".fvm/flutter_sdk" + "dart.flutterSdkPath": ".fvm/versions/3.35.2" } \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md index 471d5a95..b8a5b0e1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,146 @@ # Changelog -All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. +## [5.1.0](https://github.com/KRTirtho/spotube/compare/v5.0.0...v5.1.0) (2025-11-14) + +### Features + +- Show plugin source and set the only plugin as default if no plugin is there +- **playback**: Add dab music source +- **playback**: Add uncompressed flac playback support +- Add plugin audio source models and api service +- **plugins**: Filter plugins by abilities in plugins page and show abilities as badge +- Add setting default audio source support +- Move away from track source query and preferences audio quality and codec +- Add NewPipe support for desktop platforms +- Add default plugin loading capability +- **queue**: Add multi-select and bulk actions to queue ([#2839](https://github.com/KRTirtho/spotube/issues/2839)) +- **android**: Add 16KB page size support + +### Bug Fixes + +- Change plugin download directory to application support +- **playback**: Play next not working +- Downloaded tracks are not tagged with metadata +- Download not working in different devices and slow +- **playback**: Use stream instead of chunked serving of audio bytes + +## [5.0.0](https://github.com/KRTirtho/spotube/compare/v4.0.2...v5.0.0) (2025-09-08) + +### Features + +- Add ISRC track search for YouTube ([#2594](https://github.com/KRTirtho/spotube/issues/2594)) +- Add new icons #2676 by @alexio-dev ([#2678](https://github.com/KRTirtho/spotube/issues/2678)) +- Add connect confirmation dialog +- Add metadata api service and models +- **metadata-plugin**: Add pagination support, feed and playlist CRUD endpoints +- **metadata-plugin**: Add local storage api +- Add webview, totp and setInterval apis for plugins +- Enhance local storage and webview APIs with improved error handling and resource management +- **metadata_plugin**: Add logout method +- Update plugin configuration with more fields +- Implement metadata plugins based on hetu +- Update models to match hetu_spotube_plugin signature +- Add user endpoint calls in metadata and paginated async notifiers +- Add playlist endpoint and providers +- Add albums metadata endpoint and provider +- Add artist and album providers +- Add track endpoint for metadata service +- Remove green corp names formally +- **metadata**: Add plugin form +- Add support for entity specific search +- Enhance image handling +- Add support for automatic plugin repository from github and codeberg +- Use isolate for youtube_explode engine +- Add repository and plugin API version fields to metadata plugins +- Update new pipe version +- **metadata**: Add plugin update checker and dialog for available updates +- Optimize track options and related artists +- Add plugin scrobbling support and support button +- Add ErrorBox and NoDefaultMetadataPlugin components + +### Bug Fixes + +- Calling /track/:streamId endpoint causes active sourced track to be anything +- **mobile**: Dialogs in bottom sheet are not opening +- Default accent color is orange but it shows blue in settings +- Artist images are not loading up +- CVE: Remote path traversal through websocket when devices are on same network +- Endless playback not working +- **android**: NewPipe invalid search content filters +- Make YoutubeExplode engine faster +- Create and delete playlist not working +- Local track not working and images of local not showing up +- Local playback not working for tracks with special # (hashtag) characters +- Inaccessible streaming url causing rapid skips +- **yt**: Fallback to different search result if all streaming url is inaccessible +- **playback**: Skip network requests if cached file already exists +- Yt-dlp playback not working and add partial support for HLS streaming +- Windows webview2 environment permission issue +- **playback**: Play not fetching full playlist if playlist is too long +- **track_options**: Tapping on option doesn't close the menu +- **playback**: Alternative track sources switch not working +- **ui**: Lyrics white text in white background and small player buttons + +### Translation + +- Add Traditional Chinese translation ([#2762](https://github.com/KRTirtho/spotube/issues/2762)) +- Fix Japanese translations ([#2732](https://github.com/KRTirtho/spotube/issues/2732)) +- Correction of the dutch language ([#1306](https://github.com/KRTirtho/spotube/issues/1306)) + +## [4.0.2](https://github.com/krtirtho/spotube/compare/v4.0.1...v4.0.2) (2025-03-16) + +### Bug Fixes + +- invalid access token exception #2525 + +## [4.0.1](https://github.com/krtirtho/spotube/compare/v4.0.0...v4.0.1) (2025-03-15) + +### Bug Fixes + +- **android**: navigation overlaying in app navigation +- add to playlist not working in smaller screen devices +- language picker search broken +- **generate_playlist**: create playlist not adding tracks nor navigating to playlist page +- **desktop**: double titlebar in local library folders and massive space in overlay player +- lastfm form broken in other locales #2447 +- spotify login broken due to new totp requirement #2494 +- spotify authentication 429 errors + +### Features + +- **local_library**: add support for x-flac, opus and x-wav +- **translation**: add tagalog language support #2504 +- **translation**: add tamil translation for spotube #2501 + +## [4.0.0](https://github.com/krtirtho/spotube/compare/v3.9.0...v4.0.0) (2025-03-07) + +## Changes + +### Bug Fixes + +- SafeArea for global bottom items not working +- SafeArea not working for bottom floating widgets +- youtube video unplayable issue and use more ytClients +- **ios**: downloads not working due to permission errors (#2180) +- remove automaticallyImplyLeading from root tabs +- **android**: back button not working and safe area issues +- duplicates in recent section +- youtube source taking too long to buffer +- youtube tracks keeps skipping despite being matched correctly +- follow artist not working #2057 +- youtube_explode_dart failing for many videos due to youtube ios client visitor data change +- piped api not working + +### Features + +- **android**: home widget support (#2148) +- rewrite entire app in shadcn-ui replacing material eww +- grid/list customizable playbutton view +- flag to hide spotify generated images with patterns +- show placeholder images where there is no item or on empty page +- pause playback when no internet connection +- implement yt-dlp for desktop and NewPipeExtractor for Android (#2316) +- custom piped & invidious instance support ## [3.9.0](https://github.com/krtirtho/spotube/compare/v3.8.3...v3.9.0) (2024-12-08) @@ -17,7 +157,6 @@ All notable changes to this project will be documented in this file. See [standa - add invidious audio source and fix auto skipping tracks (#2005) - track caching and cached track export support (#2117) - ## [3.8.3](https://github.com/krtirtho/spotube/compare/v3.8.2...v3.8.3) (2024-10-09) ## Changes @@ -38,7 +177,6 @@ All notable changes to this project will be documented in this file. See [standa - endless song loading issue and no playback #1925 - ## [3.8.1](https://github.com/krtirtho/spotube/compare/v3.8.0...v3.8.1) (2024-09-15) ## Changes @@ -65,7 +203,6 @@ All notable changes to this project will be documented in this file. See [standa - **desktop**: show error dialog if webview is not found on login #1871 - manually detect and define touch behavior #1763 - ## [3.8.0](https://github.com/krtirtho/spotube/compare/v3.7.1...v3.8.0) (2024-06-06) ### Features @@ -100,708 +237,672 @@ All notable changes to this project will be documented in this file. See [standa - popup menu item opacity - linux: change app id in flatpak environment - ## [3.7.1](https://github.com/krtirtho/spotube/compare/v3.7.0...v3.7.1) (2024-06-06) - ### Bug Fixes -* alternative sources not showing up for SongLink matched results ([37d002d](https://github.com/krtirtho/spotube/commit/37d002d133cacb3a34884713ac8f6637694af57c)) -* **android:** Media Controls not working above Android 14 [#1561](https://github.com/krtirtho/spotube/issues/1561) ([3394c1b](https://github.com/krtirtho/spotube/commit/3394c1b0574e44dc624b2b2f0bf32f343ae9f049)) -* browse anonymously button takes to wrong route ([73c5b30](https://github.com/krtirtho/spotube/commit/73c5b30b63a4c82bb7ad5b52bc10c5594566a800)) -* **desktop:** titlebar drag to move not working ([5f280a1](https://github.com/krtirtho/spotube/commit/5f280a19f4d5f8882ae2ff60c6cc595ded7a5a1d)) -* **desktop:** window is not centered ([47f98b9](https://github.com/krtirtho/spotube/commit/47f98b98aafab9b426733ed44cab2be8a646a98e)) -* **ios:** download not working [#1575](https://github.com/krtirtho/spotube/issues/1575) ([6591ec0](https://github.com/krtirtho/spotube/commit/6591ec0e1b441dd8fd535eace19a58c7749389ca)) -* **linux:** application window not visible after launch ([8fc44ed](https://github.com/krtirtho/spotube/commit/8fc44ed6550e8b2b804991ff82df08afb1c94ca8)) -* local track not showing up in queue ([d82261c](https://github.com/krtirtho/spotube/commit/d82261cb25ece63f85af0e40216cf32dccdc9dd5)) -* use weak match for Jiosaavn fallback to improve matching ([6cb2986](https://github.com/krtirtho/spotube/commit/6cb29868d2030b5e9312863c17e8f24889942e24)) -* **windows:** media controls not showing up [#1542](https://github.com/krtirtho/spotube/issues/1542) ([d7d864f](https://github.com/krtirtho/spotube/commit/d7d864ff2bc937675a544a7edf645c5148ec836a)) -* **windows:** revert Flutter version to 3.19.6 to avoid distortion [#1553](https://github.com/krtirtho/spotube/issues/1553) ([982cf0b](https://github.com/krtirtho/spotube/commit/982cf0bd435638fa20f17ef527fe21d031b5ffaf)) +- alternative sources not showing up for SongLink matched results ([37d002d](https://github.com/krtirtho/spotube/commit/37d002d133cacb3a34884713ac8f6637694af57c)) +- **android:** Media Controls not working above Android 14 [#1561](https://github.com/krtirtho/spotube/issues/1561) ([3394c1b](https://github.com/krtirtho/spotube/commit/3394c1b0574e44dc624b2b2f0bf32f343ae9f049)) +- browse anonymously button takes to wrong route ([73c5b30](https://github.com/krtirtho/spotube/commit/73c5b30b63a4c82bb7ad5b52bc10c5594566a800)) +- **desktop:** titlebar drag to move not working ([5f280a1](https://github.com/krtirtho/spotube/commit/5f280a19f4d5f8882ae2ff60c6cc595ded7a5a1d)) +- **desktop:** window is not centered ([47f98b9](https://github.com/krtirtho/spotube/commit/47f98b98aafab9b426733ed44cab2be8a646a98e)) +- **ios:** download not working [#1575](https://github.com/krtirtho/spotube/issues/1575) ([6591ec0](https://github.com/krtirtho/spotube/commit/6591ec0e1b441dd8fd535eace19a58c7749389ca)) +- **linux:** application window not visible after launch ([8fc44ed](https://github.com/krtirtho/spotube/commit/8fc44ed6550e8b2b804991ff82df08afb1c94ca8)) +- local track not showing up in queue ([d82261c](https://github.com/krtirtho/spotube/commit/d82261cb25ece63f85af0e40216cf32dccdc9dd5)) +- use weak match for Jiosaavn fallback to improve matching ([6cb2986](https://github.com/krtirtho/spotube/commit/6cb29868d2030b5e9312863c17e8f24889942e24)) +- **windows:** media controls not showing up [#1542](https://github.com/krtirtho/spotube/issues/1542) ([d7d864f](https://github.com/krtirtho/spotube/commit/d7d864ff2bc937675a544a7edf645c5148ec836a)) +- **windows:** revert Flutter version to 3.19.6 to avoid distortion [#1553](https://github.com/krtirtho/spotube/issues/1553) ([982cf0b](https://github.com/krtirtho/spotube/commit/982cf0bd435638fa20f17ef527fe21d031b5ffaf)) ## [3.7.0](https://github.com/krtirtho/spotube/compare/v3.6.0...v3.7.0) (2024-06-03) - ### Features -* local library folder cards ([fc5bfa0](https://github.com/krtirtho/spotube/commit/fc5bfa089ce2f46ab786565d6750564d704ee7e0)) -* Local music library ([#1479](https://github.com/krtirtho/spotube/issues/1479)) ([22caa81](https://github.com/krtirtho/spotube/commit/22caa818f4ac31626aaff6952e43512b42237d00)) -* personalized stats based on local music history ([#1522](https://github.com/krtirtho/spotube/issues/1522)) ([82307bc](https://github.com/krtirtho/spotube/commit/82307bc030035b03ab1b8d8ec7b24da19a866b12)) -* play initially available tracks of playlist/album immediately and fetch rest in background [#670](https://github.com/krtirtho/spotube/issues/670) ([02acbd9](https://github.com/krtirtho/spotube/commit/02acbd93271145dde365f6c547e0d9d902be65f1)) -* **player:** add volume slider floating label showing percentage ([#1445](https://github.com/krtirtho/spotube/issues/1445)) ([8fad225](https://github.com/krtirtho/spotube/commit/8fad2251b3536e9468e0fb193939ead98bad3bc6)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) -* **translations:** add Basque translation ([#1493](https://github.com/krtirtho/spotube/issues/1493)) ([dbc1c45](https://github.com/krtirtho/spotube/commit/dbc1c452dd53153c61589f956ea9836cea7bf2bb)) -* **translations:** add Finnish translations ([#1449](https://github.com/krtirtho/spotube/issues/1449)) ([edc997e](https://github.com/krtirtho/spotube/commit/edc997e7470ce17f60c96b8198dc8851cbf21f18)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) -* **translations:** add georgian language ([#1450](https://github.com/krtirtho/spotube/issues/1450)) ([1e7f0e1](https://github.com/krtirtho/spotube/commit/1e7f0e1fe71e0a8d86614fc884861f8791469112)) -* **translations:** add Indonesian translation ([#1426](https://github.com/krtirtho/spotube/issues/1426)) ([0280654](https://github.com/krtirtho/spotube/commit/0280654bb6bad373aee521f5a866228d2d38f038)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) -* **translations:** Improve tr locales ([#1419](https://github.com/krtirtho/spotube/issues/1419)) ([bf45681](https://github.com/krtirtho/spotube/commit/bf45681deb951c772bf6ca05e213c949c04bded1)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) -* upgrade to Flutter 3.22.0 ([71341ec](https://github.com/krtirtho/spotube/commit/71341ec0bda6ed985b43836712075b97a2cf8bac)) - +- local library folder cards ([fc5bfa0](https://github.com/krtirtho/spotube/commit/fc5bfa089ce2f46ab786565d6750564d704ee7e0)) +- Local music library ([#1479](https://github.com/krtirtho/spotube/issues/1479)) ([22caa81](https://github.com/krtirtho/spotube/commit/22caa818f4ac31626aaff6952e43512b42237d00)) +- personalized stats based on local music history ([#1522](https://github.com/krtirtho/spotube/issues/1522)) ([82307bc](https://github.com/krtirtho/spotube/commit/82307bc030035b03ab1b8d8ec7b24da19a866b12)) +- play initially available tracks of playlist/album immediately and fetch rest in background [#670](https://github.com/krtirtho/spotube/issues/670) ([02acbd9](https://github.com/krtirtho/spotube/commit/02acbd93271145dde365f6c547e0d9d902be65f1)) +- **player:** add volume slider floating label showing percentage ([#1445](https://github.com/krtirtho/spotube/issues/1445)) ([8fad225](https://github.com/krtirtho/spotube/commit/8fad2251b3536e9468e0fb193939ead98bad3bc6)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) +- **translations:** add Basque translation ([#1493](https://github.com/krtirtho/spotube/issues/1493)) ([dbc1c45](https://github.com/krtirtho/spotube/commit/dbc1c452dd53153c61589f956ea9836cea7bf2bb)) +- **translations:** add Finnish translations ([#1449](https://github.com/krtirtho/spotube/issues/1449)) ([edc997e](https://github.com/krtirtho/spotube/commit/edc997e7470ce17f60c96b8198dc8851cbf21f18)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) +- **translations:** add georgian language ([#1450](https://github.com/krtirtho/spotube/issues/1450)) ([1e7f0e1](https://github.com/krtirtho/spotube/commit/1e7f0e1fe71e0a8d86614fc884861f8791469112)) +- **translations:** add Indonesian translation ([#1426](https://github.com/krtirtho/spotube/issues/1426)) ([0280654](https://github.com/krtirtho/spotube/commit/0280654bb6bad373aee521f5a866228d2d38f038)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) +- **translations:** Improve tr locales ([#1419](https://github.com/krtirtho/spotube/issues/1419)) ([bf45681](https://github.com/krtirtho/spotube/commit/bf45681deb951c772bf6ca05e213c949c04bded1)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) +- upgrade to Flutter 3.22.0 ([71341ec](https://github.com/krtirtho/spotube/commit/71341ec0bda6ed985b43836712075b97a2cf8bac)) ### Bug Fixes -* fallback to LRCLIB when lyrics line less than 6 lines [#1461](https://github.com/krtirtho/spotube/issues/1461) ([9aea354](https://github.com/krtirtho/spotube/commit/9aea35468fa7cd176ddc8810b37b90c2d8246931)) -* **linux:** tray icon not showing [#541](https://github.com/krtirtho/spotube/issues/541) ([7ac7917](https://github.com/krtirtho/spotube/commit/7ac791757abb30f40374c169c4211916287bb3f3)) -* local track not showing up in queue ([d82261c](https://github.com/krtirtho/spotube/commit/d82261cb25ece63f85af0e40216cf32dccdc9dd5)) -* **macos:** Logs directory not created by default [#1353](https://github.com/krtirtho/spotube/issues/1353) ([4ca8939](https://github.com/krtirtho/spotube/commit/4ca893950b07f678acf7db690112c47d21e54782)) -* **playback:** skipping tracks with unplayable sources instead of falling back [#1492](https://github.com/krtirtho/spotube/issues/1492) ([c607a33](https://github.com/krtirtho/spotube/commit/c607a330ed279dfbebe8d4bd325745ac6301a58f)) -* **search:** load more button not working [#1417](https://github.com/krtirtho/spotube/issues/1417) ([7e07c2e](https://github.com/krtirtho/spotube/commit/7e07c2e1985da7ccb96b1fac2ecd703720068d26)) -* some text are garbled in different parts of the app [#1463](https://github.com/krtirtho/spotube/issues/1463) [#1505](https://github.com/krtirtho/spotube/issues/1505) ([d2683c5](https://github.com/krtirtho/spotube/commit/d2683c52d81d807be6ff72f15b8e9eb18181e211)) -* spotify friends and user profile icon (mobile) showing when not authenticated [#1410](https://github.com/krtirtho/spotube/issues/1410) ([9bccbc9](https://github.com/krtirtho/spotube/commit/9bccbc93c63dd34f6e15ff68c276976ecd1d9a33)) -* **updater:** dead link ([#1408](https://github.com/krtirtho/spotube/issues/1408)) ([6907f9c](https://github.com/krtirtho/spotube/commit/6907f9c756d8f49aadb1b23a2a1dc8bf7d658dc0)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) -* windows SSL Certificate error breaking login [#905](https://github.com/krtirtho/spotube/issues/905) ([#1474](https://github.com/krtirtho/spotube/issues/1474)) ([937a706](https://github.com/krtirtho/spotube/commit/937a706ac9c0e59943b2609e5cc398dcdbed2344)), closes [#1468](https://github.com/krtirtho/spotube/issues/1468) -* **windows:** installer tries to install in current directory ([c3c9fc5](https://github.com/krtirtho/spotube/commit/c3c9fc544c68b3d897dd7241a61cab7a199b4539)) +- fallback to LRCLIB when lyrics line less than 6 lines [#1461](https://github.com/krtirtho/spotube/issues/1461) ([9aea354](https://github.com/krtirtho/spotube/commit/9aea35468fa7cd176ddc8810b37b90c2d8246931)) +- **linux:** tray icon not showing [#541](https://github.com/krtirtho/spotube/issues/541) ([7ac7917](https://github.com/krtirtho/spotube/commit/7ac791757abb30f40374c169c4211916287bb3f3)) +- local track not showing up in queue ([d82261c](https://github.com/krtirtho/spotube/commit/d82261cb25ece63f85af0e40216cf32dccdc9dd5)) +- **macos:** Logs directory not created by default [#1353](https://github.com/krtirtho/spotube/issues/1353) ([4ca8939](https://github.com/krtirtho/spotube/commit/4ca893950b07f678acf7db690112c47d21e54782)) +- **playback:** skipping tracks with unplayable sources instead of falling back [#1492](https://github.com/krtirtho/spotube/issues/1492) ([c607a33](https://github.com/krtirtho/spotube/commit/c607a330ed279dfbebe8d4bd325745ac6301a58f)) +- **search:** load more button not working [#1417](https://github.com/krtirtho/spotube/issues/1417) ([7e07c2e](https://github.com/krtirtho/spotube/commit/7e07c2e1985da7ccb96b1fac2ecd703720068d26)) +- some text are garbled in different parts of the app [#1463](https://github.com/krtirtho/spotube/issues/1463) [#1505](https://github.com/krtirtho/spotube/issues/1505) ([d2683c5](https://github.com/krtirtho/spotube/commit/d2683c52d81d807be6ff72f15b8e9eb18181e211)) +- spotify friends and user profile icon (mobile) showing when not authenticated [#1410](https://github.com/krtirtho/spotube/issues/1410) ([9bccbc9](https://github.com/krtirtho/spotube/commit/9bccbc93c63dd34f6e15ff68c276976ecd1d9a33)) +- **updater:** dead link ([#1408](https://github.com/krtirtho/spotube/issues/1408)) ([6907f9c](https://github.com/krtirtho/spotube/commit/6907f9c756d8f49aadb1b23a2a1dc8bf7d658dc0)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1171](https://github.com/krtirtho/spotube/issues/1171) [#1082](https://github.com/krtirtho/spotube/issues/1082) [#1082](https://github.com/krtirtho/spotube/issues/1082) +- windows SSL Certificate error breaking login [#905](https://github.com/krtirtho/spotube/issues/905) ([#1474](https://github.com/krtirtho/spotube/issues/1474)) ([937a706](https://github.com/krtirtho/spotube/commit/937a706ac9c0e59943b2609e5cc398dcdbed2344)), closes [#1468](https://github.com/krtirtho/spotube/issues/1468) +- **windows:** installer tries to install in current directory ([c3c9fc5](https://github.com/krtirtho/spotube/commit/c3c9fc544c68b3d897dd7241a61cab7a199b4539)) ## [3.6.0](https://github.com/krtirtho/spotube/compare/v3.5.0...v3.6.0) (2024-04-15) - ### Features -* add Spotify homepage personalized recommendations ([#1402](https://github.com/krtirtho/spotube/issues/1402)) ([9e25c74](https://github.com/krtirtho/spotube/commit/9e25c742d4e43e4e10d2b48afb8e6d90288ffa11)) -* add user profile page ([39e97ee](https://github.com/krtirtho/spotube/commit/39e97eef34d87348a264843e145f31f82832d12e)) -* **android:** Filter Device To Force High Frame Rate ([#880](https://github.com/krtirtho/spotube/issues/880)) ([6e41b10](https://github.com/krtirtho/spotube/commit/6e41b106fa989adee393d3ce2535e75446ad3eea)) -* improved caching based on riverpod ([#1343](https://github.com/krtirtho/spotube/issues/1343)) ([6673e5a](https://github.com/krtirtho/spotube/commit/6673e5a8a86b9667cf9dbff9bb7c40ea6b7de771)) -* LAN connect a.k.a control remote Spotube playback and local output device selection ([#1355](https://github.com/krtirtho/spotube/issues/1355)) ([68374ef](https://github.com/krtirtho/spotube/commit/68374efd3ec556f31b937e5b96920787b54eec78)) -* **lyrics:** add LRCLIB lyrics provider as fallback ([5afe823](https://github.com/krtirtho/spotube/commit/5afe823abdb198340b55d138d8173d886a811632)) -* search history support [#1236](https://github.com/krtirtho/spotube/issues/1236) ([82b1cfa](https://github.com/krtirtho/spotube/commit/82b1cfa0d775e3958c666280943a893c9113d468)) -* **translations:** Add Czech translation ([#1401](https://github.com/krtirtho/spotube/issues/1401)) ([5a6b800](https://github.com/krtirtho/spotube/commit/5a6b80091259359bc38c4b91cd8cb496c4270fa4)) -* **translations:** add Thai Language ([#1319](https://github.com/krtirtho/spotube/issues/1319)) ([b70f250](https://github.com/krtirtho/spotube/commit/b70f250e8d5137fd990787ec9e3d058126cf14f3)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) - +- add Spotify homepage personalized recommendations ([#1402](https://github.com/krtirtho/spotube/issues/1402)) ([9e25c74](https://github.com/krtirtho/spotube/commit/9e25c742d4e43e4e10d2b48afb8e6d90288ffa11)) +- add user profile page ([39e97ee](https://github.com/krtirtho/spotube/commit/39e97eef34d87348a264843e145f31f82832d12e)) +- **android:** Filter Device To Force High Frame Rate ([#880](https://github.com/krtirtho/spotube/issues/880)) ([6e41b10](https://github.com/krtirtho/spotube/commit/6e41b106fa989adee393d3ce2535e75446ad3eea)) +- improved caching based on riverpod ([#1343](https://github.com/krtirtho/spotube/issues/1343)) ([6673e5a](https://github.com/krtirtho/spotube/commit/6673e5a8a86b9667cf9dbff9bb7c40ea6b7de771)) +- LAN connect a.k.a control remote Spotube playback and local output device selection ([#1355](https://github.com/krtirtho/spotube/issues/1355)) ([68374ef](https://github.com/krtirtho/spotube/commit/68374efd3ec556f31b937e5b96920787b54eec78)) +- **lyrics:** add LRCLIB lyrics provider as fallback ([5afe823](https://github.com/krtirtho/spotube/commit/5afe823abdb198340b55d138d8173d886a811632)) +- search history support [#1236](https://github.com/krtirtho/spotube/issues/1236) ([82b1cfa](https://github.com/krtirtho/spotube/commit/82b1cfa0d775e3958c666280943a893c9113d468)) +- **translations:** Add Czech translation ([#1401](https://github.com/krtirtho/spotube/issues/1401)) ([5a6b800](https://github.com/krtirtho/spotube/commit/5a6b80091259359bc38c4b91cd8cb496c4270fa4)) +- **translations:** add Thai Language ([#1319](https://github.com/krtirtho/spotube/issues/1319)) ([b70f250](https://github.com/krtirtho/spotube/commit/b70f250e8d5137fd990787ec9e3d058126cf14f3)), closes [#1310](https://github.com/krtirtho/spotube/issues/1310) [#1311](https://github.com/krtirtho/spotube/issues/1311) ### Bug Fixes -* instance of Artist bug [#1362](https://github.com/krtirtho/spotube/issues/1362) ([c8dd802](https://github.com/krtirtho/spotube/commit/c8dd8025ec96bd78ed77cae35f1429aa48c16fde)) -* **playback:** sponsor block skips and stutters in same position ([0d080b7](https://github.com/krtirtho/spotube/commit/0d080b77b72529c0be5ebc27ace1c52307511f73)) +- instance of Artist bug [#1362](https://github.com/krtirtho/spotube/issues/1362) ([c8dd802](https://github.com/krtirtho/spotube/commit/c8dd8025ec96bd78ed77cae35f1429aa48c16fde)) +- **playback:** sponsor block skips and stutters in same position ([0d080b7](https://github.com/krtirtho/spotube/commit/0d080b77b72529c0be5ebc27ace1c52307511f73)) ## [3.5.0](https://github.com/krtirtho/spotube/compare/v3.4.1...v3.5.0) (2024-03-08) - ### Features -* add endless playback support [#285](https://github.com/krtirtho/spotube/issues/285) ([9dfd49c](https://github.com/krtirtho/spotube/commit/9dfd49ca04f0e915e333e205b17ac70456873f6e)) -* add getting started page ([96a2a1f](https://github.com/krtirtho/spotube/commit/96a2a1f5a622cb3c580041417d5023e37fa69716)) -* Add iOS background play support ([#1166](https://github.com/krtirtho/spotube/issues/1166)) ([095587e](https://github.com/krtirtho/spotube/commit/095587ee84f7d867c69fcf4b09ed608d63478e1e)) -* add songlink based track matching for youtube and open song link button ([9095a8c](https://github.com/krtirtho/spotube/commit/9095a8c8f849e42daabb7efcc20085cfb863c974)) -* **playlist:** show confirmation before deleting user playlist [#1222](https://github.com/krtirtho/spotube/issues/1222) ([9f92440](https://github.com/krtirtho/spotube/commit/9f9244062a39759aa0ce28d2d5f7c8fa53d73003)) -* Sort by Duration ([#1238](https://github.com/krtirtho/spotube/issues/1238)) ([6f8271f](https://github.com/krtirtho/spotube/commit/6f8271f5e9394cb4053e41dd222aa2844c34d609)) -* start radio support ([4defeef](https://github.com/krtirtho/spotube/commit/4defeefe7e5947aa00a2afb2a06577ec141cdc52)) -* **translations:** add Korean translation ([#1275](https://github.com/krtirtho/spotube/issues/1275)) ([fdea930](https://github.com/krtirtho/spotube/commit/fdea9307bbfb8f3f62cfb795bfb3ca58c38c33d9)) -* **translations:** Added Vietnamese ([#1135](https://github.com/krtirtho/spotube/issues/1135)) ([019ba86](https://github.com/krtirtho/spotube/commit/019ba865e20a8b54ea3490c01e47158eaf3a4c8d)) -* **windows:** Install Visual C++ 2015-2022 Redistributable if missing when installing ([ba69496](https://github.com/krtirtho/spotube/commit/ba69496dcc9a1b7f6ea4e104e71764a854d27f1f)) - +- add endless playback support [#285](https://github.com/krtirtho/spotube/issues/285) ([9dfd49c](https://github.com/krtirtho/spotube/commit/9dfd49ca04f0e915e333e205b17ac70456873f6e)) +- add getting started page ([96a2a1f](https://github.com/krtirtho/spotube/commit/96a2a1f5a622cb3c580041417d5023e37fa69716)) +- Add iOS background play support ([#1166](https://github.com/krtirtho/spotube/issues/1166)) ([095587e](https://github.com/krtirtho/spotube/commit/095587ee84f7d867c69fcf4b09ed608d63478e1e)) +- add songlink based track matching for youtube and open song link button ([9095a8c](https://github.com/krtirtho/spotube/commit/9095a8c8f849e42daabb7efcc20085cfb863c974)) +- **playlist:** show confirmation before deleting user playlist [#1222](https://github.com/krtirtho/spotube/issues/1222) ([9f92440](https://github.com/krtirtho/spotube/commit/9f9244062a39759aa0ce28d2d5f7c8fa53d73003)) +- Sort by Duration ([#1238](https://github.com/krtirtho/spotube/issues/1238)) ([6f8271f](https://github.com/krtirtho/spotube/commit/6f8271f5e9394cb4053e41dd222aa2844c34d609)) +- start radio support ([4defeef](https://github.com/krtirtho/spotube/commit/4defeefe7e5947aa00a2afb2a06577ec141cdc52)) +- **translations:** add Korean translation ([#1275](https://github.com/krtirtho/spotube/issues/1275)) ([fdea930](https://github.com/krtirtho/spotube/commit/fdea9307bbfb8f3f62cfb795bfb3ca58c38c33d9)) +- **translations:** Added Vietnamese ([#1135](https://github.com/krtirtho/spotube/issues/1135)) ([019ba86](https://github.com/krtirtho/spotube/commit/019ba865e20a8b54ea3490c01e47158eaf3a4c8d)) +- **windows:** Install Visual C++ 2015-2022 Redistributable if missing when installing ([ba69496](https://github.com/krtirtho/spotube/commit/ba69496dcc9a1b7f6ea4e104e71764a854d27f1f)) ### Bug Fixes -* album images are small in certain places ([ca76a39](https://github.com/krtirtho/spotube/commit/ca76a39910b1a5af91aa7882a0d33c9d71db58a2)) -* album, artist page not loading [#1282](https://github.com/krtirtho/spotube/issues/1282) ([a9a1d4c](https://github.com/krtirtho/spotube/commit/a9a1d4c9dc24aaf3181dc4090d1822ebfe755991)) -* **android:** audio issue when screen is off and broadcast audio session id ([#1221](https://github.com/krtirtho/spotube/issues/1221) & [#1247](https://github.com/krtirtho/spotube/issues/1247)) ([17105a6](https://github.com/krtirtho/spotube/commit/17105a640bf5107bd5d333b9b4d097c14a3949a2)), closes [KRTirtho/spotube#571](https://github.com/KRTirtho/spotube/issues/571) -* **android:** only ask battery optimization once [#1252](https://github.com/krtirtho/spotube/issues/1252) ([e516afb](https://github.com/krtirtho/spotube/commit/e516afb185f616471822ea745495a3d1d1281bd3)) -* **android:** pressing back button in any other tab other than home exits the app ([c3289a0](https://github.com/krtirtho/spotube/commit/c3289a0ba4e7de094a15246677ffcb940504ebde)) -* **android:** system back button in player page exits the app ([3294f65](https://github.com/krtirtho/spotube/commit/3294f657fe8a03b18d9be8974968b6508465963d)) -* cleanTitle removing feat and ft from words instead of whole words ([8612345](https://github.com/krtirtho/spotube/commit/86123456f2ff577921cf62cffca180427dfe1dd5)) -* friends list not scrollable with mouse drag ([ab08c82](https://github.com/krtirtho/spotube/commit/ab08c82c8dd501263049f3adcbd48907ba13e3a9)) -* no draggable scrollbar in playlist/album page [#1158](https://github.com/krtirtho/spotube/issues/1158) ([6f71e52](https://github.com/krtirtho/spotube/commit/6f71e52ea8a5712d2c3527f2a524af9fbb718bef)) -* non-banger songs breaking the queue if sources not found ([90f7c53](https://github.com/krtirtho/spotube/commit/90f7c531cdc8640afdbabf5a0592159715ea1e6f)) -* track loading when not found in Youtube ([e964f61](https://github.com/krtirtho/spotube/commit/e964f61d38cb303e3d3fd60c866414f57207181c)) -* **translations:** Update app_nl.arb ([#1168](https://github.com/krtirtho/spotube/issues/1168)) ([8167963](https://github.com/krtirtho/spotube/commit/8167963212eeb5dfb0b4fb2eadf81d466659a9f1)) +- album images are small in certain places ([ca76a39](https://github.com/krtirtho/spotube/commit/ca76a39910b1a5af91aa7882a0d33c9d71db58a2)) +- album, artist page not loading [#1282](https://github.com/krtirtho/spotube/issues/1282) ([a9a1d4c](https://github.com/krtirtho/spotube/commit/a9a1d4c9dc24aaf3181dc4090d1822ebfe755991)) +- **android:** audio issue when screen is off and broadcast audio session id ([#1221](https://github.com/krtirtho/spotube/issues/1221) & [#1247](https://github.com/krtirtho/spotube/issues/1247)) ([17105a6](https://github.com/krtirtho/spotube/commit/17105a640bf5107bd5d333b9b4d097c14a3949a2)), closes [KRTirtho/spotube#571](https://github.com/KRTirtho/spotube/issues/571) +- **android:** only ask battery optimization once [#1252](https://github.com/krtirtho/spotube/issues/1252) ([e516afb](https://github.com/krtirtho/spotube/commit/e516afb185f616471822ea745495a3d1d1281bd3)) +- **android:** pressing back button in any other tab other than home exits the app ([c3289a0](https://github.com/krtirtho/spotube/commit/c3289a0ba4e7de094a15246677ffcb940504ebde)) +- **android:** system back button in player page exits the app ([3294f65](https://github.com/krtirtho/spotube/commit/3294f657fe8a03b18d9be8974968b6508465963d)) +- cleanTitle removing feat and ft from words instead of whole words ([8612345](https://github.com/krtirtho/spotube/commit/86123456f2ff577921cf62cffca180427dfe1dd5)) +- friends list not scrollable with mouse drag ([ab08c82](https://github.com/krtirtho/spotube/commit/ab08c82c8dd501263049f3adcbd48907ba13e3a9)) +- no draggable scrollbar in playlist/album page [#1158](https://github.com/krtirtho/spotube/issues/1158) ([6f71e52](https://github.com/krtirtho/spotube/commit/6f71e52ea8a5712d2c3527f2a524af9fbb718bef)) +- non-banger songs breaking the queue if sources not found ([90f7c53](https://github.com/krtirtho/spotube/commit/90f7c531cdc8640afdbabf5a0592159715ea1e6f)) +- track loading when not found in Youtube ([e964f61](https://github.com/krtirtho/spotube/commit/e964f61d38cb303e3d3fd60c866414f57207181c)) +- **translations:** Update app_nl.arb ([#1168](https://github.com/krtirtho/spotube/issues/1168)) ([8167963](https://github.com/krtirtho/spotube/commit/8167963212eeb5dfb0b4fb2eadf81d466659a9f1)) ## [3.4.1](https://personal.github.com/krtirtho/spotube/compare/v3.4.0...v3.4.1) (2024-01-27) - ### Features -* add create playlist button in add playlist dialog ([2168a64](https://personal.github.com/krtirtho/spotube/commit/2168a640af3104a43139c303d78e2c2326a1bda7)) -* add spotify friends activity ([#1130](https://personal.github.com/krtirtho/spotube/issues/1130)) ([7983932](https://personal.github.com/krtirtho/spotube/commit/79839329b0970acccb0c566a31eee508adbc8557)) -* **deep-link:** add track opening page ([988a975](https://personal.github.com/krtirtho/spotube/commit/988a975bf1a675df0cfc7b17776bcec74c67f1f2)) -* haptic feedback on long press and reordering actions ([6242200](https://personal.github.com/krtirtho/spotube/commit/624220090572eb643dce37ca8ffd85d2b3f5c9df)) -* improve youtube/piped matching by suffixing "- Topic" ([8184555](https://personal.github.com/krtirtho/spotube/commit/8184555ee89fd30aaf886af9fc1d52c142fdebb0)) -* **translations:** add Nepali (नेपाली) translations ([#1111](https://personal.github.com/krtirtho/spotube/issues/1111)) ([c3ebf56](https://personal.github.com/krtirtho/spotube/commit/c3ebf56ac149b0af8815a5533fe6c386df743440)), closes [#1074](https://personal.github.com/krtirtho/spotube/issues/1074) [#1100](https://personal.github.com/krtirtho/spotube/issues/1100) - +- add create playlist button in add playlist dialog ([2168a64](https://personal.github.com/krtirtho/spotube/commit/2168a640af3104a43139c303d78e2c2326a1bda7)) +- add spotify friends activity ([#1130](https://personal.github.com/krtirtho/spotube/issues/1130)) ([7983932](https://personal.github.com/krtirtho/spotube/commit/79839329b0970acccb0c566a31eee508adbc8557)) +- **deep-link:** add track opening page ([988a975](https://personal.github.com/krtirtho/spotube/commit/988a975bf1a675df0cfc7b17776bcec74c67f1f2)) +- haptic feedback on long press and reordering actions ([6242200](https://personal.github.com/krtirtho/spotube/commit/624220090572eb643dce37ca8ffd85d2b3f5c9df)) +- improve youtube/piped matching by suffixing "- Topic" ([8184555](https://personal.github.com/krtirtho/spotube/commit/8184555ee89fd30aaf886af9fc1d52c142fdebb0)) +- **translations:** add Nepali (नेपाली) translations ([#1111](https://personal.github.com/krtirtho/spotube/issues/1111)) ([c3ebf56](https://personal.github.com/krtirtho/spotube/commit/c3ebf56ac149b0af8815a5533fe6c386df743440)), closes [#1074](https://personal.github.com/krtirtho/spotube/issues/1074) [#1100](https://personal.github.com/krtirtho/spotube/issues/1100) ### Bug Fixes -* alternative searched sources doesn't play [#1059](https://personal.github.com/krtirtho/spotube/issues/1059) ([a8e9b82](https://personal.github.com/krtirtho/spotube/commit/a8e9b824f33add8f6a83f0d147e889eb6beeb442)) -* alternative source doesn't persist on next restart [#840](https://personal.github.com/krtirtho/spotube/issues/840) ([62fde50](https://personal.github.com/krtirtho/spotube/commit/62fde50442f04f93255b5b1b1dcca23d116a13ec)) -* **android:** download failing for permission issues [#1015](https://personal.github.com/krtirtho/spotube/issues/1015) ([5509cae](https://personal.github.com/krtirtho/spotube/commit/5509cae91c8b1f5cb9fac179060f477397a4a27f)) -* artist page error [#1018](https://personal.github.com/krtirtho/spotube/issues/1018) ([8cd650b](https://personal.github.com/krtirtho/spotube/commit/8cd650b07e5f4c4c2f296bf4374e5ee67fb3eb50)) -* audio resumes after a phone call even if it was paused before [#926](https://personal.github.com/krtirtho/spotube/issues/926) ([fd1899f](https://personal.github.com/krtirtho/spotube/commit/fd1899f162395752142d7aa7320d1c39b0995070)) -* better error message for failing to find lyrics [#1085](https://personal.github.com/krtirtho/spotube/issues/1085) ([e58e18d](https://personal.github.com/krtirtho/spotube/commit/e58e18de33d7bc6fb0e4ddd7ccf6ea14472642b1)) -* Black window flash when starting the app ([#1003](https://personal.github.com/krtirtho/spotube/issues/1003)) ([02e44fc](https://personal.github.com/krtirtho/spotube/commit/02e44fc6b849a873adad382f5d46ed8caf32359f)) -* **linux:** crash after login ([0dfd401](https://personal.github.com/krtirtho/spotube/commit/0dfd40153714b7a4b83ac30f0c56830bc0c05ffd)) -* **macos:** backbutton and window button overlap and unused empty space on home ([b9417ca](https://personal.github.com/krtirtho/spotube/commit/b9417ca3575992673357230dab49e0124dd576b1)) -* **macos:** download folder unchangeable ([9d74cf5](https://personal.github.com/krtirtho/spotube/commit/9d74cf5fc250a6a143321d49b8e045519b4c2872)) -* **macos:** Respect Minimize to tray option ([#1001](https://personal.github.com/krtirtho/spotube/issues/1001)) ([69559ba](https://personal.github.com/krtirtho/spotube/commit/69559ba24285636e42b2f2231f956c31388c5cf3)) -* **macos:** system tray shows name and sidebar weird gap [#1083](https://personal.github.com/krtirtho/spotube/issues/1083) ([27057ea](https://personal.github.com/krtirtho/spotube/commit/27057ea0c8d83c9701057c18b473f1af4e4e82be)) -* releases section is empty when user doesn't follow any artists [#1104](https://personal.github.com/krtirtho/spotube/issues/1104) ([682e88e](https://personal.github.com/krtirtho/spotube/commit/682e88e0c55bc0f4708bc0b4681b129e5c61c999)) -* search page vertical scrollbar moves on horizontal scroll [#1017](https://personal.github.com/krtirtho/spotube/issues/1017) ([c203ac6](https://personal.github.com/krtirtho/spotube/commit/c203ac69ee74ba8722dae3da4b47761cd8d59c34)) -* songs doesn't play when sources with preferred audio codec is empty ([#976](https://personal.github.com/krtirtho/spotube/issues/976)) ([ba4e11a](https://personal.github.com/krtirtho/spotube/commit/ba4e11a40ab18308437a05333a46eace6f8eeb5a)) -* track index not showing after 200 ([a752cf4](https://personal.github.com/krtirtho/spotube/commit/a752cf4c978d1b05851aabb6c84c7862de551320)) -* track pad horizontal scrolling not working ([59e0e6b](https://personal.github.com/krtirtho/spotube/commit/59e0e6bb659b70831f6e0ae064100381c57f149c)) +- alternative searched sources doesn't play [#1059](https://personal.github.com/krtirtho/spotube/issues/1059) ([a8e9b82](https://personal.github.com/krtirtho/spotube/commit/a8e9b824f33add8f6a83f0d147e889eb6beeb442)) +- alternative source doesn't persist on next restart [#840](https://personal.github.com/krtirtho/spotube/issues/840) ([62fde50](https://personal.github.com/krtirtho/spotube/commit/62fde50442f04f93255b5b1b1dcca23d116a13ec)) +- **android:** download failing for permission issues [#1015](https://personal.github.com/krtirtho/spotube/issues/1015) ([5509cae](https://personal.github.com/krtirtho/spotube/commit/5509cae91c8b1f5cb9fac179060f477397a4a27f)) +- artist page error [#1018](https://personal.github.com/krtirtho/spotube/issues/1018) ([8cd650b](https://personal.github.com/krtirtho/spotube/commit/8cd650b07e5f4c4c2f296bf4374e5ee67fb3eb50)) +- audio resumes after a phone call even if it was paused before [#926](https://personal.github.com/krtirtho/spotube/issues/926) ([fd1899f](https://personal.github.com/krtirtho/spotube/commit/fd1899f162395752142d7aa7320d1c39b0995070)) +- better error message for failing to find lyrics [#1085](https://personal.github.com/krtirtho/spotube/issues/1085) ([e58e18d](https://personal.github.com/krtirtho/spotube/commit/e58e18de33d7bc6fb0e4ddd7ccf6ea14472642b1)) +- Black window flash when starting the app ([#1003](https://personal.github.com/krtirtho/spotube/issues/1003)) ([02e44fc](https://personal.github.com/krtirtho/spotube/commit/02e44fc6b849a873adad382f5d46ed8caf32359f)) +- **linux:** crash after login ([0dfd401](https://personal.github.com/krtirtho/spotube/commit/0dfd40153714b7a4b83ac30f0c56830bc0c05ffd)) +- **macos:** backbutton and window button overlap and unused empty space on home ([b9417ca](https://personal.github.com/krtirtho/spotube/commit/b9417ca3575992673357230dab49e0124dd576b1)) +- **macos:** download folder unchangeable ([9d74cf5](https://personal.github.com/krtirtho/spotube/commit/9d74cf5fc250a6a143321d49b8e045519b4c2872)) +- **macos:** Respect Minimize to tray option ([#1001](https://personal.github.com/krtirtho/spotube/issues/1001)) ([69559ba](https://personal.github.com/krtirtho/spotube/commit/69559ba24285636e42b2f2231f956c31388c5cf3)) +- **macos:** system tray shows name and sidebar weird gap [#1083](https://personal.github.com/krtirtho/spotube/issues/1083) ([27057ea](https://personal.github.com/krtirtho/spotube/commit/27057ea0c8d83c9701057c18b473f1af4e4e82be)) +- releases section is empty when user doesn't follow any artists [#1104](https://personal.github.com/krtirtho/spotube/issues/1104) ([682e88e](https://personal.github.com/krtirtho/spotube/commit/682e88e0c55bc0f4708bc0b4681b129e5c61c999)) +- search page vertical scrollbar moves on horizontal scroll [#1017](https://personal.github.com/krtirtho/spotube/issues/1017) ([c203ac6](https://personal.github.com/krtirtho/spotube/commit/c203ac69ee74ba8722dae3da4b47761cd8d59c34)) +- songs doesn't play when sources with preferred audio codec is empty ([#976](https://personal.github.com/krtirtho/spotube/issues/976)) ([ba4e11a](https://personal.github.com/krtirtho/spotube/commit/ba4e11a40ab18308437a05333a46eace6f8eeb5a)) +- track index not showing after 200 ([a752cf4](https://personal.github.com/krtirtho/spotube/commit/a752cf4c978d1b05851aabb6c84c7862de551320)) +- track pad horizontal scrolling not working ([59e0e6b](https://personal.github.com/krtirtho/spotube/commit/59e0e6bb659b70831f6e0ae064100381c57f149c)) ## [3.4.0](https://github.com/KRTirtho/spotube/compare/v3.3.0...v3.4.0) (2023-12-30) - ### Features -* Add Go to Album option in track option [#917](https://github.com/KRTirtho/spotube/issues/917) ([b0beeca](https://github.com/KRTirtho/spotube/commit/b0beeca0cbaf810fae27832cff98cfda95715050)) -* **translations:** add Italian language translations ([#818](https://github.com/KRTirtho/spotube/issues/818)) ([e4eb0e2](https://github.com/KRTirtho/spotube/commit/e4eb0e2596ade2bb5195e183f03af42742fc8486)), closes [#676](https://github.com/KRTirtho/spotube/issues/676) [#676](https://github.com/KRTirtho/spotube/issues/676) -* compact genre view in home page ([82ed5e9](https://github.com/KRTirtho/spotube/commit/82ed5e90576b57ef32e61a65015e04862ab15461)) -* Deep link support ([#950](https://github.com/KRTirtho/spotube/issues/950)) ([4050f55](https://github.com/KRTirtho/spotube/commit/4050f556400aaec5515231578512cf1a6b990110)) -* improve loading animations ([b92583d](https://github.com/KRTirtho/spotube/commit/b92583d0df7b8dee0d121cd2bb666b14c77d8c86)) -* toggle for discord rpc ([24a2294](https://github.com/KRTirtho/spotube/commit/24a2294512bb0c4aff77bc8dcad9b4de3e8b45c6)) -* **translations:** add Dutch Language ([#969](https://github.com/KRTirtho/spotube/issues/969)) ([3ad7ba6](https://github.com/KRTirtho/spotube/commit/3ad7ba66b56e93e69d2181d47029b7549ed225fc)) - +- Add Go to Album option in track option [#917](https://github.com/KRTirtho/spotube/issues/917) ([b0beeca](https://github.com/KRTirtho/spotube/commit/b0beeca0cbaf810fae27832cff98cfda95715050)) +- **translations:** add Italian language translations ([#818](https://github.com/KRTirtho/spotube/issues/818)) ([e4eb0e2](https://github.com/KRTirtho/spotube/commit/e4eb0e2596ade2bb5195e183f03af42742fc8486)), closes [#676](https://github.com/KRTirtho/spotube/issues/676) [#676](https://github.com/KRTirtho/spotube/issues/676) +- compact genre view in home page ([82ed5e9](https://github.com/KRTirtho/spotube/commit/82ed5e90576b57ef32e61a65015e04862ab15461)) +- Deep link support ([#950](https://github.com/KRTirtho/spotube/issues/950)) ([4050f55](https://github.com/KRTirtho/spotube/commit/4050f556400aaec5515231578512cf1a6b990110)) +- improve loading animations ([b92583d](https://github.com/KRTirtho/spotube/commit/b92583d0df7b8dee0d121cd2bb666b14c77d8c86)) +- toggle for discord rpc ([24a2294](https://github.com/KRTirtho/spotube/commit/24a2294512bb0c4aff77bc8dcad9b4de3e8b45c6)) +- **translations:** add Dutch Language ([#969](https://github.com/KRTirtho/spotube/issues/969)) ([3ad7ba6](https://github.com/KRTirtho/spotube/commit/3ad7ba66b56e93e69d2181d47029b7549ed225fc)) ### Bug Fixes -* add safe area in home ([9ee6067](https://github.com/KRTirtho/spotube/commit/9ee60677f6d50df7468e12dc6653ecedefa2494f)) -* amoled mode and color scheme can't be changed ([840e014](https://github.com/KRTirtho/spotube/commit/840e014f2b18f193d040baef0e0cd595088a4a84)) -* doesn't minimize to tray when system title bar close button is used [#866](https://github.com/KRTirtho/spotube/issues/866) ([bb8f250](https://github.com/KRTirtho/spotube/commit/bb8f250f5f351c1a353791b77b25b9de7586191f)) -* genre border issues ([2fb16e6](https://github.com/KRTirtho/spotube/commit/2fb16e64e9cdfca54d633cdf287b0544ecdda3b6)) -* Incorrect "Artist" label/heading on Search Results Page [#920](https://github.com/KRTirtho/spotube/issues/920) ([f86d544](https://github.com/KRTirtho/spotube/commit/f86d5449168068e338f769d7f504d2146b86dc79)) -* metadata not getting added for YouTube tracks [#916](https://github.com/KRTirtho/spotube/issues/916) and Wrong duration of downloaded tracks [#912](https://github.com/KRTirtho/spotube/issues/912) ([a7b9398](https://github.com/KRTirtho/spotube/commit/a7b9398708ede865dc2c25fb791c8e98eeff7a38)) -* Playlist refresh not working [#915](https://github.com/KRTirtho/spotube/issues/915) ([5f1df5a](https://github.com/KRTirtho/spotube/commit/5f1df5a87d8fb7980b52cf57b7b6bedea57a1269)) -* track view header title overflow and player view drag glitch ([b04d884](https://github.com/KRTirtho/spotube/commit/b04d8849e7169824ec5b980236b5d61b2629f56e)) -* wrong artist name sent while scrobbling [#958](https://github.com/KRTirtho/spotube/issues/958) ([dcbe729](https://github.com/KRTirtho/spotube/commit/dcbe7294b742d43fbff4e89ab4c4825e94421dd9)) +- add safe area in home ([9ee6067](https://github.com/KRTirtho/spotube/commit/9ee60677f6d50df7468e12dc6653ecedefa2494f)) +- amoled mode and color scheme can't be changed ([840e014](https://github.com/KRTirtho/spotube/commit/840e014f2b18f193d040baef0e0cd595088a4a84)) +- doesn't minimize to tray when system title bar close button is used [#866](https://github.com/KRTirtho/spotube/issues/866) ([bb8f250](https://github.com/KRTirtho/spotube/commit/bb8f250f5f351c1a353791b77b25b9de7586191f)) +- genre border issues ([2fb16e6](https://github.com/KRTirtho/spotube/commit/2fb16e64e9cdfca54d633cdf287b0544ecdda3b6)) +- Incorrect "Artist" label/heading on Search Results Page [#920](https://github.com/KRTirtho/spotube/issues/920) ([f86d544](https://github.com/KRTirtho/spotube/commit/f86d5449168068e338f769d7f504d2146b86dc79)) +- metadata not getting added for YouTube tracks [#916](https://github.com/KRTirtho/spotube/issues/916) and Wrong duration of downloaded tracks [#912](https://github.com/KRTirtho/spotube/issues/912) ([a7b9398](https://github.com/KRTirtho/spotube/commit/a7b9398708ede865dc2c25fb791c8e98eeff7a38)) +- Playlist refresh not working [#915](https://github.com/KRTirtho/spotube/issues/915) ([5f1df5a](https://github.com/KRTirtho/spotube/commit/5f1df5a87d8fb7980b52cf57b7b6bedea57a1269)) +- track view header title overflow and player view drag glitch ([b04d884](https://github.com/KRTirtho/spotube/commit/b04d8849e7169824ec5b980236b5d61b2629f56e)) +- wrong artist name sent while scrobbling [#958](https://github.com/KRTirtho/spotube/issues/958) ([dcbe729](https://github.com/KRTirtho/spotube/commit/dcbe7294b742d43fbff4e89ab4c4825e94421dd9)) ## [3.3.0](https://github.com/KRTirtho/spotube/compare/v3.2.0...v3.3.0) (2023-11-27) - ### Features -* Add JioSaavn as audio source ([#881](https://github.com/KRTirtho/spotube/issues/881)) ([14069cd](https://github.com/KRTirtho/spotube/commit/14069cd4fe08597c8d9aa0810270fb4c386c1d55)) -* **android:** better quick scroll/drag to scroll implementation ([2e2c44f](https://github.com/KRTirtho/spotube/commit/2e2c44f0afef69bf9bc485db97d45127a0847c8e)) -* **artist:** modularize page and add wikipedia section ([2a69886](https://github.com/KRTirtho/spotube/commit/2a698865567883271471ace9a44123bbfd8fcd2f)) -* discord RPC integration [#98](https://github.com/KRTirtho/spotube/issues/98) ([88b8785](https://github.com/KRTirtho/spotube/commit/88b8785cb86a19900f3a867b044c1ccb2fe400bb)) -* **mini_player:** show/hide lyrics [#851](https://github.com/KRTirtho/spotube/issues/851) ([dcbb156](https://github.com/KRTirtho/spotube/commit/dcbb1568337969841acc0abe0e7185ee5e4c3590)) -* paginated playlist and album page ([28a5d6b](https://github.com/KRTirtho/spotube/commit/28a5d6bb3820ab0bd4007664f73d685f6e1d2c90)) -* **translations:** add Turkish translations ([0c22469](https://github.com/KRTirtho/spotube/commit/0c22469503f32dbbf1a5d31419c1b76c699fa966)) - +- Add JioSaavn as audio source ([#881](https://github.com/KRTirtho/spotube/issues/881)) ([14069cd](https://github.com/KRTirtho/spotube/commit/14069cd4fe08597c8d9aa0810270fb4c386c1d55)) +- **android:** better quick scroll/drag to scroll implementation ([2e2c44f](https://github.com/KRTirtho/spotube/commit/2e2c44f0afef69bf9bc485db97d45127a0847c8e)) +- **artist:** modularize page and add wikipedia section ([2a69886](https://github.com/KRTirtho/spotube/commit/2a698865567883271471ace9a44123bbfd8fcd2f)) +- discord RPC integration [#98](https://github.com/KRTirtho/spotube/issues/98) ([88b8785](https://github.com/KRTirtho/spotube/commit/88b8785cb86a19900f3a867b044c1ccb2fe400bb)) +- **mini_player:** show/hide lyrics [#851](https://github.com/KRTirtho/spotube/issues/851) ([dcbb156](https://github.com/KRTirtho/spotube/commit/dcbb1568337969841acc0abe0e7185ee5e4c3590)) +- paginated playlist and album page ([28a5d6b](https://github.com/KRTirtho/spotube/commit/28a5d6bb3820ab0bd4007664f73d685f6e1d2c90)) +- **translations:** add Turkish translations ([0c22469](https://github.com/KRTirtho/spotube/commit/0c22469503f32dbbf1a5d31419c1b76c699fa966)) ### Bug Fixes -* "Add () to Playlist" option not showing in favorited playlists [#904](https://github.com/KRTirtho/spotube/issues/904) ([96021e1](https://github.com/KRTirtho/spotube/commit/96021e1a49d22bd25fd052c122f49f439c2bea43)) -* 0:00 media duration in queue after application restart [#782](https://github.com/KRTirtho/spotube/issues/782) ([83c0b49](https://github.com/KRTirtho/spotube/commit/83c0b49da962d9f3d40de9525f90f0b320e8f7b8)) -* Add to Playlist Dialog memory leak [#817](https://github.com/KRTirtho/spotube/issues/817) ([fed36ec](https://github.com/KRTirtho/spotube/commit/fed36ecdd81e8a0f8358693eff0a6233dea32e5d)) -* **album_card:** show loading state during adding track to queue/play ([5633367](https://github.com/KRTirtho/spotube/commit/5633367397812148f6d712d06e97a4f84033f968)) -* alternative track source safearea overflow [#876](https://github.com/KRTirtho/spotube/issues/876) ([7b72a90](https://github.com/KRTirtho/spotube/commit/7b72a90bc65b541cbe2e24ef2234524b522ad71d)) -* android invalid download location Download not starting or not explaining error [#720](https://github.com/KRTirtho/spotube/issues/720) ([d056dbf](https://github.com/KRTirtho/spotube/commit/d056dbf9eeef7033dbc012d0c05800063e820042)) -* changed settings are not persisting after force stop [#821](https://github.com/KRTirtho/spotube/issues/821) ([e29a38d](https://github.com/KRTirtho/spotube/commit/e29a38dfa43ddf7a38046d1d40424f01dbe62261)) -* check for unsynced lyrics and error handling for timed lyrics query ([1d77556](https://github.com/KRTirtho/spotube/commit/1d77556157d158600f29cf2ea5f26c567607dec7)) -* **genres:** lag while scrolling ([dc980b0](https://github.com/KRTirtho/spotube/commit/dc980b024edad3132e72cbb2f0087297a4b76469)) -* infinite list disappearing for a moment everytime new page is fetched ([1334a62](https://github.com/KRTirtho/spotube/commit/1334a62aaea31f97031b3ebf455e94c583f37314)) -* last track of queue keeps repeating [#718](https://github.com/KRTirtho/spotube/issues/718) ([58e5698](https://github.com/KRTirtho/spotube/commit/58e569864dddd74c3064624998dfc184046e97eb)) -* Navigating to settings, redirects to home page [#812](https://github.com/KRTirtho/spotube/issues/812) ([da04f06](https://github.com/KRTirtho/spotube/commit/da04f068f9b7effff8d50cb5714d93ea80c22b7f)) -* new releases section flickering on scroll glitch ([ee94b7c](https://github.com/KRTirtho/spotube/commit/ee94b7cbb24e0f0bc22a6d49c830d4055aa02895)) -* **playbutton_card:** annoying animation ([574406d](https://github.com/KRTirtho/spotube/commit/574406dd5fc410914b27e7fce374323696845012)) -* scrobbling not working for first track or single track ([0a6b54d](https://github.com/KRTirtho/spotube/commit/0a6b54da367345b73fe6e954f1d9368d9f9ead71)) -* settings page scrollbar position ([ee82290](https://github.com/KRTirtho/spotube/commit/ee8229020b3b03fc074b316db4b322af13b807bd)) -* shuffle doesn't move active track to top ([4956bf3](https://github.com/KRTirtho/spotube/commit/4956bf367baae39c88b5de7c6c136513a14f8ad2)) -* spotube doesn't exit properly, hangs in infinite loop [#768](https://github.com/KRTirtho/spotube/issues/768) ([353ca79](https://github.com/KRTirtho/spotube/commit/353ca79be334077c3ac27b4f64e8b4b15eca7175)) -* trim login field padding ([286ef83](https://github.com/KRTirtho/spotube/commit/286ef83e8ec516db70019398d9e3e724437a4172)) -* use CustomScrollView for personalized page ([7d05c40](https://github.com/KRTirtho/spotube/commit/7d05c40dc0d04208b059f2483c1e4de199c8b51d)) -* user_playlists layout, track tile index, ([487c2ed](https://github.com/KRTirtho/spotube/commit/487c2ed6bdc4af33006ba52532eb4eaaa261dceb)) -* **windows:** media control not working [#641](https://github.com/KRTirtho/spotube/issues/641) ([7818574](https://github.com/KRTirtho/spotube/commit/7818574356d0fb8ff567e1f6a83fd0b6f2ee7c8a)) +- "Add () to Playlist" option not showing in favorited playlists [#904](https://github.com/KRTirtho/spotube/issues/904) ([96021e1](https://github.com/KRTirtho/spotube/commit/96021e1a49d22bd25fd052c122f49f439c2bea43)) +- 0:00 media duration in queue after application restart [#782](https://github.com/KRTirtho/spotube/issues/782) ([83c0b49](https://github.com/KRTirtho/spotube/commit/83c0b49da962d9f3d40de9525f90f0b320e8f7b8)) +- Add to Playlist Dialog memory leak [#817](https://github.com/KRTirtho/spotube/issues/817) ([fed36ec](https://github.com/KRTirtho/spotube/commit/fed36ecdd81e8a0f8358693eff0a6233dea32e5d)) +- **album_card:** show loading state during adding track to queue/play ([5633367](https://github.com/KRTirtho/spotube/commit/5633367397812148f6d712d06e97a4f84033f968)) +- alternative track source safearea overflow [#876](https://github.com/KRTirtho/spotube/issues/876) ([7b72a90](https://github.com/KRTirtho/spotube/commit/7b72a90bc65b541cbe2e24ef2234524b522ad71d)) +- android invalid download location Download not starting or not explaining error [#720](https://github.com/KRTirtho/spotube/issues/720) ([d056dbf](https://github.com/KRTirtho/spotube/commit/d056dbf9eeef7033dbc012d0c05800063e820042)) +- changed settings are not persisting after force stop [#821](https://github.com/KRTirtho/spotube/issues/821) ([e29a38d](https://github.com/KRTirtho/spotube/commit/e29a38dfa43ddf7a38046d1d40424f01dbe62261)) +- check for unsynced lyrics and error handling for timed lyrics query ([1d77556](https://github.com/KRTirtho/spotube/commit/1d77556157d158600f29cf2ea5f26c567607dec7)) +- **genres:** lag while scrolling ([dc980b0](https://github.com/KRTirtho/spotube/commit/dc980b024edad3132e72cbb2f0087297a4b76469)) +- infinite list disappearing for a moment everytime new page is fetched ([1334a62](https://github.com/KRTirtho/spotube/commit/1334a62aaea31f97031b3ebf455e94c583f37314)) +- last track of queue keeps repeating [#718](https://github.com/KRTirtho/spotube/issues/718) ([58e5698](https://github.com/KRTirtho/spotube/commit/58e569864dddd74c3064624998dfc184046e97eb)) +- Navigating to settings, redirects to home page [#812](https://github.com/KRTirtho/spotube/issues/812) ([da04f06](https://github.com/KRTirtho/spotube/commit/da04f068f9b7effff8d50cb5714d93ea80c22b7f)) +- new releases section flickering on scroll glitch ([ee94b7c](https://github.com/KRTirtho/spotube/commit/ee94b7cbb24e0f0bc22a6d49c830d4055aa02895)) +- **playbutton_card:** annoying animation ([574406d](https://github.com/KRTirtho/spotube/commit/574406dd5fc410914b27e7fce374323696845012)) +- scrobbling not working for first track or single track ([0a6b54d](https://github.com/KRTirtho/spotube/commit/0a6b54da367345b73fe6e954f1d9368d9f9ead71)) +- settings page scrollbar position ([ee82290](https://github.com/KRTirtho/spotube/commit/ee8229020b3b03fc074b316db4b322af13b807bd)) +- shuffle doesn't move active track to top ([4956bf3](https://github.com/KRTirtho/spotube/commit/4956bf367baae39c88b5de7c6c136513a14f8ad2)) +- spotube doesn't exit properly, hangs in infinite loop [#768](https://github.com/KRTirtho/spotube/issues/768) ([353ca79](https://github.com/KRTirtho/spotube/commit/353ca79be334077c3ac27b4f64e8b4b15eca7175)) +- trim login field padding ([286ef83](https://github.com/KRTirtho/spotube/commit/286ef83e8ec516db70019398d9e3e724437a4172)) +- use CustomScrollView for personalized page ([7d05c40](https://github.com/KRTirtho/spotube/commit/7d05c40dc0d04208b059f2483c1e4de199c8b51d)) +- user_playlists layout, track tile index, ([487c2ed](https://github.com/KRTirtho/spotube/commit/487c2ed6bdc4af33006ba52532eb4eaaa261dceb)) +- **windows:** media control not working [#641](https://github.com/KRTirtho/spotube/issues/641) ([7818574](https://github.com/KRTirtho/spotube/commit/7818574356d0fb8ff567e1f6a83fd0b6f2ee7c8a)) ## [3.2.0](https://github.com/KRTirtho/spotube/compare/v3.1.2...v3.2.0) (2023-10-16) - ### Features -* ability to select/copy lyrics [#802](https://github.com/KRTirtho/spotube/issues/802) ([0eb9ee8](https://github.com/KRTirtho/spotube/commit/0eb9ee8648bee43a8009e6752674b1be646c0916)) -* add Amoled theme [#724](https://github.com/KRTirtho/spotube/issues/724) ([5c5dbf6](https://github.com/KRTirtho/spotube/commit/5c5dbf69ecea95c92d3c3900ad690a500d75b4e2)) -* add audio normalization [#164](https://github.com/KRTirtho/spotube/issues/164) ([da10ab2](https://github.com/KRTirtho/spotube/commit/da10ab2e291d4ba4d3082b9a6ae535639fb8f1b7)) -* add restore default settings button ([94c3866](https://github.com/KRTirtho/spotube/commit/94c386638f2e5a42d21c8f157835443333ee6d5c)) -* configurable audio normalization switch ([c325911](https://github.com/KRTirtho/spotube/commit/c325911c0d87758a203a52df02179c1513bad3fd)) -* customizable stream/download file formats ([#757](https://github.com/KRTirtho/spotube/issues/757)) ([e54762b](https://github.com/KRTirtho/spotube/commit/e54762be6add6524ab614d103fc3557a101c75f4)) -* improve and unify the logging framework ([#738](https://github.com/KRTirtho/spotube/issues/738)) ([c7432bb](https://github.com/KRTirtho/spotube/commit/c7432bbd986d576a93957f0a22bdbca5c1e87f20)) -* LastFM scrobbling support ([#761](https://github.com/KRTirtho/spotube/issues/761)) ([f5bd907](https://github.com/KRTirtho/spotube/commit/f5bd90731d9abc19d684c8bcb231eff399e73023)) -* loading indicator for genre and personalized pages ([ffe8d9c](https://github.com/KRTirtho/spotube/commit/ffe8d9ca6da25cb3e6fd2c781d5ed3a7b919510e)) -* manual offline detection ([854ab89](https://github.com/KRTirtho/spotube/commit/854ab8910dffb2837c011d3439173a1f0ebe9c6c)) -* show error dialog on failed to login ([101c325](https://github.com/KRTirtho/spotube/commit/101c32523d3be8c05527261f6f63f939d388ad79)) -* sliding up player support ([083319f](https://github.com/KRTirtho/spotube/commit/083319fd2445ab179e3dcda0a6aeaca6f13dda29)) -* swipe to open player view ([#765](https://github.com/KRTirtho/spotube/issues/765)) ([9aee056](https://github.com/KRTirtho/spotube/commit/9aee0568bf42eed9fea8d517e960a010abf0ebf2)) -* thicken the scrollbars & make 'em interactive for mobile ([#764](https://github.com/KRTirtho/spotube/issues/764)) ([84a4bcd](https://github.com/KRTirtho/spotube/commit/84a4bcd948ab459489aaf6f39d6954776c3401d7)) -* **translations:** add Arabic Translations ([#740](https://github.com/KRTirtho/spotube/issues/740)) ([38493f9](https://github.com/KRTirtho/spotube/commit/38493f9dd75303890857a626c0b276ee1ab75bb2)) -* **translations:** add Farsi Translations ([#760](https://github.com/KRTirtho/spotube/issues/760)) ([fe42cfe](https://github.com/KRTirtho/spotube/commit/fe42cfe8430035d9b67dd158fb7b835ee4071497)) - +- ability to select/copy lyrics [#802](https://github.com/KRTirtho/spotube/issues/802) ([0eb9ee8](https://github.com/KRTirtho/spotube/commit/0eb9ee8648bee43a8009e6752674b1be646c0916)) +- add Amoled theme [#724](https://github.com/KRTirtho/spotube/issues/724) ([5c5dbf6](https://github.com/KRTirtho/spotube/commit/5c5dbf69ecea95c92d3c3900ad690a500d75b4e2)) +- add audio normalization [#164](https://github.com/KRTirtho/spotube/issues/164) ([da10ab2](https://github.com/KRTirtho/spotube/commit/da10ab2e291d4ba4d3082b9a6ae535639fb8f1b7)) +- add restore default settings button ([94c3866](https://github.com/KRTirtho/spotube/commit/94c386638f2e5a42d21c8f157835443333ee6d5c)) +- configurable audio normalization switch ([c325911](https://github.com/KRTirtho/spotube/commit/c325911c0d87758a203a52df02179c1513bad3fd)) +- customizable stream/download file formats ([#757](https://github.com/KRTirtho/spotube/issues/757)) ([e54762b](https://github.com/KRTirtho/spotube/commit/e54762be6add6524ab614d103fc3557a101c75f4)) +- improve and unify the logging framework ([#738](https://github.com/KRTirtho/spotube/issues/738)) ([c7432bb](https://github.com/KRTirtho/spotube/commit/c7432bbd986d576a93957f0a22bdbca5c1e87f20)) +- LastFM scrobbling support ([#761](https://github.com/KRTirtho/spotube/issues/761)) ([f5bd907](https://github.com/KRTirtho/spotube/commit/f5bd90731d9abc19d684c8bcb231eff399e73023)) +- loading indicator for genre and personalized pages ([ffe8d9c](https://github.com/KRTirtho/spotube/commit/ffe8d9ca6da25cb3e6fd2c781d5ed3a7b919510e)) +- manual offline detection ([854ab89](https://github.com/KRTirtho/spotube/commit/854ab8910dffb2837c011d3439173a1f0ebe9c6c)) +- show error dialog on failed to login ([101c325](https://github.com/KRTirtho/spotube/commit/101c32523d3be8c05527261f6f63f939d388ad79)) +- sliding up player support ([083319f](https://github.com/KRTirtho/spotube/commit/083319fd2445ab179e3dcda0a6aeaca6f13dda29)) +- swipe to open player view ([#765](https://github.com/KRTirtho/spotube/issues/765)) ([9aee056](https://github.com/KRTirtho/spotube/commit/9aee0568bf42eed9fea8d517e960a010abf0ebf2)) +- thicken the scrollbars & make 'em interactive for mobile ([#764](https://github.com/KRTirtho/spotube/issues/764)) ([84a4bcd](https://github.com/KRTirtho/spotube/commit/84a4bcd948ab459489aaf6f39d6954776c3401d7)) +- **translations:** add Arabic Translations ([#740](https://github.com/KRTirtho/spotube/issues/740)) ([38493f9](https://github.com/KRTirtho/spotube/commit/38493f9dd75303890857a626c0b276ee1ab75bb2)) +- **translations:** add Farsi Translations ([#760](https://github.com/KRTirtho/spotube/issues/760)) ([fe42cfe](https://github.com/KRTirtho/spotube/commit/fe42cfe8430035d9b67dd158fb7b835ee4071497)) ### Bug Fixes -* add libmpv1 for ubuntu-based systems ([#739](https://github.com/KRTirtho/spotube/issues/739)) ([5115e04](https://github.com/KRTirtho/spotube/commit/5115e041e78c20fce798a80f1d844bfc60746958)) -* add xdg-user-dirs as deps ([f3e331e](https://github.com/KRTirtho/spotube/commit/f3e331ecf733995da24c9b907efc5ed4bd02ffdd)) -* **android :** file_selector getDirectoryPath returns unusable content urls [#720](https://github.com/KRTirtho/spotube/issues/720) ([b3cf639](https://github.com/KRTirtho/spotube/commit/b3cf639ee2f970f4df9b394b260c3ad8a5732a9c)) -* **android:** audio doesn't resume on interruption end ([15d466a](https://github.com/KRTirtho/spotube/commit/15d466a04538ec70c3a0c132f2baaaf8690f8d4e)) -* **android:** system navigator back doesn't close player ([20d7092](https://github.com/KRTirtho/spotube/commit/20d70927c909347e84ffa8e456f8fab88d49d179)) -* get rid of overflow errors & status bar dark color ([5bb8231](https://github.com/KRTirtho/spotube/commit/5bb8231782287faf75c778fadb3a03ac774d14f0)) -* keyboard shortcuts changing route but not update sidebar ([2d93441](https://github.com/KRTirtho/spotube/commit/2d934411887bd104d8265236df5bf595c5ad2278)) -* last track repeats ([ed6ca00](https://github.com/KRTirtho/spotube/commit/ed6ca006ce237ed8d509cde9ed47cd6ea3396b63)) -* minor glitches ([e5d0aaf](https://github.com/KRTirtho/spotube/commit/e5d0aaf80d22b2291b6f7e7c5e18dd99ae1a7a82)) -* not fetching all followed artists ([#759](https://github.com/KRTirtho/spotube/issues/759)) ([c09a572](https://github.com/KRTirtho/spotube/commit/c09a5729251d8df820442d55477455f78c19c52e)) -* use audio_service_mpris plugin ([e29cc25](https://github.com/KRTirtho/spotube/commit/e29cc2578cab36729e235b117c1b5489c3452902)) -* valid non-ASCII characters get removed from downloaded file name [#745](https://github.com/KRTirtho/spotube/issues/745) ([a7e102f](https://github.com/KRTirtho/spotube/commit/a7e102ffc726d00df369560ec9a7f742f9d387bb)) +- add libmpv1 for ubuntu-based systems ([#739](https://github.com/KRTirtho/spotube/issues/739)) ([5115e04](https://github.com/KRTirtho/spotube/commit/5115e041e78c20fce798a80f1d844bfc60746958)) +- add xdg-user-dirs as deps ([f3e331e](https://github.com/KRTirtho/spotube/commit/f3e331ecf733995da24c9b907efc5ed4bd02ffdd)) +- **android :** file_selector getDirectoryPath returns unusable content urls [#720](https://github.com/KRTirtho/spotube/issues/720) ([b3cf639](https://github.com/KRTirtho/spotube/commit/b3cf639ee2f970f4df9b394b260c3ad8a5732a9c)) +- **android:** audio doesn't resume on interruption end ([15d466a](https://github.com/KRTirtho/spotube/commit/15d466a04538ec70c3a0c132f2baaaf8690f8d4e)) +- **android:** system navigator back doesn't close player ([20d7092](https://github.com/KRTirtho/spotube/commit/20d70927c909347e84ffa8e456f8fab88d49d179)) +- get rid of overflow errors & status bar dark color ([5bb8231](https://github.com/KRTirtho/spotube/commit/5bb8231782287faf75c778fadb3a03ac774d14f0)) +- keyboard shortcuts changing route but not update sidebar ([2d93441](https://github.com/KRTirtho/spotube/commit/2d934411887bd104d8265236df5bf595c5ad2278)) +- last track repeats ([ed6ca00](https://github.com/KRTirtho/spotube/commit/ed6ca006ce237ed8d509cde9ed47cd6ea3396b63)) +- minor glitches ([e5d0aaf](https://github.com/KRTirtho/spotube/commit/e5d0aaf80d22b2291b6f7e7c5e18dd99ae1a7a82)) +- not fetching all followed artists ([#759](https://github.com/KRTirtho/spotube/issues/759)) ([c09a572](https://github.com/KRTirtho/spotube/commit/c09a5729251d8df820442d55477455f78c19c52e)) +- use audio_service_mpris plugin ([e29cc25](https://github.com/KRTirtho/spotube/commit/e29cc2578cab36729e235b117c1b5489c3452902)) +- valid non-ASCII characters get removed from downloaded file name [#745](https://github.com/KRTirtho/spotube/issues/745) ([a7e102f](https://github.com/KRTirtho/spotube/commit/a7e102ffc726d00df369560ec9a7f742f9d387bb)) ## [3.1.2](https://github.com/KRTirtho/spotube/compare/v3.1.1...v3.1.2) (2023-09-15) - ### Features -* **player_queue:** filtering track support ([d4f99ec](https://github.com/KRTirtho/spotube/commit/d4f99ec89927ea78f070707509ff3222ec402942)) -* right click to open track option ([1540999](https://github.com/KRTirtho/spotube/commit/1540999f50d7ba78d9706d73127483b98d800d86)) -* search loading animation ([b9d5c70](https://github.com/KRTirtho/spotube/commit/b9d5c70301dd33ec26332e5e9a456ce5bfe73da0)) -* show loading indicator on play track ([d12ea48](https://github.com/KRTirtho/spotube/commit/d12ea48b97596205d6309012d561ce83e5cbc9c1)) - +- **player_queue:** filtering track support ([d4f99ec](https://github.com/KRTirtho/spotube/commit/d4f99ec89927ea78f070707509ff3222ec402942)) +- right click to open track option ([1540999](https://github.com/KRTirtho/spotube/commit/1540999f50d7ba78d9706d73127483b98d800d86)) +- search loading animation ([b9d5c70](https://github.com/KRTirtho/spotube/commit/b9d5c70301dd33ec26332e5e9a456ce5bfe73da0)) +- show loading indicator on play track ([d12ea48](https://github.com/KRTirtho/spotube/commit/d12ea48b97596205d6309012d561ce83e5cbc9c1)) ### Bug Fixes -* add missing dependency in debian package ([#704](https://github.com/KRTirtho/spotube/issues/704)) ([c987ea7](https://github.com/KRTirtho/spotube/commit/c987ea78414f094dead2c2b35ddf8ae83a70f2fe)) -* hour not showing for tracks longer than 60 minutes ([#648](https://github.com/KRTirtho/spotube/issues/648)) ([de335f4](https://github.com/KRTirtho/spotube/commit/de335f48342e45a077d6c3202706ef48dfb0a326)) -* liked tracks card play not working ([d3e1cef](https://github.com/KRTirtho/spotube/commit/d3e1cef8a21ef7d64e74ca4e99b4b57b653b60a7)) -* limit cover image upload to allowed 256kb size ([1c50612](https://github.com/KRTirtho/spotube/commit/1c50612559a78dce9c108f7e7b816d1b84540fe4)) -* playlist grey page ([#707](https://github.com/KRTirtho/spotube/issues/707)) ([0df8d9c](https://github.com/KRTirtho/spotube/commit/0df8d9cacee718fbb4cf3ec7b950b489630f3145)) -* rewind breaks track progress bar ([#695](https://github.com/KRTirtho/spotube/issues/695)) ([e321743](https://github.com/KRTirtho/spotube/commit/e3217436c9985b86c68dab93ea65ee414b32fb49)) -* Windows memory leak due refetchOnStale user-liked-tracks ([#705](https://github.com/KRTirtho/spotube/issues/705)) ([142dc49](https://github.com/KRTirtho/spotube/commit/142dc498f8f9d26e6b370c9c52f790a20832fc38)) +- add missing dependency in debian package ([#704](https://github.com/KRTirtho/spotube/issues/704)) ([c987ea7](https://github.com/KRTirtho/spotube/commit/c987ea78414f094dead2c2b35ddf8ae83a70f2fe)) +- hour not showing for tracks longer than 60 minutes ([#648](https://github.com/KRTirtho/spotube/issues/648)) ([de335f4](https://github.com/KRTirtho/spotube/commit/de335f48342e45a077d6c3202706ef48dfb0a326)) +- liked tracks card play not working ([d3e1cef](https://github.com/KRTirtho/spotube/commit/d3e1cef8a21ef7d64e74ca4e99b4b57b653b60a7)) +- limit cover image upload to allowed 256kb size ([1c50612](https://github.com/KRTirtho/spotube/commit/1c50612559a78dce9c108f7e7b816d1b84540fe4)) +- playlist grey page ([#707](https://github.com/KRTirtho/spotube/issues/707)) ([0df8d9c](https://github.com/KRTirtho/spotube/commit/0df8d9cacee718fbb4cf3ec7b950b489630f3145)) +- rewind breaks track progress bar ([#695](https://github.com/KRTirtho/spotube/issues/695)) ([e321743](https://github.com/KRTirtho/spotube/commit/e3217436c9985b86c68dab93ea65ee414b32fb49)) +- Windows memory leak due refetchOnStale user-liked-tracks ([#705](https://github.com/KRTirtho/spotube/issues/705)) ([142dc49](https://github.com/KRTirtho/spotube/commit/142dc498f8f9d26e6b370c9c52f790a20832fc38)) ## [3.1.1](https://github.com/KRTirtho/spotube/compare/v3.1.0...v3.1.1) (2023-08-28) - ### Features -* ability to toggle system title bar & custom title bar ([#185](https://github.com/KRTirtho/spotube/issues/185)) ([8d46029](https://github.com/KRTirtho/spotube/commit/8d4602962be20ea4bafc20db10eae1160f83ac52)) -* jump to specific time on lyric click ([#590](https://github.com/KRTirtho/spotube/issues/590)) ([a14fb9e](https://github.com/KRTirtho/spotube/commit/a14fb9ec389822e5ffa0c537e162b87cbba34e6c)) -* paginated user albums ([d239d64](https://github.com/KRTirtho/spotube/commit/d239d641ff8f1b3edd64243994fd4a58cf71a5d3)) -* **translations:** add Russian translation ([#661](https://github.com/KRTirtho/spotube/issues/661)) ([e9a0911](https://github.com/KRTirtho/spotube/commit/e9a0911bfcea2374ee282aee738c12ad9ed93b02)), closes [#625](https://github.com/KRTirtho/spotube/issues/625) -* **translations:** added Portuguese (Brazil) translation ([#634](https://github.com/KRTirtho/spotube/issues/634)) ([76f30a0](https://github.com/KRTirtho/spotube/commit/76f30a0f20f2b09680d27525cde3d1c9617fad5a)) - +- ability to toggle system title bar & custom title bar ([#185](https://github.com/KRTirtho/spotube/issues/185)) ([8d46029](https://github.com/KRTirtho/spotube/commit/8d4602962be20ea4bafc20db10eae1160f83ac52)) +- jump to specific time on lyric click ([#590](https://github.com/KRTirtho/spotube/issues/590)) ([a14fb9e](https://github.com/KRTirtho/spotube/commit/a14fb9ec389822e5ffa0c537e162b87cbba34e6c)) +- paginated user albums ([d239d64](https://github.com/KRTirtho/spotube/commit/d239d641ff8f1b3edd64243994fd4a58cf71a5d3)) +- **translations:** add Russian translation ([#661](https://github.com/KRTirtho/spotube/issues/661)) ([e9a0911](https://github.com/KRTirtho/spotube/commit/e9a0911bfcea2374ee282aee738c12ad9ed93b02)), closes [#625](https://github.com/KRTirtho/spotube/issues/625) +- **translations:** added Portuguese (Brazil) translation ([#634](https://github.com/KRTirtho/spotube/issues/634)) ([76f30a0](https://github.com/KRTirtho/spotube/commit/76f30a0f20f2b09680d27525cde3d1c9617fad5a)) ### Bug Fixes -* always fetching SponsorBlock if no segments found & download failing ([6ced0a0](https://github.com/KRTirtho/spotube/commit/6ced0a0fad06f9f431636ca0fe5dae83eafe33ce)) -* debian bookworm invalid dependencies ([633415d](https://github.com/KRTirtho/spotube/commit/633415dd3e702a38c5a7e7d7b3b1c2713d9c9cc9)) -* disable android auto for playstore version :"( ([0f0d240](https://github.com/KRTirtho/spotube/commit/0f0d240c04d77db6f7c127d59ba8b331d5534469)) -* infinite route push glitch ([e90eceb](https://github.com/KRTirtho/spotube/commit/e90eceb285a84028df690c25a687ff9b5168bba8)) -* jump to track going to wrong track ([190df17](https://github.com/KRTirtho/spotube/commit/190df17adcf4c01cb2bcebfdec47908828b33816)) -* last track of queue never plays & repeat playlist never works ([c3c09f5](https://github.com/KRTirtho/spotube/commit/c3c09f5b76c9547a306d15cd3768dacc1622876d)) -* lyrics page text contrast ([179d536](https://github.com/KRTirtho/spotube/commit/179d536ccc10a5e63f11a63680a6e61c2d1314c8)) -* replace connectivity_plus with internet_connection_checker ([f23e871](https://github.com/KRTirtho/spotube/commit/f23e8719eec7f5bed677ea866cb4bfab7aee5373)) -* sanitize song title for file name ([#644](https://github.com/KRTirtho/spotube/issues/644)) ([1a7ea0c](https://github.com/KRTirtho/spotube/commit/1a7ea0ce6aae1a7cbe195f6b2fae7d99082bb828)) -* sorting by date crashes app ([#551](https://github.com/KRTirtho/spotube/issues/551)) ([48e90a4](https://github.com/KRTirtho/spotube/commit/48e90a42294a6287cad65f840a7cc305988d34ff)) -* window size remains same after exiting mini player ([#618](https://github.com/KRTirtho/spotube/issues/618)) ([fb36003](https://github.com/KRTirtho/spotube/commit/fb360035ade09c270b46a0c3b99ab1594ece07c0)) +- always fetching SponsorBlock if no segments found & download failing ([6ced0a0](https://github.com/KRTirtho/spotube/commit/6ced0a0fad06f9f431636ca0fe5dae83eafe33ce)) +- debian bookworm invalid dependencies ([633415d](https://github.com/KRTirtho/spotube/commit/633415dd3e702a38c5a7e7d7b3b1c2713d9c9cc9)) +- disable android auto for playstore version :"( ([0f0d240](https://github.com/KRTirtho/spotube/commit/0f0d240c04d77db6f7c127d59ba8b331d5534469)) +- infinite route push glitch ([e90eceb](https://github.com/KRTirtho/spotube/commit/e90eceb285a84028df690c25a687ff9b5168bba8)) +- jump to track going to wrong track ([190df17](https://github.com/KRTirtho/spotube/commit/190df17adcf4c01cb2bcebfdec47908828b33816)) +- last track of queue never plays & repeat playlist never works ([c3c09f5](https://github.com/KRTirtho/spotube/commit/c3c09f5b76c9547a306d15cd3768dacc1622876d)) +- lyrics page text contrast ([179d536](https://github.com/KRTirtho/spotube/commit/179d536ccc10a5e63f11a63680a6e61c2d1314c8)) +- replace connectivity_plus with internet_connection_checker ([f23e871](https://github.com/KRTirtho/spotube/commit/f23e8719eec7f5bed677ea866cb4bfab7aee5373)) +- sanitize song title for file name ([#644](https://github.com/KRTirtho/spotube/issues/644)) ([1a7ea0c](https://github.com/KRTirtho/spotube/commit/1a7ea0ce6aae1a7cbe195f6b2fae7d99082bb828)) +- sorting by date crashes app ([#551](https://github.com/KRTirtho/spotube/issues/551)) ([48e90a4](https://github.com/KRTirtho/spotube/commit/48e90a42294a6287cad65f840a7cc305988d34ff)) +- window size remains same after exiting mini player ([#618](https://github.com/KRTirtho/spotube/issues/618)) ([fb36003](https://github.com/KRTirtho/spotube/commit/fb360035ade09c270b46a0c3b99ab1594ece07c0)) ## [3.1.0](https://github.com/KRTirtho/spotube/compare/v3.0.1...v3.1.0) (2023-08-18) - ### Features -* add android auto media session control support ([0f5748a](https://github.com/KRTirtho/spotube/commit/0f5748a24b4b5a32862f7b2f28151e2d42bcce33)) -* better track matching on youtube API ([904a0d3](https://github.com/KRTirtho/spotube/commit/904a0d3e15a8e4a76d0842e978809d2838439f86)) -* blazingly™ fast download manager ([#619](https://github.com/KRTirtho/spotube/issues/619)) ([38dc4be](https://github.com/KRTirtho/spotube/commit/38dc4beb44827a20044afd120d7c32f097938660)) -* paginated user playlists ([e7c6813](https://github.com/KRTirtho/spotube/commit/e7c6813ccb2afcda9a8b044570a1c0a27785a59f)) -* show error dialog on piped API 500 error ([c69f81e](https://github.com/KRTirtho/spotube/commit/c69f81ec6f01f0f67ad06446e890aa1351516626)) -* **translation:** add catalan translations ([#621](https://github.com/KRTirtho/spotube/issues/621)) ([c94e5ba](https://github.com/KRTirtho/spotube/commit/c94e5ba4301ed0cb760daff56c56d2701b35131f)) -* **translations:** add polish translation ([#631](https://github.com/KRTirtho/spotube/issues/631)) ([f90e9be](https://github.com/KRTirtho/spotube/commit/f90e9bee3104a3812c2f775dd16cabbb56f668cb)) -* web compatibility ([cf7b849](https://github.com/KRTirtho/spotube/commit/cf7b849cddca3260d9c3a6a064418b0ba2d63270)) - +- add android auto media session control support ([0f5748a](https://github.com/KRTirtho/spotube/commit/0f5748a24b4b5a32862f7b2f28151e2d42bcce33)) +- better track matching on youtube API ([904a0d3](https://github.com/KRTirtho/spotube/commit/904a0d3e15a8e4a76d0842e978809d2838439f86)) +- blazingly™ fast download manager ([#619](https://github.com/KRTirtho/spotube/issues/619)) ([38dc4be](https://github.com/KRTirtho/spotube/commit/38dc4beb44827a20044afd120d7c32f097938660)) +- paginated user playlists ([e7c6813](https://github.com/KRTirtho/spotube/commit/e7c6813ccb2afcda9a8b044570a1c0a27785a59f)) +- show error dialog on piped API 500 error ([c69f81e](https://github.com/KRTirtho/spotube/commit/c69f81ec6f01f0f67ad06446e890aa1351516626)) +- **translation:** add catalan translations ([#621](https://github.com/KRTirtho/spotube/issues/621)) ([c94e5ba](https://github.com/KRTirtho/spotube/commit/c94e5ba4301ed0cb760daff56c56d2701b35131f)) +- **translations:** add polish translation ([#631](https://github.com/KRTirtho/spotube/issues/631)) ([f90e9be](https://github.com/KRTirtho/spotube/commit/f90e9bee3104a3812c2f775dd16cabbb56f668cb)) +- web compatibility ([cf7b849](https://github.com/KRTirtho/spotube/commit/cf7b849cddca3260d9c3a6a064418b0ba2d63270)) ### Bug Fixes -* always showing play in playlist/album views ([8521cc5](https://github.com/KRTirtho/spotube/commit/8521cc5c88730caa9db74da6c04b679bf29ed56d)) -* **android:** android 13 local tracks not showing up ([e3f4344](https://github.com/KRTirtho/spotube/commit/e3f4344ae9c1ec93860d4c5d1b8de1a803b29569)) -* default to youtube API by default ([5a8a1e4](https://github.com/KRTirtho/spotube/commit/5a8a1e41e93fb74756b8c88f6325b8b46d7af131)) -* generate playlist page max width ([4adf695](https://github.com/KRTirtho/spotube/commit/4adf6951d9ee78ac4b198d541dada28dc00ca0cb)) -* tracks doesn't change when ended ([aa4ac86](https://github.com/KRTirtho/spotube/commit/aa4ac8641a7dceb4626ab675ba376b24a3480d30)) -* windows media controls not working ([ae5edd1](https://github.com/KRTirtho/spotube/commit/ae5edd17ef24f2a38ec1cc9c9623868b6ee9e352)) +- always showing play in playlist/album views ([8521cc5](https://github.com/KRTirtho/spotube/commit/8521cc5c88730caa9db74da6c04b679bf29ed56d)) +- **android:** android 13 local tracks not showing up ([e3f4344](https://github.com/KRTirtho/spotube/commit/e3f4344ae9c1ec93860d4c5d1b8de1a803b29569)) +- default to youtube API by default ([5a8a1e4](https://github.com/KRTirtho/spotube/commit/5a8a1e41e93fb74756b8c88f6325b8b46d7af131)) +- generate playlist page max width ([4adf695](https://github.com/KRTirtho/spotube/commit/4adf6951d9ee78ac4b198d541dada28dc00ca0cb)) +- tracks doesn't change when ended ([aa4ac86](https://github.com/KRTirtho/spotube/commit/aa4ac8641a7dceb4626ab675ba376b24a3480d30)) +- windows media controls not working ([ae5edd1](https://github.com/KRTirtho/spotube/commit/ae5edd17ef24f2a38ec1cc9c9623868b6ee9e352)) ## [3.0.1](https://github.com/KRTirtho/spotube/compare/v3.0.0...v3.1.0) (2023-08-04) - ### Features -* Force High Refresh Rate on some Android devices ([#607](https://github.com/KRTirtho/spotube/issues/607)) ([6dff099](https://github.com/KRTirtho/spotube/commit/6dff0996bdfee603acf242b1316f8793d625267c)) -* **translations:** add spanish translations ([#585](https://github.com/KRTirtho/spotube/issues/585)) ([042d7a4](https://github.com/KRTirtho/spotube/commit/042d7a4a10c78dd93a56a2f32d18a0fb74dbe697)) -* **translations:** add Simplified Chinese translation. ([#556](https://github.com/KRTirtho/spotube/issues/556)) ([26dbd52](https://github.com/KRTirtho/spotube/commit/26dbd523737d868114a47e82acd412cdae622b7c)) - +- Force High Refresh Rate on some Android devices ([#607](https://github.com/KRTirtho/spotube/issues/607)) ([6dff099](https://github.com/KRTirtho/spotube/commit/6dff0996bdfee603acf242b1316f8793d625267c)) +- **translations:** add spanish translations ([#585](https://github.com/KRTirtho/spotube/issues/585)) ([042d7a4](https://github.com/KRTirtho/spotube/commit/042d7a4a10c78dd93a56a2f32d18a0fb74dbe697)) +- **translations:** add Simplified Chinese translation. ([#556](https://github.com/KRTirtho/spotube/issues/556)) ([26dbd52](https://github.com/KRTirtho/spotube/commit/26dbd523737d868114a47e82acd412cdae622b7c)) ### Bug Fixes -* alternative track source textfield safe area ([b8c6d7e](https://github.com/KRTirtho/spotube/commit/b8c6d7eb6ae1c54bdc83a455850dfca0f27bd881)) -* avoid sponsor block for first few seconds to not break the stream ([d8cf2ae](https://github.com/KRTirtho/spotube/commit/d8cf2ae1315dc3848fe1ac12286faafe90fdbed7)) -* cache segments casting error ([dfd60bd](https://github.com/KRTirtho/spotube/commit/dfd60bd4cc0fe8fe90e0cbfd26331df505cde2aa)) -* duration is always zero in PlayerView ([4885dca](https://github.com/KRTirtho/spotube/commit/4885dca04f06658391d1063e6c5a009547391a6f)) -* flags not showing up and html in descriptions ([5a563ef](https://github.com/KRTirtho/spotube/commit/5a563ef4289423ceb5c44ba13f3cfda34b2d16dd)) -* **linux:** crash when no secret service provider found ([#608](https://github.com/KRTirtho/spotube/issues/608)) ([888a4b1](https://github.com/KRTirtho/spotube/commit/888a4b1162c25371d7f6e88fae3a2473cabf1434)) -* login dialog stays after login, mention sp_gaid in tutorial ([b492840](https://github.com/KRTirtho/spotube/commit/b4928405122ae5e5d4d4560f316f2a546a2fabe4)) -* **album_sync**: negative index exception in update palette ([#561](https://github.com/KRTirtho/spotube/issues/561)) ([0089d47](https://github.com/KRTirtho/spotube/commit/0089d471ae6d595e058061e3ac44caecdba12f61)) -* remove adaptive widgets ([#520](https://github.com/KRTirtho/spotube/issues/520)) ([e4cbdd3](https://github.com/KRTirtho/spotube/commit/e4cbdd37479a572198c1ca27fcbbba0232275513)) -* shuffle not working ([#562](https://github.com/KRTirtho/spotube/issues/562)) ([dc76634](https://github.com/KRTirtho/spotube/commit/dc76634a6e4ccdca0f09d63a2db82cce53d950d7)) -* track not skipping to next even when source is available ([0b7affd](https://github.com/KRTirtho/spotube/commit/0b7affdc058c028982266d5c93215697301846bd)) +- alternative track source textfield safe area ([b8c6d7e](https://github.com/KRTirtho/spotube/commit/b8c6d7eb6ae1c54bdc83a455850dfca0f27bd881)) +- avoid sponsor block for first few seconds to not break the stream ([d8cf2ae](https://github.com/KRTirtho/spotube/commit/d8cf2ae1315dc3848fe1ac12286faafe90fdbed7)) +- cache segments casting error ([dfd60bd](https://github.com/KRTirtho/spotube/commit/dfd60bd4cc0fe8fe90e0cbfd26331df505cde2aa)) +- duration is always zero in PlayerView ([4885dca](https://github.com/KRTirtho/spotube/commit/4885dca04f06658391d1063e6c5a009547391a6f)) +- flags not showing up and html in descriptions ([5a563ef](https://github.com/KRTirtho/spotube/commit/5a563ef4289423ceb5c44ba13f3cfda34b2d16dd)) +- **linux:** crash when no secret service provider found ([#608](https://github.com/KRTirtho/spotube/issues/608)) ([888a4b1](https://github.com/KRTirtho/spotube/commit/888a4b1162c25371d7f6e88fae3a2473cabf1434)) +- login dialog stays after login, mention sp_gaid in tutorial ([b492840](https://github.com/KRTirtho/spotube/commit/b4928405122ae5e5d4d4560f316f2a546a2fabe4)) +- **album_sync**: negative index exception in update palette ([#561](https://github.com/KRTirtho/spotube/issues/561)) ([0089d47](https://github.com/KRTirtho/spotube/commit/0089d471ae6d595e058061e3ac44caecdba12f61)) +- remove adaptive widgets ([#520](https://github.com/KRTirtho/spotube/issues/520)) ([e4cbdd3](https://github.com/KRTirtho/spotube/commit/e4cbdd37479a572198c1ca27fcbbba0232275513)) +- shuffle not working ([#562](https://github.com/KRTirtho/spotube/issues/562)) ([dc76634](https://github.com/KRTirtho/spotube/commit/dc76634a6e4ccdca0f09d63a2db82cce53d950d7)) +- track not skipping to next even when source is available ([0b7affd](https://github.com/KRTirtho/spotube/commit/0b7affdc058c028982266d5c93215697301846bd)) ## [3.0.0](https://github.com/KRTirtho/spotube/compare/v2.7.1...v3.0.0) (2023-07-02) - ### Features -* adaptive controllers ([c8b7de0](https://github.com/KRTirtho/spotube/commit/c8b7de087917ec3037c015d5b55693cb3dbdecca)) -* adaptive popup and bottom sheet list widget ([ddc1c5f](https://github.com/KRTirtho/spotube/commit/ddc1c5f373a4d72cc231c35dd70c3d577b84f7f5)) -* add generated to playlist(s) ([c91d8c8](https://github.com/KRTirtho/spotube/commit/c91d8c8efa8b526c64881fa829992b8c250e7c89)) -* add german locale ([ba3f428](https://github.com/KRTirtho/spotube/commit/ba3f4281f1a6bc7ddf38775e9d40dad863ed3692)) -* add piped search mode ([17a25a5](https://github.com/KRTirtho/spotube/commit/17a25a501e0d5e2512d8de0921fd602ea906d30f)) -* add sleep timer support ([4a75f3d](https://github.com/KRTirtho/spotube/commit/4a75f3dbd1e7e6f68899de001df70e809533f142)) -* adjust lyric page blurriness and player playbutton ([54d5907](https://github.com/KRTirtho/spotube/commit/54d5907f14df04f3f983ba2b0401ba05785da03b)) -* album art dominant color as accent color ([#447](https://github.com/KRTirtho/spotube/issues/447)) ([31b9249](https://github.com/KRTirtho/spotube/commit/31b9249cc8f7313a132a514a9ca825c2ae1e2256)) -* **android:** add splash screen ([c232fcc](https://github.com/KRTirtho/spotube/commit/c232fcc6dd1479ed33a8baa9887de2702a8ea22e)) -* **android:** disable battery optimization for better playback ([fe5b429](https://github.com/KRTirtho/spotube/commit/fe5b429ddacc576fc9fdb5e66718782cda163b27)) -* artist card redesign ([92a418c](https://github.com/KRTirtho/spotube/commit/92a418c8a8a9df99e27407b628e5e3cc9ccb4115)) -* Better download manager with download progress ([6752adc](https://github.com/KRTirtho/spotube/commit/6752adc9398818f51b69fced226b4b8410fb9e9b)) -* better language picker, adaptive select tile and settings section contrast ([6430a25](https://github.com/KRTirtho/spotube/commit/6430a2587075aa24483ab26ce6f0f6b2b630e139)) -* cache encryption for sensitive data ([b110d83](https://github.com/KRTirtho/spotube/commit/b110d834561ac53129ac9ec80238c014c84832ec)) -* color scheme picker dialog vertical list view instead of wrap ([bb60b01](https://github.com/KRTirtho/spotube/commit/bb60b01ef2f2ba3da8f6fe3a19add168b2ee8a4e)) -* compact and adaptive playbutton card design ([eeb8cab](https://github.com/KRTirtho/spotube/commit/eeb8cabf491d5242bd434b3c71c39363f24bdcf9)) -* compact button tabbar ([67380f6](https://github.com/KRTirtho/spotube/commit/67380f68765f18e4dcd3d60117083c7e9c6761c2)) -* create a basic installer script ([1763a36](https://github.com/KRTirtho/spotube/commit/1763a36a262178306df61d4588c17ff795a32790)) -* curved navigation bar ([776edf8](https://github.com/KRTirtho/spotube/commit/776edf84afcf99f96cf6e337b0c84ed89034ca8e)) -* custom error toast ([96f04c1](https://github.com/KRTirtho/spotube/commit/96f04c17565c0ab7f115d5c1f167f6660a69480d)) -* custom playlist generator ([f4b0d13](https://github.com/KRTirtho/spotube/commit/f4b0d134ca724b75bf65b885bce4dba206f1e090)) -* desktop mini player support ([471812d](https://github.com/KRTirtho/spotube/commit/471812d789eb2c861268ab8451d50104ac2fbe2e)) -* **desktop:** close button for minimize notification ([1688f99](https://github.com/KRTirtho/spotube/commit/1688f99096af940ead65e67832c6f061a6f635ac)) -* **desktop:** show minimized to system tray notification ([296f96c](https://github.com/KRTirtho/spotube/commit/296f96cf17cf21ff09b406cc24952ff60da52d5e)) -* disable/enable smtc on demand ([7fa50e5](https://github.com/KRTirtho/spotube/commit/7fa50e5c5ee9ca3ba95dca55f8a4831047f17570)) -* download button on each track ([925fa86](https://github.com/KRTirtho/spotube/commit/925fa86271fa10ff77b8137ba8d09b8067d0e819)) -* enable caching of queue ([ec11af5](https://github.com/KRTirtho/spotube/commit/ec11af53a16c435fbea3d0c81910dca371be9ce7)) -* heart button animation ([8432dc6](https://github.com/KRTirtho/spotube/commit/8432dc6286fbdfda52bbeb39c6d4ababa05881bc)) -* improved track item API and UI ([617aa89](https://github.com/KRTirtho/spotube/commit/617aa89409ce29eb3c197aee5c1189d763a3913c)) -* **installer:** get latest version from Github API ([957c085](https://github.com/KRTirtho/spotube/commit/957c085e1243ec0af5bc45d38691c74e2ba91ad8)) -* **local_tracks:** delete local track ([#484](https://github.com/KRTirtho/spotube/issues/484)) ([52835b2](https://github.com/KRTirtho/spotube/commit/52835b2ce2212925f80e1a1595c0ca30e6860a8d)) -* locale category/genre title ([88137f0](https://github.com/KRTirtho/spotube/commit/88137f01b27150b306327a01e67ec8a35a60e82e)) -* **locale:** add bengali translations for search page ([a1cdbad](https://github.com/KRTirtho/spotube/commit/a1cdbad18782a74b43f0625facfe3e35c516bf43)) -* **locale:** localize search, library, lyrics, artist with both Bengali and English ([11fe9ec](https://github.com/KRTirtho/spotube/commit/11fe9ec74462441b67a0ab0df73824f36dd15e2d)) -* **locale:** player, playlist view, track tile bengali and english translations ([c55133d](https://github.com/KRTirtho/spotube/commit/c55133dc8bba307823e5b67a30cfab03c923cb7f)) -* localize settings, about, login, player queue with Bengali and English translations ([a5c36bb](https://github.com/KRTirtho/spotube/commit/a5c36bbb20cc69d609bfb5ab973c7e288c1ea9de)) -* logs page in settings ([b78e7f5](https://github.com/KRTirtho/spotube/commit/b78e7f57a05db344aae59206cbb0f43b3ee199a9)) -* macos title bar spacing and lyrics page margin separate ([a0b3771](https://github.com/KRTirtho/spotube/commit/a0b377104f9822561d3b46dbc6551bb561842480)) -* make snackbar floating ([9dbb817](https://github.com/KRTirtho/spotube/commit/9dbb8171a6d6b81120ca7ccd74577e5c890ff930)) -* merge floating player with nav bar and nav bar translucent bg ([a90261e](https://github.com/KRTirtho/spotube/commit/a90261ed199f6cff9e8d0fe24934f8f1d8e9ed98)) -* **mini_player:** remove window shadow ([6259014](https://github.com/KRTirtho/spotube/commit/625901482ada4b441b838f640c0ab7167119b321)) -* **mini_player:** show/hide UI on hover toggle ([2e8b647](https://github.com/KRTirtho/spotube/commit/2e8b647a51f87840c2bd39f0a1dc25ddc91528fc)) -* new sidebar widget and translucent bottom player ([4ba1e70](https://github.com/KRTirtho/spotube/commit/4ba1e70636b4ba43697663128fc5422b1d0b2a2f)) -* newly released albums of user followed artist ([33cb794](https://github.com/KRTirtho/spotube/commit/33cb7947d63d0a2692a004f87a2ccd5777bf054e)) -* optimize image load + genre page and reduce page size of loaded categories ([7131efa](https://github.com/KRTirtho/spotube/commit/7131efa07fdbcf17965fc59ff635a6198b0e5e25)) -* persistent volume percentage ([3724bd5](https://github.com/KRTirtho/spotube/commit/3724bd5a10eef7a099d6f596fd038e6fea228359)) -* personal playlist recommendations ([ae820a2](https://github.com/KRTirtho/spotube/commit/ae820a22f291082c49554d621c25cc62212a6708)) -* piped instance picker on settings ([bed0d3b](https://github.com/KRTirtho/spotube/commit/bed0d3bd70438df413633ee03fd258a2ca4a1688)) -* platform specific title bar buttons ([6267720](https://github.com/KRTirtho/spotube/commit/62677209a23172162defb7a8e542d981569eba08)) -* **playback:** integrate android, ios, macos with JustAudio ([d487fe5](https://github.com/KRTirtho/spotube/commit/d487fe55630993e2b729050ebd0bf4e1e4be1fb3)) -* **playback:** use assets_audio_player to fix macos double duration problems and android high loading latency ([1fff0f1](https://github.com/KRTirtho/spotube/commit/1fff0f1bd0d811c348f293f733cbcb7cd57e02f8)) -* player details dialog and separate location of lyrics button in player page ([ce38233](https://github.com/KRTirtho/spotube/commit/ce38233de8f4775018a1d01e951b1635776fe743)) -* **player:** add playlist related methods to audio player ([f1080e1](https://github.com/KRTirtho/spotube/commit/f1080e1675aee1208d05658adfabfbed04ff45b6)) -* **player:** animated gradient background ([49b5d0e](https://github.com/KRTirtho/spotube/commit/49b5d0e6948d80abb8ee09203e0d655d68377245)) -* **player:** custom playlist implementation for media_kit to replace unpredictable playlist of mpv ([eaf65b6](https://github.com/KRTirtho/spotube/commit/eaf65b6db208aaad745821d4d42afc05f51cee7c)) -* **player:** proper coloring of elements ([b2c4ea1](https://github.com/KRTirtho/spotube/commit/b2c4ea13f6157c2b7bec3957e1f7f50fbf0002c7)) -* **player:** replace bg blur with gradient, proper fg color and align title and artist name ([159f03e](https://github.com/KRTirtho/spotube/commit/159f03e7ca62e6b3b86389e2795da84de61fba78)) -* playlist create support for generated playlist ([91c72f9](https://github.com/KRTirtho/spotube/commit/91c72f9ec9556f301c5d129fc82e19e791a02fbe)) -* playlist generation all parameters support ([9877d5f](https://github.com/KRTirtho/spotube/commit/9877d5f51736db03d5839dadf164d11d0cce82f0)) -* **playlist,album page:** play and shuffle take full width on smaller screens, add new xs breakpoint ([dce1b88](https://github.com/KRTirtho/spotube/commit/dce1b88694cfcb6b7e63d6ee614ac1dbbd017f6e)) -* **queue:** add track(s) for playing next ([#460](https://github.com/KRTirtho/spotube/issues/460)) ([cac8ea6](https://github.com/KRTirtho/spotube/commit/cac8ea638812f5d9cb4305144b6351141a2cf407)) -* **queue:** reorder tracks support ([441b43b](https://github.com/KRTirtho/spotube/commit/441b43bef6b92fd7df6c4e1bef39d67b4a76cd22)) -* re-designed playlist/album page ([0cedc7a](https://github.com/KRTirtho/spotube/commit/0cedc7a4187771efce8152003f890e242116c78c)) -* re-introduce youtube API along with piped ([b54ee96](https://github.com/KRTirtho/spotube/commit/b54ee96233b29d7517eba66e3f8dd9270c2790df)) -* reactive volume slider and slicker bottom bar with lowered height ([9d14517](https://github.com/KRTirtho/spotube/commit/9d14517202d5c9d993a947808bf0c6520ed54ea3)) -* remove SponsorBlock in favor of YT Music and remove pocketbase backend track support ([fb780da](https://github.com/KRTirtho/spotube/commit/fb780da327a213d7a82cbc3b567ece858dc2f0e8)) -* repeat button all 3 mode and disable player controls when track is fetching ([1418378](https://github.com/KRTirtho/spotube/commit/14183781dd3f1e16c121e78ad637a326de7b5dcf)) -* replace YouTube API with piped API ([1ecc36d](https://github.com/KRTirtho/spotube/commit/1ecc36da57af61fd9c2ca928589088cd4325f605)) -* responsive playlist generate page and scrollable multi autocomplete ([d57aad5](https://github.com/KRTirtho/spotube/commit/d57aad5612f7622dcd638ea8c0ec4d96f741de2b)) -* search alternative track source ([dfea195](https://github.com/KRTirtho/spotube/commit/dfea195ec178de733717cfe3226cede7521ee2d3)) -* setup localization (l10n) and language switcher, add sidebar and navbar locale ([f12d812](https://github.com/KRTirtho/spotube/commit/f12d81259f9e7005e681a7ca9867291d9228a8b1)) -* show album release year ([#387](https://github.com/KRTirtho/spotube/issues/387)) and fix layout of artist's album ([6a6ddf6](https://github.com/KRTirtho/spotube/commit/6a6ddf6e1f6dc72b794cae49adf8348da272babd)) -* show country code piped instance list ([60328a6](https://github.com/KRTirtho/spotube/commit/60328a6bafcbff1b7d0ee5099825f0e3d545b60f)) -* show loading when track metadata is being fetched, android, ios, macos enable shuffling ([bf59570](https://github.com/KRTirtho/spotube/commit/bf59570251720a80efe0aa6be481899864da5079)) -* sort tracks by newest and oldest dates ([b4713e3](https://github.com/KRTirtho/spotube/commit/b4713e377a938cbebe70089874216f86fe550c34)) -* supabase integration ([8bcce92](https://github.com/KRTirtho/spotube/commit/8bcce9282eae08c5996a27f16f89cbc187a06823)) -* system tray support ([#31](https://github.com/KRTirtho/spotube/issues/31)) ([06a0437](https://github.com/KRTirtho/spotube/commit/06a043764d9f65fb448fcf088ccf2737145e23e8)) -* track populate sibling support ([3aeb026](https://github.com/KRTirtho/spotube/commit/3aeb026776716b6e2eb89c8406a4996a86c7ca60)) -* **translation:** add hindi and french translations using ChatGPT ([6d836bd](https://github.com/KRTirtho/spotube/commit/6d836bdb658c180ca8e2c71e7e290fafa3520727)) -* **translation:** add Japanase locale ([4b52a71](https://github.com/KRTirtho/spotube/commit/4b52a71c0914bda6c831d8f637a5934f7bcf8fcb)) -* use system color scheme ([862c4b8](https://github.com/KRTirtho/spotube/commit/862c4b8faf2c751d803e373e29981a116bf08ed5)) -* volume slider in player page ([7abe2c1](https://github.com/KRTirtho/spotube/commit/7abe2c10735bc38c644487139557a731d25e80e6)) -* windows OS media control panel support ([f0b426a](https://github.com/KRTirtho/spotube/commit/f0b426ae89f2e01f4a9c8757ef4e0b4a21b50c7b)) - +- adaptive controllers ([c8b7de0](https://github.com/KRTirtho/spotube/commit/c8b7de087917ec3037c015d5b55693cb3dbdecca)) +- adaptive popup and bottom sheet list widget ([ddc1c5f](https://github.com/KRTirtho/spotube/commit/ddc1c5f373a4d72cc231c35dd70c3d577b84f7f5)) +- add generated to playlist(s) ([c91d8c8](https://github.com/KRTirtho/spotube/commit/c91d8c8efa8b526c64881fa829992b8c250e7c89)) +- add german locale ([ba3f428](https://github.com/KRTirtho/spotube/commit/ba3f4281f1a6bc7ddf38775e9d40dad863ed3692)) +- add piped search mode ([17a25a5](https://github.com/KRTirtho/spotube/commit/17a25a501e0d5e2512d8de0921fd602ea906d30f)) +- add sleep timer support ([4a75f3d](https://github.com/KRTirtho/spotube/commit/4a75f3dbd1e7e6f68899de001df70e809533f142)) +- adjust lyric page blurriness and player playbutton ([54d5907](https://github.com/KRTirtho/spotube/commit/54d5907f14df04f3f983ba2b0401ba05785da03b)) +- album art dominant color as accent color ([#447](https://github.com/KRTirtho/spotube/issues/447)) ([31b9249](https://github.com/KRTirtho/spotube/commit/31b9249cc8f7313a132a514a9ca825c2ae1e2256)) +- **android:** add splash screen ([c232fcc](https://github.com/KRTirtho/spotube/commit/c232fcc6dd1479ed33a8baa9887de2702a8ea22e)) +- **android:** disable battery optimization for better playback ([fe5b429](https://github.com/KRTirtho/spotube/commit/fe5b429ddacc576fc9fdb5e66718782cda163b27)) +- artist card redesign ([92a418c](https://github.com/KRTirtho/spotube/commit/92a418c8a8a9df99e27407b628e5e3cc9ccb4115)) +- Better download manager with download progress ([6752adc](https://github.com/KRTirtho/spotube/commit/6752adc9398818f51b69fced226b4b8410fb9e9b)) +- better language picker, adaptive select tile and settings section contrast ([6430a25](https://github.com/KRTirtho/spotube/commit/6430a2587075aa24483ab26ce6f0f6b2b630e139)) +- cache encryption for sensitive data ([b110d83](https://github.com/KRTirtho/spotube/commit/b110d834561ac53129ac9ec80238c014c84832ec)) +- color scheme picker dialog vertical list view instead of wrap ([bb60b01](https://github.com/KRTirtho/spotube/commit/bb60b01ef2f2ba3da8f6fe3a19add168b2ee8a4e)) +- compact and adaptive playbutton card design ([eeb8cab](https://github.com/KRTirtho/spotube/commit/eeb8cabf491d5242bd434b3c71c39363f24bdcf9)) +- compact button tabbar ([67380f6](https://github.com/KRTirtho/spotube/commit/67380f68765f18e4dcd3d60117083c7e9c6761c2)) +- create a basic installer script ([1763a36](https://github.com/KRTirtho/spotube/commit/1763a36a262178306df61d4588c17ff795a32790)) +- curved navigation bar ([776edf8](https://github.com/KRTirtho/spotube/commit/776edf84afcf99f96cf6e337b0c84ed89034ca8e)) +- custom error toast ([96f04c1](https://github.com/KRTirtho/spotube/commit/96f04c17565c0ab7f115d5c1f167f6660a69480d)) +- custom playlist generator ([f4b0d13](https://github.com/KRTirtho/spotube/commit/f4b0d134ca724b75bf65b885bce4dba206f1e090)) +- desktop mini player support ([471812d](https://github.com/KRTirtho/spotube/commit/471812d789eb2c861268ab8451d50104ac2fbe2e)) +- **desktop:** close button for minimize notification ([1688f99](https://github.com/KRTirtho/spotube/commit/1688f99096af940ead65e67832c6f061a6f635ac)) +- **desktop:** show minimized to system tray notification ([296f96c](https://github.com/KRTirtho/spotube/commit/296f96cf17cf21ff09b406cc24952ff60da52d5e)) +- disable/enable smtc on demand ([7fa50e5](https://github.com/KRTirtho/spotube/commit/7fa50e5c5ee9ca3ba95dca55f8a4831047f17570)) +- download button on each track ([925fa86](https://github.com/KRTirtho/spotube/commit/925fa86271fa10ff77b8137ba8d09b8067d0e819)) +- enable caching of queue ([ec11af5](https://github.com/KRTirtho/spotube/commit/ec11af53a16c435fbea3d0c81910dca371be9ce7)) +- heart button animation ([8432dc6](https://github.com/KRTirtho/spotube/commit/8432dc6286fbdfda52bbeb39c6d4ababa05881bc)) +- improved track item API and UI ([617aa89](https://github.com/KRTirtho/spotube/commit/617aa89409ce29eb3c197aee5c1189d763a3913c)) +- **installer:** get latest version from Github API ([957c085](https://github.com/KRTirtho/spotube/commit/957c085e1243ec0af5bc45d38691c74e2ba91ad8)) +- **local_tracks:** delete local track ([#484](https://github.com/KRTirtho/spotube/issues/484)) ([52835b2](https://github.com/KRTirtho/spotube/commit/52835b2ce2212925f80e1a1595c0ca30e6860a8d)) +- locale category/genre title ([88137f0](https://github.com/KRTirtho/spotube/commit/88137f01b27150b306327a01e67ec8a35a60e82e)) +- **locale:** add bengali translations for search page ([a1cdbad](https://github.com/KRTirtho/spotube/commit/a1cdbad18782a74b43f0625facfe3e35c516bf43)) +- **locale:** localize search, library, lyrics, artist with both Bengali and English ([11fe9ec](https://github.com/KRTirtho/spotube/commit/11fe9ec74462441b67a0ab0df73824f36dd15e2d)) +- **locale:** player, playlist view, track tile bengali and english translations ([c55133d](https://github.com/KRTirtho/spotube/commit/c55133dc8bba307823e5b67a30cfab03c923cb7f)) +- localize settings, about, login, player queue with Bengali and English translations ([a5c36bb](https://github.com/KRTirtho/spotube/commit/a5c36bbb20cc69d609bfb5ab973c7e288c1ea9de)) +- logs page in settings ([b78e7f5](https://github.com/KRTirtho/spotube/commit/b78e7f57a05db344aae59206cbb0f43b3ee199a9)) +- macos title bar spacing and lyrics page margin separate ([a0b3771](https://github.com/KRTirtho/spotube/commit/a0b377104f9822561d3b46dbc6551bb561842480)) +- make snackbar floating ([9dbb817](https://github.com/KRTirtho/spotube/commit/9dbb8171a6d6b81120ca7ccd74577e5c890ff930)) +- merge floating player with nav bar and nav bar translucent bg ([a90261e](https://github.com/KRTirtho/spotube/commit/a90261ed199f6cff9e8d0fe24934f8f1d8e9ed98)) +- **mini_player:** remove window shadow ([6259014](https://github.com/KRTirtho/spotube/commit/625901482ada4b441b838f640c0ab7167119b321)) +- **mini_player:** show/hide UI on hover toggle ([2e8b647](https://github.com/KRTirtho/spotube/commit/2e8b647a51f87840c2bd39f0a1dc25ddc91528fc)) +- new sidebar widget and translucent bottom player ([4ba1e70](https://github.com/KRTirtho/spotube/commit/4ba1e70636b4ba43697663128fc5422b1d0b2a2f)) +- newly released albums of user followed artist ([33cb794](https://github.com/KRTirtho/spotube/commit/33cb7947d63d0a2692a004f87a2ccd5777bf054e)) +- optimize image load + genre page and reduce page size of loaded categories ([7131efa](https://github.com/KRTirtho/spotube/commit/7131efa07fdbcf17965fc59ff635a6198b0e5e25)) +- persistent volume percentage ([3724bd5](https://github.com/KRTirtho/spotube/commit/3724bd5a10eef7a099d6f596fd038e6fea228359)) +- personal playlist recommendations ([ae820a2](https://github.com/KRTirtho/spotube/commit/ae820a22f291082c49554d621c25cc62212a6708)) +- piped instance picker on settings ([bed0d3b](https://github.com/KRTirtho/spotube/commit/bed0d3bd70438df413633ee03fd258a2ca4a1688)) +- platform specific title bar buttons ([6267720](https://github.com/KRTirtho/spotube/commit/62677209a23172162defb7a8e542d981569eba08)) +- **playback:** integrate android, ios, macos with JustAudio ([d487fe5](https://github.com/KRTirtho/spotube/commit/d487fe55630993e2b729050ebd0bf4e1e4be1fb3)) +- **playback:** use assets_audio_player to fix macos double duration problems and android high loading latency ([1fff0f1](https://github.com/KRTirtho/spotube/commit/1fff0f1bd0d811c348f293f733cbcb7cd57e02f8)) +- player details dialog and separate location of lyrics button in player page ([ce38233](https://github.com/KRTirtho/spotube/commit/ce38233de8f4775018a1d01e951b1635776fe743)) +- **player:** add playlist related methods to audio player ([f1080e1](https://github.com/KRTirtho/spotube/commit/f1080e1675aee1208d05658adfabfbed04ff45b6)) +- **player:** animated gradient background ([49b5d0e](https://github.com/KRTirtho/spotube/commit/49b5d0e6948d80abb8ee09203e0d655d68377245)) +- **player:** custom playlist implementation for media_kit to replace unpredictable playlist of mpv ([eaf65b6](https://github.com/KRTirtho/spotube/commit/eaf65b6db208aaad745821d4d42afc05f51cee7c)) +- **player:** proper coloring of elements ([b2c4ea1](https://github.com/KRTirtho/spotube/commit/b2c4ea13f6157c2b7bec3957e1f7f50fbf0002c7)) +- **player:** replace bg blur with gradient, proper fg color and align title and artist name ([159f03e](https://github.com/KRTirtho/spotube/commit/159f03e7ca62e6b3b86389e2795da84de61fba78)) +- playlist create support for generated playlist ([91c72f9](https://github.com/KRTirtho/spotube/commit/91c72f9ec9556f301c5d129fc82e19e791a02fbe)) +- playlist generation all parameters support ([9877d5f](https://github.com/KRTirtho/spotube/commit/9877d5f51736db03d5839dadf164d11d0cce82f0)) +- **playlist,album page:** play and shuffle take full width on smaller screens, add new xs breakpoint ([dce1b88](https://github.com/KRTirtho/spotube/commit/dce1b88694cfcb6b7e63d6ee614ac1dbbd017f6e)) +- **queue:** add track(s) for playing next ([#460](https://github.com/KRTirtho/spotube/issues/460)) ([cac8ea6](https://github.com/KRTirtho/spotube/commit/cac8ea638812f5d9cb4305144b6351141a2cf407)) +- **queue:** reorder tracks support ([441b43b](https://github.com/KRTirtho/spotube/commit/441b43bef6b92fd7df6c4e1bef39d67b4a76cd22)) +- re-designed playlist/album page ([0cedc7a](https://github.com/KRTirtho/spotube/commit/0cedc7a4187771efce8152003f890e242116c78c)) +- re-introduce youtube API along with piped ([b54ee96](https://github.com/KRTirtho/spotube/commit/b54ee96233b29d7517eba66e3f8dd9270c2790df)) +- reactive volume slider and slicker bottom bar with lowered height ([9d14517](https://github.com/KRTirtho/spotube/commit/9d14517202d5c9d993a947808bf0c6520ed54ea3)) +- remove SponsorBlock in favor of YT Music and remove pocketbase backend track support ([fb780da](https://github.com/KRTirtho/spotube/commit/fb780da327a213d7a82cbc3b567ece858dc2f0e8)) +- repeat button all 3 mode and disable player controls when track is fetching ([1418378](https://github.com/KRTirtho/spotube/commit/14183781dd3f1e16c121e78ad637a326de7b5dcf)) +- replace YouTube API with piped API ([1ecc36d](https://github.com/KRTirtho/spotube/commit/1ecc36da57af61fd9c2ca928589088cd4325f605)) +- responsive playlist generate page and scrollable multi autocomplete ([d57aad5](https://github.com/KRTirtho/spotube/commit/d57aad5612f7622dcd638ea8c0ec4d96f741de2b)) +- search alternative track source ([dfea195](https://github.com/KRTirtho/spotube/commit/dfea195ec178de733717cfe3226cede7521ee2d3)) +- setup localization (l10n) and language switcher, add sidebar and navbar locale ([f12d812](https://github.com/KRTirtho/spotube/commit/f12d81259f9e7005e681a7ca9867291d9228a8b1)) +- show album release year ([#387](https://github.com/KRTirtho/spotube/issues/387)) and fix layout of artist's album ([6a6ddf6](https://github.com/KRTirtho/spotube/commit/6a6ddf6e1f6dc72b794cae49adf8348da272babd)) +- show country code piped instance list ([60328a6](https://github.com/KRTirtho/spotube/commit/60328a6bafcbff1b7d0ee5099825f0e3d545b60f)) +- show loading when track metadata is being fetched, android, ios, macos enable shuffling ([bf59570](https://github.com/KRTirtho/spotube/commit/bf59570251720a80efe0aa6be481899864da5079)) +- sort tracks by newest and oldest dates ([b4713e3](https://github.com/KRTirtho/spotube/commit/b4713e377a938cbebe70089874216f86fe550c34)) +- supabase integration ([8bcce92](https://github.com/KRTirtho/spotube/commit/8bcce9282eae08c5996a27f16f89cbc187a06823)) +- system tray support ([#31](https://github.com/KRTirtho/spotube/issues/31)) ([06a0437](https://github.com/KRTirtho/spotube/commit/06a043764d9f65fb448fcf088ccf2737145e23e8)) +- track populate sibling support ([3aeb026](https://github.com/KRTirtho/spotube/commit/3aeb026776716b6e2eb89c8406a4996a86c7ca60)) +- **translation:** add hindi and french translations using ChatGPT ([6d836bd](https://github.com/KRTirtho/spotube/commit/6d836bdb658c180ca8e2c71e7e290fafa3520727)) +- **translation:** add Japanase locale ([4b52a71](https://github.com/KRTirtho/spotube/commit/4b52a71c0914bda6c831d8f637a5934f7bcf8fcb)) +- use system color scheme ([862c4b8](https://github.com/KRTirtho/spotube/commit/862c4b8faf2c751d803e373e29981a116bf08ed5)) +- volume slider in player page ([7abe2c1](https://github.com/KRTirtho/spotube/commit/7abe2c10735bc38c644487139557a731d25e80e6)) +- windows OS media control panel support ([f0b426a](https://github.com/KRTirtho/spotube/commit/f0b426ae89f2e01f4a9c8757ef4e0b4a21b50c7b)) ### Bug Fixes -* add to playlist dialog not showing playlist name ([8944581](https://github.com/KRTirtho/spotube/commit/8944581c09eec0162220e7ff684205484fafb599)) -* album sync not working ([74906f3](https://github.com/KRTirtho/spotube/commit/74906f393250934c36530a73ad7312f59f8627ed)) -* alternative track source not playing new source ([a9b5a71](https://github.com/KRTirtho/spotube/commit/a9b5a714e47d40407d799966ae95f84338f9b59a)) -* **android:** use multi assetAudioPlayer instance fix patch and disable Pre-download and play by default in Android too ([cdb3268](https://github.com/KRTirtho/spotube/commit/cdb32685e4bbb899706ed16d58ef9a3a074e283a)) -* **artist:** follower count shows as float when < 1000 ([#482](https://github.com/KRTirtho/spotube/issues/482)) ([fd1846e](https://github.com/KRTirtho/spotube/commit/fd1846eecf9632e59e4b70fb70e97c556b6374f5)) -* bottom navbar first item icon color not changing on primary color change ([6eb4244](https://github.com/KRTirtho/spotube/commit/6eb4244f3244a96fe6858261534cc03eb3de803c)) -* cached currently playing track infinite loading ([9401718](https://github.com/KRTirtho/spotube/commit/94017189c6b9bf55ec62cbf29cd6b0e9fffca42a)) -* cached queue tracks expired stream ([ed29ab5](https://github.com/KRTirtho/spotube/commit/ed29ab5137416d9fb2e7e9fe840f56ef52df6f61)) -* collection currently playing state persist on restart ([1c89e3e](https://github.com/KRTirtho/spotube/commit/1c89e3efb0f05c648fc1c8e09039e62333de18d1)) -* color not syncing and add new screenshot ([6205501](https://github.com/KRTirtho/spotube/commit/62055018feade0b895663a0bfc5f85f265ae2154)) -* content going below bottom player or nav bar ([1bdce9f](https://github.com/KRTirtho/spotube/commit/1bdce9fe964de88a667bb160846c11dc70b77c00)) -* disable background_downloader due to android build failures ([7d23bee](https://github.com/KRTirtho/spotube/commit/7d23beec5ef07c4d649185a69e7a2b9697dc6953)) -* disable play when loading track and buffering event ([30c933c](https://github.com/KRTirtho/spotube/commit/30c933cdf3d4524be164e171094afdd27b0252b7)) -* error log ([e3d8239](https://github.com/KRTirtho/spotube/commit/e3d8239b9f5700bfb17c4758d95ba1db1f0e718a)) -* excessive repaints caused by Player progress bar ([09b24cf](https://github.com/KRTirtho/spotube/commit/09b24cf1fd1644c549f85904545db54b39cc2431)) -* failed download no error icon ([1266a3f](https://github.com/KRTirtho/spotube/commit/1266a3f1607de11e793a294071850996527d494a)) -* **home:** bottom player transparency ([20c424c](https://github.com/KRTirtho/spotube/commit/20c424c77fe273a693213ebf88d50e4025bc8608)) -* language changer not working ([7b7b1f2](https://github.com/KRTirtho/spotube/commit/7b7b1f2647591b7cd4cc7841526716c6a2877e55)) -* less frequent position updates ([0a49b56](https://github.com/KRTirtho/spotube/commit/0a49b56566abd00cf7703e4207cfa90f93c381fd)) -* linux mpris not showing up and overall media notification service ([1abcad1](https://github.com/KRTirtho/spotube/commit/1abcad1de510c209a34196f2de17045af4dd3bc2)) -* local tracks getting fetched on first load ([73c012c](https://github.com/KRTirtho/spotube/commit/73c012c71ab5050636f79e010d654b4390978ee7)) -* local tracks not working when there's a invalid music file in the folder ([5855820](https://github.com/KRTirtho/spotube/commit/5855820569dfad7cd26f1e0f0c985babd0d9485d)) -* lyrics page blur in player and cut off text when line too big ([6b4584e](https://github.com/KRTirtho/spotube/commit/6b4584e91bd4f4aee0c56e48a7aec7015c7c418b)) -* macos build by removing media_kit native event loop ([62fc773](https://github.com/KRTirtho/spotube/commit/62fc7739b508f0e874978408a2bab0a1d422deb6)) -* macos build error, mobile player duration and playing state and background disposal of player ([be91e33](https://github.com/KRTirtho/spotube/commit/be91e33828630c7062886cd15e4d57496daaa4d5)) -* **macos,ios:** use regular shared prefs ([1b5bfec](https://github.com/KRTirtho/spotube/commit/1b5bfec27fbcfe9faabff64d46296bdeebe00161)) -* memoize child of animated widget and make player bg animation faster ([fcb5c8f](https://github.com/KRTirtho/spotube/commit/fcb5c8f8dabd0d4e3033f80ea3e5d006243cdfb5)) -* mini player not working in release mode ([28ff321](https://github.com/KRTirtho/spotube/commit/28ff3216efee81184798eedfbb10ba66395bbf36)) -* **mkPlayer:** remove method and wrong active index on modifying playlist ([3bafa7b](https://github.com/KRTirtho/spotube/commit/3bafa7b80c963fa52b90ed4cb1393fb121cac713)) -* mobile audio notification not working ([8f9303b](https://github.com/KRTirtho/spotube/commit/8f9303bc0fddb9d179303a1f0eb76dd5b02410e7)) -* multiple instance of theme ([4ec0424](https://github.com/KRTirtho/spotube/commit/4ec04240a5bde6af5c920a61ab6260e7a93bfc54)) -* navigation to settings not working ([ce10aa1](https://github.com/KRTirtho/spotube/commit/ce10aa1fe2c95d4738835687f613930cf7829f3a)) -* no progress update when track changed ([6ae8964](https://github.com/KRTirtho/spotube/commit/6ae896441a787fce1bc6e5eb5379856dc2f4e96d)) -* null exception on proxy playlist and audio player ([a455a89](https://github.com/KRTirtho/spotube/commit/a455a89c5861fd455f6950c7b68beae24bdcc6ed)) -* overflowing clickable artists links ([4077fac](https://github.com/KRTirtho/spotube/commit/4077fac39fb667b87e959e53d2dcaceefb63cd2d)) -* personalized playlists not loading ([caa3408](https://github.com/KRTirtho/spotube/commit/caa340803fdf3859fe5a8a996abae1502ef2e4e7)) -* playback not moving to next track after a track ends ([27e8acb](https://github.com/KRTirtho/spotube/commit/27e8acbfe75a37c0a8fa69a444fdd86e92dbe4f0)) -* **player:** gradient bg not taking full height ([62ad86e](https://github.com/KRTirtho/spotube/commit/62ad86e88d74b5114af78138b221314192e5a801)) -* **player:** playback element placement ([5e47faa](https://github.com/KRTirtho/spotube/commit/5e47faa6060d7a8aa0d143060e812dc06b8dd790)) -* **player:** queue button not showing when not logged in ([6c2d655](https://github.com/KRTirtho/spotube/commit/6c2d65587b0e6e167be1d0b086df103c7e72d4b2)) -* **player:** volume slider, prefetching of media_kit and stuttering on sponsorblock skip ([1f32554](https://github.com/KRTirtho/spotube/commit/1f3255481f058c50968561db88172e56b58494f4)) -* playlist generate slider shape ([2b35c04](https://github.com/KRTirtho/spotube/commit/2b35c044adb15a97a58692e7880694a251899732)) -* pop sheet list not scrollable ([cca5625](https://github.com/KRTirtho/spotube/commit/cca5625df7e432da8581a4504306baad154deb48)) -* re-enable add to queue and play next support, favorite button query exceptions ([e529c79](https://github.com/KRTirtho/spotube/commit/e529c79c4f0cd964b7d89e010d3fe51378ea7222)) -* re-enable download manager ([ea45c4f](https://github.com/KRTirtho/spotube/commit/ea45c4f42ae89b8991e470e84a5290b3be3b0f36)) -* remove unnecessary broadcast stream conversions ([bf04962](https://github.com/KRTirtho/spotube/commit/bf04962e90ea5345a2bc0d4793999f7db712cab2)) -* remove useBreakpoints as it clogs up memory with unnecessary state updates ([e1c0f5c](https://github.com/KRTirtho/spotube/commit/e1c0f5cf1e4cece2c4aa235bfbf8511ad7b1fe59)) -* replace download multiple pops and add translations ([4a21249](https://github.com/KRTirtho/spotube/commit/4a21249ee386426b0974451542a93e84f532fb3f)) -* screen breakpoints and persist lyrics delay across screens ([df79638](https://github.com/KRTirtho/spotube/commit/df79638fb622a55aaa2b36c9a1425c2d9c4a8e52)) -* sidebar task counter badge and bottom player play button progress color ([af278d8](https://github.com/KRTirtho/spotube/commit/af278d8feaa08c14528627766bce6d724b846954)) -* status bar color of playlist/album page ([65fa3cb](https://github.com/KRTirtho/spotube/commit/65fa3cb624c240360de5a06778a1f72ad10bbe2d)) -* system color scheme not persisting on restart when system color scheme changed ([e04515d](https://github.com/KRTirtho/spotube/commit/e04515d8e213b4c7f85d11385959a33b042bd9b1)) -* track collection view status bar not transparent ([9251121](https://github.com/KRTirtho/spotube/commit/9251121ba0154599975e33819a43719477c644f8)) -* track doesn't play after change ([17e5ab6](https://github.com/KRTirtho/spotube/commit/17e5ab611cc417cce7c17cafc7045f5aa2eb970e)) -* track stops at last second ([f554f6d](https://github.com/KRTirtho/spotube/commit/f554f6d43bb714f662a27977f501d7ad44b070c3)) -* **track_collection_view:** keyboard focus on scroll and no space for search results in playlist/album ([7a8bd92](https://github.com/KRTirtho/spotube/commit/7a8bd921047e3766dbbf24449e2873afe3dbecf8)) -* track_table_view table headers ([d88d287](https://github.com/KRTirtho/spotube/commit/d88d287fc586ec33351de9f3b4359f189054868b)) -* track_tile active and blacklist color, playbutton card action positioning ([3f5a1b9](https://github.com/KRTirtho/spotube/commit/3f5a1b9587efe9b7b2c69008345867933f79ec67)) -* use id based source getters instead of index ([a074463](https://github.com/KRTirtho/spotube/commit/a0744630ba2dc713babdb8db6500f9dd0f1e6096)) +- add to playlist dialog not showing playlist name ([8944581](https://github.com/KRTirtho/spotube/commit/8944581c09eec0162220e7ff684205484fafb599)) +- album sync not working ([74906f3](https://github.com/KRTirtho/spotube/commit/74906f393250934c36530a73ad7312f59f8627ed)) +- alternative track source not playing new source ([a9b5a71](https://github.com/KRTirtho/spotube/commit/a9b5a714e47d40407d799966ae95f84338f9b59a)) +- **android:** use multi assetAudioPlayer instance fix patch and disable Pre-download and play by default in Android too ([cdb3268](https://github.com/KRTirtho/spotube/commit/cdb32685e4bbb899706ed16d58ef9a3a074e283a)) +- **artist:** follower count shows as float when < 1000 ([#482](https://github.com/KRTirtho/spotube/issues/482)) ([fd1846e](https://github.com/KRTirtho/spotube/commit/fd1846eecf9632e59e4b70fb70e97c556b6374f5)) +- bottom navbar first item icon color not changing on primary color change ([6eb4244](https://github.com/KRTirtho/spotube/commit/6eb4244f3244a96fe6858261534cc03eb3de803c)) +- cached currently playing track infinite loading ([9401718](https://github.com/KRTirtho/spotube/commit/94017189c6b9bf55ec62cbf29cd6b0e9fffca42a)) +- cached queue tracks expired stream ([ed29ab5](https://github.com/KRTirtho/spotube/commit/ed29ab5137416d9fb2e7e9fe840f56ef52df6f61)) +- collection currently playing state persist on restart ([1c89e3e](https://github.com/KRTirtho/spotube/commit/1c89e3efb0f05c648fc1c8e09039e62333de18d1)) +- color not syncing and add new screenshot ([6205501](https://github.com/KRTirtho/spotube/commit/62055018feade0b895663a0bfc5f85f265ae2154)) +- content going below bottom player or nav bar ([1bdce9f](https://github.com/KRTirtho/spotube/commit/1bdce9fe964de88a667bb160846c11dc70b77c00)) +- disable background_downloader due to android build failures ([7d23bee](https://github.com/KRTirtho/spotube/commit/7d23beec5ef07c4d649185a69e7a2b9697dc6953)) +- disable play when loading track and buffering event ([30c933c](https://github.com/KRTirtho/spotube/commit/30c933cdf3d4524be164e171094afdd27b0252b7)) +- error log ([e3d8239](https://github.com/KRTirtho/spotube/commit/e3d8239b9f5700bfb17c4758d95ba1db1f0e718a)) +- excessive repaints caused by Player progress bar ([09b24cf](https://github.com/KRTirtho/spotube/commit/09b24cf1fd1644c549f85904545db54b39cc2431)) +- failed download no error icon ([1266a3f](https://github.com/KRTirtho/spotube/commit/1266a3f1607de11e793a294071850996527d494a)) +- **home:** bottom player transparency ([20c424c](https://github.com/KRTirtho/spotube/commit/20c424c77fe273a693213ebf88d50e4025bc8608)) +- language changer not working ([7b7b1f2](https://github.com/KRTirtho/spotube/commit/7b7b1f2647591b7cd4cc7841526716c6a2877e55)) +- less frequent position updates ([0a49b56](https://github.com/KRTirtho/spotube/commit/0a49b56566abd00cf7703e4207cfa90f93c381fd)) +- linux mpris not showing up and overall media notification service ([1abcad1](https://github.com/KRTirtho/spotube/commit/1abcad1de510c209a34196f2de17045af4dd3bc2)) +- local tracks getting fetched on first load ([73c012c](https://github.com/KRTirtho/spotube/commit/73c012c71ab5050636f79e010d654b4390978ee7)) +- local tracks not working when there's a invalid music file in the folder ([5855820](https://github.com/KRTirtho/spotube/commit/5855820569dfad7cd26f1e0f0c985babd0d9485d)) +- lyrics page blur in player and cut off text when line too big ([6b4584e](https://github.com/KRTirtho/spotube/commit/6b4584e91bd4f4aee0c56e48a7aec7015c7c418b)) +- macos build by removing media_kit native event loop ([62fc773](https://github.com/KRTirtho/spotube/commit/62fc7739b508f0e874978408a2bab0a1d422deb6)) +- macos build error, mobile player duration and playing state and background disposal of player ([be91e33](https://github.com/KRTirtho/spotube/commit/be91e33828630c7062886cd15e4d57496daaa4d5)) +- **macos,ios:** use regular shared prefs ([1b5bfec](https://github.com/KRTirtho/spotube/commit/1b5bfec27fbcfe9faabff64d46296bdeebe00161)) +- memoize child of animated widget and make player bg animation faster ([fcb5c8f](https://github.com/KRTirtho/spotube/commit/fcb5c8f8dabd0d4e3033f80ea3e5d006243cdfb5)) +- mini player not working in release mode ([28ff321](https://github.com/KRTirtho/spotube/commit/28ff3216efee81184798eedfbb10ba66395bbf36)) +- **mkPlayer:** remove method and wrong active index on modifying playlist ([3bafa7b](https://github.com/KRTirtho/spotube/commit/3bafa7b80c963fa52b90ed4cb1393fb121cac713)) +- mobile audio notification not working ([8f9303b](https://github.com/KRTirtho/spotube/commit/8f9303bc0fddb9d179303a1f0eb76dd5b02410e7)) +- multiple instance of theme ([4ec0424](https://github.com/KRTirtho/spotube/commit/4ec04240a5bde6af5c920a61ab6260e7a93bfc54)) +- navigation to settings not working ([ce10aa1](https://github.com/KRTirtho/spotube/commit/ce10aa1fe2c95d4738835687f613930cf7829f3a)) +- no progress update when track changed ([6ae8964](https://github.com/KRTirtho/spotube/commit/6ae896441a787fce1bc6e5eb5379856dc2f4e96d)) +- null exception on proxy playlist and audio player ([a455a89](https://github.com/KRTirtho/spotube/commit/a455a89c5861fd455f6950c7b68beae24bdcc6ed)) +- overflowing clickable artists links ([4077fac](https://github.com/KRTirtho/spotube/commit/4077fac39fb667b87e959e53d2dcaceefb63cd2d)) +- personalized playlists not loading ([caa3408](https://github.com/KRTirtho/spotube/commit/caa340803fdf3859fe5a8a996abae1502ef2e4e7)) +- playback not moving to next track after a track ends ([27e8acb](https://github.com/KRTirtho/spotube/commit/27e8acbfe75a37c0a8fa69a444fdd86e92dbe4f0)) +- **player:** gradient bg not taking full height ([62ad86e](https://github.com/KRTirtho/spotube/commit/62ad86e88d74b5114af78138b221314192e5a801)) +- **player:** playback element placement ([5e47faa](https://github.com/KRTirtho/spotube/commit/5e47faa6060d7a8aa0d143060e812dc06b8dd790)) +- **player:** queue button not showing when not logged in ([6c2d655](https://github.com/KRTirtho/spotube/commit/6c2d65587b0e6e167be1d0b086df103c7e72d4b2)) +- **player:** volume slider, prefetching of media_kit and stuttering on sponsorblock skip ([1f32554](https://github.com/KRTirtho/spotube/commit/1f3255481f058c50968561db88172e56b58494f4)) +- playlist generate slider shape ([2b35c04](https://github.com/KRTirtho/spotube/commit/2b35c044adb15a97a58692e7880694a251899732)) +- pop sheet list not scrollable ([cca5625](https://github.com/KRTirtho/spotube/commit/cca5625df7e432da8581a4504306baad154deb48)) +- re-enable add to queue and play next support, favorite button query exceptions ([e529c79](https://github.com/KRTirtho/spotube/commit/e529c79c4f0cd964b7d89e010d3fe51378ea7222)) +- re-enable download manager ([ea45c4f](https://github.com/KRTirtho/spotube/commit/ea45c4f42ae89b8991e470e84a5290b3be3b0f36)) +- remove unnecessary broadcast stream conversions ([bf04962](https://github.com/KRTirtho/spotube/commit/bf04962e90ea5345a2bc0d4793999f7db712cab2)) +- remove useBreakpoints as it clogs up memory with unnecessary state updates ([e1c0f5c](https://github.com/KRTirtho/spotube/commit/e1c0f5cf1e4cece2c4aa235bfbf8511ad7b1fe59)) +- replace download multiple pops and add translations ([4a21249](https://github.com/KRTirtho/spotube/commit/4a21249ee386426b0974451542a93e84f532fb3f)) +- screen breakpoints and persist lyrics delay across screens ([df79638](https://github.com/KRTirtho/spotube/commit/df79638fb622a55aaa2b36c9a1425c2d9c4a8e52)) +- sidebar task counter badge and bottom player play button progress color ([af278d8](https://github.com/KRTirtho/spotube/commit/af278d8feaa08c14528627766bce6d724b846954)) +- status bar color of playlist/album page ([65fa3cb](https://github.com/KRTirtho/spotube/commit/65fa3cb624c240360de5a06778a1f72ad10bbe2d)) +- system color scheme not persisting on restart when system color scheme changed ([e04515d](https://github.com/KRTirtho/spotube/commit/e04515d8e213b4c7f85d11385959a33b042bd9b1)) +- track collection view status bar not transparent ([9251121](https://github.com/KRTirtho/spotube/commit/9251121ba0154599975e33819a43719477c644f8)) +- track doesn't play after change ([17e5ab6](https://github.com/KRTirtho/spotube/commit/17e5ab611cc417cce7c17cafc7045f5aa2eb970e)) +- track stops at last second ([f554f6d](https://github.com/KRTirtho/spotube/commit/f554f6d43bb714f662a27977f501d7ad44b070c3)) +- **track_collection_view:** keyboard focus on scroll and no space for search results in playlist/album ([7a8bd92](https://github.com/KRTirtho/spotube/commit/7a8bd921047e3766dbbf24449e2873afe3dbecf8)) +- track_table_view table headers ([d88d287](https://github.com/KRTirtho/spotube/commit/d88d287fc586ec33351de9f3b4359f189054868b)) +- track_tile active and blacklist color, playbutton card action positioning ([3f5a1b9](https://github.com/KRTirtho/spotube/commit/3f5a1b9587efe9b7b2c69008345867933f79ec67)) +- use id based source getters instead of index ([a074463](https://github.com/KRTirtho/spotube/commit/a0744630ba2dc713babdb8db6500f9dd0f1e6096)) ### [2.7.1](https://github.com/KRTirtho/spotube/compare/v2.7.0...v2.7.1) (2023-04-10) - ### Bug Fixes -* fallback for lyrics when anonymous ([f160ec7](https://github.com/KRTirtho/spotube/commit/f160ec767d9941d33f83aba1752b28df629d0e10)) -* **android:** audio notification stuck in play state ([448c9b3](https://github.com/KRTirtho/spotube/commit/448c9b39f407668ad92a695afe3c9741baeca20d)) -* **macos:** crashing on startup ([c46b428](https://github.com/KRTirtho/spotube/commit/c46b4284b1d46a614cbcebc8c2f2e52714921b9b)) -* spotify query hooks overriding default query params ([ec9a02e](https://github.com/KRTirtho/spotube/commit/ec9a02e8b8d988e15ed58027054d2a9090d98873)) +- fallback for lyrics when anonymous ([f160ec7](https://github.com/KRTirtho/spotube/commit/f160ec767d9941d33f83aba1752b28df629d0e10)) +- **android:** audio notification stuck in play state ([448c9b3](https://github.com/KRTirtho/spotube/commit/448c9b39f407668ad92a695afe3c9741baeca20d)) +- **macos:** crashing on startup ([c46b428](https://github.com/KRTirtho/spotube/commit/c46b4284b1d46a614cbcebc8c2f2e52714921b9b)) +- spotify query hooks overriding default query params ([ec9a02e](https://github.com/KRTirtho/spotube/commit/ec9a02e8b8d988e15ed58027054d2a9090d98873)) ## [2.7.0](https://github.com/KRTirtho/spotube/compare/v2.6.0...v2.7.0) (2023-03-07) - ### Features -* add or remove track, playlist or album to queue support ([b8f3493](https://github.com/KRTirtho/spotube/commit/b8f3493138a9acd91d19efe67cfd1c0c7c269ae6)) -* basic command line argument support ([025c1ae](https://github.com/KRTirtho/spotube/commit/025c1ae20461c2ac9124b3ef41e21ff01f100498)) -* black list artist or track ([947c143](https://github.com/KRTirtho/spotube/commit/947c14353e15227400a6310673f3b850b2ff024f)) -* bring pre download on desktop, disable pre download for long videos ([1d82bb0](https://github.com/KRTirtho/spotube/commit/1d82bb098717c7321d3e338f071c7661987fc3be)) -* category/genre filter ([1dfec05](https://github.com/KRTirtho/spotube/commit/1dfec05eec7ee60cc9f6a3a97af37aef112063f1)) -* centralized icon collection with new icon set and nav bar labels hidden ([e7acb9e](https://github.com/KRTirtho/spotube/commit/e7acb9ed5cb02826b8da559818f1fccfcf7f143c)) -* compact search bar for genres and user_local_tracks page ([c343ccc](https://github.com/KRTirtho/spotube/commit/c343ccc2932868e3c1205d8cc625a9dfe9d78707)) -* compatibility with fl-query nextPage method change ([7617439](https://github.com/KRTirtho/spotube/commit/761743991520609dd2b2dcb12cd6e4e75a8f6925)) -* configure pocketbase, generate dart types, update playback to use server instead of hive cache ([ad90c11](https://github.com/KRTirtho/spotube/commit/ad90c11ab0c9f1aaba9ae9226d6076ea590f1a29)) -* failsafe pocketbase requests, removal of unneeded preferences options & vertical playbutton actions ([d68d150](https://github.com/KRTirtho/spotube/commit/d68d150d3f42260f889d86927378c2f746bb6993)) -* **home:** personalized section ([9080441](https://github.com/KRTirtho/spotube/commit/9080441b875ceb91260bbad79291365a98d5be95)) -* individual shuffle and repeat/loop button of player ([f79223c](https://github.com/KRTirtho/spotube/commit/f79223cd41c61d9836d25e7bc2811c6515ba00c8)) -* **lyrics:** use official spotify API for fetching lyrics and add zoom controls ([10d0660](https://github.com/KRTirtho/spotube/commit/10d0660972f008df0d11c280b681ce3b78f05d0b)) -* **mobile:** pull to refresh support in all refreshable list views ([9f959ce](https://github.com/KRTirtho/spotube/commit/9f959ce77cd95cfc34d01af1f5cf53dd4206b6a6)) -* new logo and compact search in playlist/album in mobile ([dc96cb3](https://github.com/KRTirtho/spotube/commit/dc96cb38cea8dc13738083f4850d22792d071019)) -* search/filter tracks inside playlist or album ([a06cd0d](https://github.com/KRTirtho/spotube/commit/a06cd0da84cc03a2a7cadbc80d70556cb0cf9310)) -* show snackbar on adding playlist or tracks to queue ([6bc1d32](https://github.com/KRTirtho/spotube/commit/6bc1d32a88ae516f77d149b83bcd536d2c888513)) -* **theme:** use material3 monet for colors and remove background color preference ([60ede5f](https://github.com/KRTirtho/spotube/commit/60ede5f92b732691d53850290d9667435298a857)) -* use catcher to handle exceptions ([84d94b0](https://github.com/KRTirtho/spotube/commit/84d94b05bc269a1676a261df2b12e508e10e4c0e)) -* use typed assets instead of hard coded paths ([59561ab](https://github.com/KRTirtho/spotube/commit/59561abdc2540576fc95b34b3b55def63567000a)) -* user local tracks searchbar ([e7f3f4e](https://github.com/KRTirtho/spotube/commit/e7f3f4eae49fe27a52fc3866fa4f6f2efb2aa479)) -* **user-library:** filtering support for user albums and user artists ([0b58155](https://github.com/KRTirtho/spotube/commit/0b58155d434f2de6359be77d7beee4484dbb7b2a)) -* **user-library:** search for user playlists ([af4d56f](https://github.com/KRTirtho/spotube/commit/af4d56fd41e57cbe6d87883e87e6b4469aaba52f)) - +- add or remove track, playlist or album to queue support ([b8f3493](https://github.com/KRTirtho/spotube/commit/b8f3493138a9acd91d19efe67cfd1c0c7c269ae6)) +- basic command line argument support ([025c1ae](https://github.com/KRTirtho/spotube/commit/025c1ae20461c2ac9124b3ef41e21ff01f100498)) +- black list artist or track ([947c143](https://github.com/KRTirtho/spotube/commit/947c14353e15227400a6310673f3b850b2ff024f)) +- bring pre download on desktop, disable pre download for long videos ([1d82bb0](https://github.com/KRTirtho/spotube/commit/1d82bb098717c7321d3e338f071c7661987fc3be)) +- category/genre filter ([1dfec05](https://github.com/KRTirtho/spotube/commit/1dfec05eec7ee60cc9f6a3a97af37aef112063f1)) +- centralized icon collection with new icon set and nav bar labels hidden ([e7acb9e](https://github.com/KRTirtho/spotube/commit/e7acb9ed5cb02826b8da559818f1fccfcf7f143c)) +- compact search bar for genres and user_local_tracks page ([c343ccc](https://github.com/KRTirtho/spotube/commit/c343ccc2932868e3c1205d8cc625a9dfe9d78707)) +- compatibility with fl-query nextPage method change ([7617439](https://github.com/KRTirtho/spotube/commit/761743991520609dd2b2dcb12cd6e4e75a8f6925)) +- configure pocketbase, generate dart types, update playback to use server instead of hive cache ([ad90c11](https://github.com/KRTirtho/spotube/commit/ad90c11ab0c9f1aaba9ae9226d6076ea590f1a29)) +- failsafe pocketbase requests, removal of unneeded preferences options & vertical playbutton actions ([d68d150](https://github.com/KRTirtho/spotube/commit/d68d150d3f42260f889d86927378c2f746bb6993)) +- **home:** personalized section ([9080441](https://github.com/KRTirtho/spotube/commit/9080441b875ceb91260bbad79291365a98d5be95)) +- individual shuffle and repeat/loop button of player ([f79223c](https://github.com/KRTirtho/spotube/commit/f79223cd41c61d9836d25e7bc2811c6515ba00c8)) +- **lyrics:** use official spotify API for fetching lyrics and add zoom controls ([10d0660](https://github.com/KRTirtho/spotube/commit/10d0660972f008df0d11c280b681ce3b78f05d0b)) +- **mobile:** pull to refresh support in all refreshable list views ([9f959ce](https://github.com/KRTirtho/spotube/commit/9f959ce77cd95cfc34d01af1f5cf53dd4206b6a6)) +- new logo and compact search in playlist/album in mobile ([dc96cb3](https://github.com/KRTirtho/spotube/commit/dc96cb38cea8dc13738083f4850d22792d071019)) +- search/filter tracks inside playlist or album ([a06cd0d](https://github.com/KRTirtho/spotube/commit/a06cd0da84cc03a2a7cadbc80d70556cb0cf9310)) +- show snackbar on adding playlist or tracks to queue ([6bc1d32](https://github.com/KRTirtho/spotube/commit/6bc1d32a88ae516f77d149b83bcd536d2c888513)) +- **theme:** use material3 monet for colors and remove background color preference ([60ede5f](https://github.com/KRTirtho/spotube/commit/60ede5f92b732691d53850290d9667435298a857)) +- use catcher to handle exceptions ([84d94b0](https://github.com/KRTirtho/spotube/commit/84d94b05bc269a1676a261df2b12e508e10e4c0e)) +- use typed assets instead of hard coded paths ([59561ab](https://github.com/KRTirtho/spotube/commit/59561abdc2540576fc95b34b3b55def63567000a)) +- user local tracks searchbar ([e7f3f4e](https://github.com/KRTirtho/spotube/commit/e7f3f4eae49fe27a52fc3866fa4f6f2efb2aa479)) +- **user-library:** filtering support for user albums and user artists ([0b58155](https://github.com/KRTirtho/spotube/commit/0b58155d434f2de6359be77d7beee4484dbb7b2a)) +- **user-library:** search for user playlists ([af4d56f](https://github.com/KRTirtho/spotube/commit/af4d56fd41e57cbe6d87883e87e6b4469aaba52f)) ### Bug Fixes -* **about:** license text hidden in the bottom of smaller screen devices ([e158dd0](https://github.com/KRTirtho/spotube/commit/e158dd0cec5657e495b538e86c412b06974a9f49)) -* **about:** wrong link of License ([a4a7f1a](https://github.com/KRTirtho/spotube/commit/a4a7f1a74f9df82927403ca93aec508a13315ae8)) -* genre and sidebar user logo not loading ([710f172](https://github.com/KRTirtho/spotube/commit/710f172dee45f60ed3e5ed83017eb538d6a626bf)) -* lyrics modal sheet out of safe area so use 80% of screen height instead of full ([3db28f4](https://github.com/KRTirtho/spotube/commit/3db28f43b4200d03f7758e8c395d8430e0f89333)) -* lyrics not changing on track change ([c809d2d](https://github.com/KRTirtho/spotube/commit/c809d2daba4beaea7c4f16c6bb0edef9efa825b8)) -* lyrics not refetching when tracked changed while being in another page and sidebar user avatar not showing on startup ([bd12675](https://github.com/KRTirtho/spotube/commit/bd126751e9594fbc926bbcad7b9a2c577fce074a)) -* macOS logo placement ([c6a5d5f](https://github.com/KRTirtho/spotube/commit/c6a5d5f7b1b1fad3a0b5e63c02c847a149e72efe)) -* mobile track collection search bar position and page_window_title_bar exception on mobile platforms ([d0aaa97](https://github.com/KRTirtho/spotube/commit/d0aaa971fe358b9cb5dc7a35cc82eaf6520f7ab4)) -* **play_overlay:** show progress indicator on song loading ([7803a48](https://github.com/KRTirtho/spotube/commit/7803a48237c91f2a57bcc86fbd30ad879142c8ff)) -* **playback:** not skipping track's sponsorblock segments ([60a5847](https://github.com/KRTirtho/spotube/commit/60a5847ae68836bbbeef748254c674c81fa5c3ea)) -* playbutton card play state not changing ([ee46d09](https://github.com/KRTirtho/spotube/commit/ee46d0970be9e227793494a41e25c0c469847cd0)) -* **playbutton_card:** play and add to queue needs 2 clicks work ([bdd7098](https://github.com/KRTirtho/spotube/commit/bdd70984e6670813e508786e74cd2ea4a1fe1d53)) -* **playbutton_card:** play and non play state correction ([b327ffb](https://github.com/KRTirtho/spotube/commit/b327ffb1084b43e5c78e13994f65fb30b3a7e67e)) -* **playbutton_card:** title text overflow ([39ee0a9](https://github.com/KRTirtho/spotube/commit/39ee0a92a8f3d74d243db206fe034330f75c0588)) -* **playbutton:** playing state is not updating when playlist is actually playing ([9bad8c9](https://github.com/KRTirtho/spotube/commit/9bad8c9eb88f7c91091a669b642b92474df0f128)) -* **player_queue:** large clear button and macos exception ([0e43504](https://github.com/KRTirtho/spotube/commit/0e43504e18d2315fb1b7975b67bd2c596cbfb1bc)) -* **playlist_queue:** load method not preserving the active track before filtering blacklisted tracks ([42b3e11](https://github.com/KRTirtho/spotube/commit/42b3e111f844f6de6a145de2760ccfd7e97e623b)) -* pre downloading not working properly, audio service circular deps and sibling not loading for backend track ([3ccb525](https://github.com/KRTirtho/spotube/commit/3ccb525260a83ba54021a353b15ed3cda6e9c876)) -* search track play button isn't working ([0751f5e](https://github.com/KRTirtho/spotube/commit/0751f5e3173882f3aeed67027854e5054b689693)) -* **search:** grey screen, only tracks update on new search string, playlists,albums,artists show up before hitting return/submit ([a774817](https://github.com/KRTirtho/spotube/commit/a774817240ef813cb95f82f53ccb798ef9acb51d)) -* **search:** has to submit twice for search results ([f5dc76a](https://github.com/KRTirtho/spotube/commit/f5dc76a98f55f0f032a6fe4208465899f932355a)) -* titlebar maximize+restore button not working and less responsive title bar buttons ([8a6ba3b](https://github.com/KRTirtho/spotube/commit/8a6ba3b35f0b6b42cf60920e945ac2065c886ecb)) -* **track_collection_view:** hide search bar when sliver is collapsed ([3d6d244](https://github.com/KRTirtho/spotube/commit/3d6d2444beed153a2b6663d6153684b2974f4152)) -* **track_tile:** cannot see track index above 99 ([78b3273](https://github.com/KRTirtho/spotube/commit/78b3273e441cdaa6d4a410ddfe29837dc1aa7000)) -* **track_tile:** track action popup not showing on narrow screens ([0c54f2d](https://github.com/KRTirtho/spotube/commit/0c54f2dcd4474b63db4c517b0e7332cbd3ab51e9)) -* **ui:** scaffold exception in fluent_ui ([8ce2192](https://github.com/KRTirtho/spotube/commit/8ce2192e5cb08e3a8be5ead510ab35b274bef2ef)) -* use chosen market for new release ([c6bf9b6](https://github.com/KRTirtho/spotube/commit/c6bf9b67995161a8bf7c3782188d01e8859c18e9)) +- **about:** license text hidden in the bottom of smaller screen devices ([e158dd0](https://github.com/KRTirtho/spotube/commit/e158dd0cec5657e495b538e86c412b06974a9f49)) +- **about:** wrong link of License ([a4a7f1a](https://github.com/KRTirtho/spotube/commit/a4a7f1a74f9df82927403ca93aec508a13315ae8)) +- genre and sidebar user logo not loading ([710f172](https://github.com/KRTirtho/spotube/commit/710f172dee45f60ed3e5ed83017eb538d6a626bf)) +- lyrics modal sheet out of safe area so use 80% of screen height instead of full ([3db28f4](https://github.com/KRTirtho/spotube/commit/3db28f43b4200d03f7758e8c395d8430e0f89333)) +- lyrics not changing on track change ([c809d2d](https://github.com/KRTirtho/spotube/commit/c809d2daba4beaea7c4f16c6bb0edef9efa825b8)) +- lyrics not refetching when tracked changed while being in another page and sidebar user avatar not showing on startup ([bd12675](https://github.com/KRTirtho/spotube/commit/bd126751e9594fbc926bbcad7b9a2c577fce074a)) +- macOS logo placement ([c6a5d5f](https://github.com/KRTirtho/spotube/commit/c6a5d5f7b1b1fad3a0b5e63c02c847a149e72efe)) +- mobile track collection search bar position and page_window_title_bar exception on mobile platforms ([d0aaa97](https://github.com/KRTirtho/spotube/commit/d0aaa971fe358b9cb5dc7a35cc82eaf6520f7ab4)) +- **play_overlay:** show progress indicator on song loading ([7803a48](https://github.com/KRTirtho/spotube/commit/7803a48237c91f2a57bcc86fbd30ad879142c8ff)) +- **playback:** not skipping track's sponsorblock segments ([60a5847](https://github.com/KRTirtho/spotube/commit/60a5847ae68836bbbeef748254c674c81fa5c3ea)) +- playbutton card play state not changing ([ee46d09](https://github.com/KRTirtho/spotube/commit/ee46d0970be9e227793494a41e25c0c469847cd0)) +- **playbutton_card:** play and add to queue needs 2 clicks work ([bdd7098](https://github.com/KRTirtho/spotube/commit/bdd70984e6670813e508786e74cd2ea4a1fe1d53)) +- **playbutton_card:** play and non play state correction ([b327ffb](https://github.com/KRTirtho/spotube/commit/b327ffb1084b43e5c78e13994f65fb30b3a7e67e)) +- **playbutton_card:** title text overflow ([39ee0a9](https://github.com/KRTirtho/spotube/commit/39ee0a92a8f3d74d243db206fe034330f75c0588)) +- **playbutton:** playing state is not updating when playlist is actually playing ([9bad8c9](https://github.com/KRTirtho/spotube/commit/9bad8c9eb88f7c91091a669b642b92474df0f128)) +- **player_queue:** large clear button and macos exception ([0e43504](https://github.com/KRTirtho/spotube/commit/0e43504e18d2315fb1b7975b67bd2c596cbfb1bc)) +- **playlist_queue:** load method not preserving the active track before filtering blacklisted tracks ([42b3e11](https://github.com/KRTirtho/spotube/commit/42b3e111f844f6de6a145de2760ccfd7e97e623b)) +- pre downloading not working properly, audio service circular deps and sibling not loading for backend track ([3ccb525](https://github.com/KRTirtho/spotube/commit/3ccb525260a83ba54021a353b15ed3cda6e9c876)) +- search track play button isn't working ([0751f5e](https://github.com/KRTirtho/spotube/commit/0751f5e3173882f3aeed67027854e5054b689693)) +- **search:** grey screen, only tracks update on new search string, playlists,albums,artists show up before hitting return/submit ([a774817](https://github.com/KRTirtho/spotube/commit/a774817240ef813cb95f82f53ccb798ef9acb51d)) +- **search:** has to submit twice for search results ([f5dc76a](https://github.com/KRTirtho/spotube/commit/f5dc76a98f55f0f032a6fe4208465899f932355a)) +- titlebar maximize+restore button not working and less responsive title bar buttons ([8a6ba3b](https://github.com/KRTirtho/spotube/commit/8a6ba3b35f0b6b42cf60920e945ac2065c886ecb)) +- **track_collection_view:** hide search bar when sliver is collapsed ([3d6d244](https://github.com/KRTirtho/spotube/commit/3d6d2444beed153a2b6663d6153684b2974f4152)) +- **track_tile:** cannot see track index above 99 ([78b3273](https://github.com/KRTirtho/spotube/commit/78b3273e441cdaa6d4a410ddfe29837dc1aa7000)) +- **track_tile:** track action popup not showing on narrow screens ([0c54f2d](https://github.com/KRTirtho/spotube/commit/0c54f2dcd4474b63db4c517b0e7332cbd3ab51e9)) +- **ui:** scaffold exception in fluent_ui ([8ce2192](https://github.com/KRTirtho/spotube/commit/8ce2192e5cb08e3a8be5ead510ab35b274bef2ef)) +- use chosen market for new release ([c6bf9b6](https://github.com/KRTirtho/spotube/commit/c6bf9b67995161a8bf7c3782188d01e8859c18e9)) ## [2.6.0](https://github.com/KRTirtho/spotube/compare/v2.5.0...v2.6.0) (2022-12-09) - ### Features -* add selected tracks to playlists, optimistic playlist remove track ([3386dac](https://github.com/KRTirtho/spotube/commit/3386dac78ee49b9e3504f5c05bf1e7b362a2e8a2)) -* added shuffle button in playlist and album section ([1fad95f](https://github.com/KRTirtho/spotube/commit/1fad95f6e370606a9faf6f2bb9738dc360e23918)) -* **android-playback:** option to download track bytes and play instead of Streaming ([dcc8ba5](https://github.com/KRTirtho/spotube/commit/dcc8ba5a54286b252d53c9c14918971bf7bea8cc)) -* change default platform option and platform specific back button ([36c5e02](https://github.com/KRTirtho/spotube/commit/36c5e02f18374100f61cc3f2957c27bfc0d8511f)) -* dialog logo for macos, settings more width for country picker ([5e96913](https://github.com/KRTirtho/spotube/commit/5e96913ba34230e7a15d62060d5dae28d80e3630)) -* initial platform_ui integration ([9eee573](https://github.com/KRTirtho/spotube/commit/9eee573ce928aa6c03dcb50bf1521350d2de32cc)) -* libadwaita theming, track tile and PlayButtonCard play button icon fix ([e795e23](https://github.com/KRTirtho/spotube/commit/e795e23e42e5f1f832963b2a4506df89b7df5baa)) -* **lyrics:** tabs for both synced and static lyrics [#182](https://github.com/KRTirtho/spotube/issues/182) ([6b6907a](https://github.com/KRTirtho/spotube/commit/6b6907af3fdb327312ad4dd9e16b3e2a850ed896)) -* new refined about page, update checker only check for same update channel ([4cadfa9](https://github.com/KRTirtho/spotube/commit/4cadfa93750cc9b3f8fbe7b60f8161a77d2a12f6)) -* pause track when seeking forward/back and keep audio session alive when paused/interrupted ([bc8a04e](https://github.com/KRTirtho/spotube/commit/bc8a04e5442ba3abe1b04ab325769559f37d9802)) -* platform bottom navigation bar add ([ff14469](https://github.com/KRTirtho/spotube/commit/ff1446982f0260c6fe231970aa6ed61c273fdf07)) -* platform slider and progress indicator integration ([46b00ba](https://github.com/KRTirtho/spotube/commit/46b00bafdf71a4add61cf50168a32198c3293181)) -* platform title bar buttons add ([54048cb](https://github.com/KRTirtho/spotube/commit/54048cbfc37d9a40c020eb81ab67b9dd5428f0d7)) -* **playback:** change current track youtube source panel and tooltips for player icon buttons ([4b21cc8](https://github.com/KRTirtho/spotube/commit/4b21cc829954fb079d1a0081b9147377063da3ec)) -* Player and Playbutton theme respect to platform ([512446d](https://github.com/KRTirtho/spotube/commit/512446dcab72aa1d7bce18e5a5793f6be8f30fcb)) -* player queue and sibling tracks platform decoration ([39a7794](https://github.com/KRTirtho/spotube/commit/39a77945d132d90ff323b0a22edc2a12a4749888)) -* **PlayerView:** shortcut button for opening lyrics [#273](https://github.com/KRTirtho/spotube/issues/273) ([1d4847a](https://github.com/KRTirtho/spotube/commit/1d4847ab0a0b18d5bc27257b3db863b995dc5843)) -* rename files to snake_case and reorganize folder structure ([7c25e1c](https://github.com/KRTirtho/spotube/commit/7c25e1cc8a35eb8aee2268799c05f299204fa3f5)) -* replace all types of buttons with platform buttons ([69739b4](https://github.com/KRTirtho/spotube/commit/69739b457296a6b209aa6f73beb378ae1f089ac5)) -* rpm packaging support ([067e9ac](https://github.com/KRTirtho/spotube/commit/067e9ac53ee85775c9ec35457fac9e064e72e4c0)) -* **search:** infinite scroll for tracks, artists, playlists and albums ([e6761a6](https://github.com/KRTirtho/spotube/commit/e6761a6f8eadf4ab260723253a8e00121b6365b5)) -* set platform to default platform on start up ([472da6b](https://github.com/KRTirtho/spotube/commit/472da6b8b1c3e06b666da58351f3feafbfe6c98a)) -* shuffle keep playing track at top, linux title bar drag no working ([1223cf2](https://github.com/KRTirtho/spotube/commit/1223cf2629c6615b0c48e9e6742b68341f33c7f8)) -* sidebar download count and proper progress color in playbutton ([a10bc5b](https://github.com/KRTirtho/spotube/commit/a10bc5b8d89207ac86872dd25f3258e47c2141a5)) -* static shimmer for track tile, playbutton card and track tile ([3ed8b0f](https://github.com/KRTirtho/spotube/commit/3ed8b0fda2cb8145a2bc6c8f8c6af82db6a40547)) -* tablet mode navigation bar & windows semi transparent bg, ([3282370](https://github.com/KRTirtho/spotube/commit/3282370f74f323c00116fa8626fd1440ee9d4922)) -* **title_bar:** platform specific title bar ([e659e3c](https://github.com/KRTirtho/spotube/commit/e659e3c56fb02ad8a1c5114bf80074f0324cc4f1)) -* titlebar complete compatibility, platform specific login, library tabbar in titlebar ([b3c27d1](https://github.com/KRTirtho/spotube/commit/b3c27d1fca233ea079803fe49134abc528376df3)) -* use platform checkbox ([2211505](https://github.com/KRTirtho/spotube/commit/2211505d713cef752d07cafb9791a49e8095eee2)) -* window blur effect add ([b0db5e7](https://github.com/KRTirtho/spotube/commit/b0db5e7d8246f98835e7ce6656c8aa7620e46bec)) - +- add selected tracks to playlists, optimistic playlist remove track ([3386dac](https://github.com/KRTirtho/spotube/commit/3386dac78ee49b9e3504f5c05bf1e7b362a2e8a2)) +- added shuffle button in playlist and album section ([1fad95f](https://github.com/KRTirtho/spotube/commit/1fad95f6e370606a9faf6f2bb9738dc360e23918)) +- **android-playback:** option to download track bytes and play instead of Streaming ([dcc8ba5](https://github.com/KRTirtho/spotube/commit/dcc8ba5a54286b252d53c9c14918971bf7bea8cc)) +- change default platform option and platform specific back button ([36c5e02](https://github.com/KRTirtho/spotube/commit/36c5e02f18374100f61cc3f2957c27bfc0d8511f)) +- dialog logo for macos, settings more width for country picker ([5e96913](https://github.com/KRTirtho/spotube/commit/5e96913ba34230e7a15d62060d5dae28d80e3630)) +- initial platform_ui integration ([9eee573](https://github.com/KRTirtho/spotube/commit/9eee573ce928aa6c03dcb50bf1521350d2de32cc)) +- libadwaita theming, track tile and PlayButtonCard play button icon fix ([e795e23](https://github.com/KRTirtho/spotube/commit/e795e23e42e5f1f832963b2a4506df89b7df5baa)) +- **lyrics:** tabs for both synced and static lyrics [#182](https://github.com/KRTirtho/spotube/issues/182) ([6b6907a](https://github.com/KRTirtho/spotube/commit/6b6907af3fdb327312ad4dd9e16b3e2a850ed896)) +- new refined about page, update checker only check for same update channel ([4cadfa9](https://github.com/KRTirtho/spotube/commit/4cadfa93750cc9b3f8fbe7b60f8161a77d2a12f6)) +- pause track when seeking forward/back and keep audio session alive when paused/interrupted ([bc8a04e](https://github.com/KRTirtho/spotube/commit/bc8a04e5442ba3abe1b04ab325769559f37d9802)) +- platform bottom navigation bar add ([ff14469](https://github.com/KRTirtho/spotube/commit/ff1446982f0260c6fe231970aa6ed61c273fdf07)) +- platform slider and progress indicator integration ([46b00ba](https://github.com/KRTirtho/spotube/commit/46b00bafdf71a4add61cf50168a32198c3293181)) +- platform title bar buttons add ([54048cb](https://github.com/KRTirtho/spotube/commit/54048cbfc37d9a40c020eb81ab67b9dd5428f0d7)) +- **playback:** change current track youtube source panel and tooltips for player icon buttons ([4b21cc8](https://github.com/KRTirtho/spotube/commit/4b21cc829954fb079d1a0081b9147377063da3ec)) +- Player and Playbutton theme respect to platform ([512446d](https://github.com/KRTirtho/spotube/commit/512446dcab72aa1d7bce18e5a5793f6be8f30fcb)) +- player queue and sibling tracks platform decoration ([39a7794](https://github.com/KRTirtho/spotube/commit/39a77945d132d90ff323b0a22edc2a12a4749888)) +- **PlayerView:** shortcut button for opening lyrics [#273](https://github.com/KRTirtho/spotube/issues/273) ([1d4847a](https://github.com/KRTirtho/spotube/commit/1d4847ab0a0b18d5bc27257b3db863b995dc5843)) +- rename files to snake_case and reorganize folder structure ([7c25e1c](https://github.com/KRTirtho/spotube/commit/7c25e1cc8a35eb8aee2268799c05f299204fa3f5)) +- replace all types of buttons with platform buttons ([69739b4](https://github.com/KRTirtho/spotube/commit/69739b457296a6b209aa6f73beb378ae1f089ac5)) +- rpm packaging support ([067e9ac](https://github.com/KRTirtho/spotube/commit/067e9ac53ee85775c9ec35457fac9e064e72e4c0)) +- **search:** infinite scroll for tracks, artists, playlists and albums ([e6761a6](https://github.com/KRTirtho/spotube/commit/e6761a6f8eadf4ab260723253a8e00121b6365b5)) +- set platform to default platform on start up ([472da6b](https://github.com/KRTirtho/spotube/commit/472da6b8b1c3e06b666da58351f3feafbfe6c98a)) +- shuffle keep playing track at top, linux title bar drag no working ([1223cf2](https://github.com/KRTirtho/spotube/commit/1223cf2629c6615b0c48e9e6742b68341f33c7f8)) +- sidebar download count and proper progress color in playbutton ([a10bc5b](https://github.com/KRTirtho/spotube/commit/a10bc5b8d89207ac86872dd25f3258e47c2141a5)) +- static shimmer for track tile, playbutton card and track tile ([3ed8b0f](https://github.com/KRTirtho/spotube/commit/3ed8b0fda2cb8145a2bc6c8f8c6af82db6a40547)) +- tablet mode navigation bar & windows semi transparent bg, ([3282370](https://github.com/KRTirtho/spotube/commit/3282370f74f323c00116fa8626fd1440ee9d4922)) +- **title_bar:** platform specific title bar ([e659e3c](https://github.com/KRTirtho/spotube/commit/e659e3c56fb02ad8a1c5114bf80074f0324cc4f1)) +- titlebar complete compatibility, platform specific login, library tabbar in titlebar ([b3c27d1](https://github.com/KRTirtho/spotube/commit/b3c27d1fca233ea079803fe49134abc528376df3)) +- use platform checkbox ([2211505](https://github.com/KRTirtho/spotube/commit/2211505d713cef752d07cafb9791a49e8095eee2)) +- window blur effect add ([b0db5e7](https://github.com/KRTirtho/spotube/commit/b0db5e7d8246f98835e7ce6656c8aa7620e46bec)) ### Bug Fixes -* **ArtistCard:** linux shadow ([c186881](https://github.com/KRTirtho/spotube/commit/c1868817e5abb8a4152646f00a0395933fee7823)) -* **auth:** refresh access token timer not working ([b3ac5ca](https://github.com/KRTirtho/spotube/commit/b3ac5ca3bbb6d5af154f4b5d715d1f19ca2f46e2)) -* bottom navigation bar settings tile not active when selected ([43557e4](https://github.com/KRTirtho/spotube/commit/43557e40df269757c2d5236a455308ea6478d95a)) -* dialog logo in android, lyrics visible timer adjust button ([3c6803b](https://github.com/KRTirtho/spotube/commit/3c6803bb3fac8eee9166764089724194a48509c6)) -* heart button showing when not logged in, wrong login redirect ([4dc26af](https://github.com/KRTirtho/spotube/commit/4dc26af23d12f76cbfdfbf4e37b0c11fcc484d3f)) -* horizontal infinite lists doesn't fill the screen ([69995be](https://github.com/KRTirtho/spotube/commit/69995bea1c6342c9212e5b22ef50bdfd6e7eba45)) -* ios dialog action buttons, local tracks crashing app, shimmer color and android wrong status bar color ([90c1200](https://github.com/KRTirtho/spotube/commit/90c1200a087f796690de0cfc8cc607d2bff44282)) -* **login:** not working in android in Brazil or Ukraine regions ([0b79a11](https://github.com/KRTirtho/spotube/commit/0b79a1181c37cf06fbfa3bfb3854cfd47097016e)) -* **macos:** black text in dark mode ([fb9c0e4](https://github.com/KRTirtho/spotube/commit/fb9c0e44be93997fc852bf0260e8a8608000c023)) -* **macos:** white text color in dark mode, text field white background ([e086b52](https://github.com/KRTirtho/spotube/commit/e086b520e745e65771136cbfa842ae0693c44872)) -* **mobile:** SafeArea bugs and back button color ([a8330ef](https://github.com/KRTirtho/spotube/commit/a8330ef2e1112012bbae19ee6a5c27a26c5fb719)) -* null exception in themes ([9465d92](https://github.com/KRTirtho/spotube/commit/9465d92fa032b8598a0752767dcec9af2541d222)) -* platform_ui local path ([00d0d38](https://github.com/KRTirtho/spotube/commit/00d0d38b5450aeb877195afdfb9424f83762d178)) -* player view artist link when local playlist is playing, lyric delay adjust button alignment ([ee5c417](https://github.com/KRTirtho/spotube/commit/ee5c417ac396ef0b1796fa74a6a494181e6e0396)) -* remove windows background ([6942964](https://github.com/KRTirtho/spotube/commit/694296418787c460bb3fa63ab30f3b0eed9184dc)) -* search field ios dark icon , lyrics tabbar ios background color ([be56ad4](https://github.com/KRTirtho/spotube/commit/be56ad44773ebcd14777d80b61e26875698dc18a)) -* settings Title alignment and play button card ripple effect in other platforms ([3b6bf27](https://github.com/KRTirtho/spotube/commit/3b6bf27a984f5d4836143638396ed4b467c0eae7)) -* shuffle play logic ([65cad07](https://github.com/KRTirtho/spotube/commit/65cad07e3a6e2188c53159057f9c3d4fe89706ea)) -* small minwidth of window in desktop, linux wrong light theme accent color, search field transparent background ([5b0e22c](https://github.com/KRTirtho/spotube/commit/5b0e22c1b639f2f57d92cb70cd11f56e30e0a457)) -* tooltips of menu and adaptive pop up menu ([261aaf1](https://github.com/KRTirtho/spotube/commit/261aaf191c51bc12b28c602ee160d53d3eacf3a5)) -* update download dialog blocking the UI ([3925f74](https://github.com/KRTirtho/spotube/commit/3925f743951e51f138cc3ca865fa167c34e776ef)) -* user playlists not updating after creating/deleting, artist follow not updating after follow/unfollow ([6cc2a18](https://github.com/KRTirtho/spotube/commit/6cc2a185d0c4c19f176e6f65b8ada19ebc76af5e)) -* **windows:** windows global title bar ([bd18f19](https://github.com/KRTirtho/spotube/commit/bd18f198217538f0089d5a1c4288dd97f982661b)) +- **ArtistCard:** linux shadow ([c186881](https://github.com/KRTirtho/spotube/commit/c1868817e5abb8a4152646f00a0395933fee7823)) +- **auth:** refresh access token timer not working ([b3ac5ca](https://github.com/KRTirtho/spotube/commit/b3ac5ca3bbb6d5af154f4b5d715d1f19ca2f46e2)) +- bottom navigation bar settings tile not active when selected ([43557e4](https://github.com/KRTirtho/spotube/commit/43557e40df269757c2d5236a455308ea6478d95a)) +- dialog logo in android, lyrics visible timer adjust button ([3c6803b](https://github.com/KRTirtho/spotube/commit/3c6803bb3fac8eee9166764089724194a48509c6)) +- heart button showing when not logged in, wrong login redirect ([4dc26af](https://github.com/KRTirtho/spotube/commit/4dc26af23d12f76cbfdfbf4e37b0c11fcc484d3f)) +- horizontal infinite lists doesn't fill the screen ([69995be](https://github.com/KRTirtho/spotube/commit/69995bea1c6342c9212e5b22ef50bdfd6e7eba45)) +- ios dialog action buttons, local tracks crashing app, shimmer color and android wrong status bar color ([90c1200](https://github.com/KRTirtho/spotube/commit/90c1200a087f796690de0cfc8cc607d2bff44282)) +- **login:** not working in android in Brazil or Ukraine regions ([0b79a11](https://github.com/KRTirtho/spotube/commit/0b79a1181c37cf06fbfa3bfb3854cfd47097016e)) +- **macos:** black text in dark mode ([fb9c0e4](https://github.com/KRTirtho/spotube/commit/fb9c0e44be93997fc852bf0260e8a8608000c023)) +- **macos:** white text color in dark mode, text field white background ([e086b52](https://github.com/KRTirtho/spotube/commit/e086b520e745e65771136cbfa842ae0693c44872)) +- **mobile:** SafeArea bugs and back button color ([a8330ef](https://github.com/KRTirtho/spotube/commit/a8330ef2e1112012bbae19ee6a5c27a26c5fb719)) +- null exception in themes ([9465d92](https://github.com/KRTirtho/spotube/commit/9465d92fa032b8598a0752767dcec9af2541d222)) +- platform_ui local path ([00d0d38](https://github.com/KRTirtho/spotube/commit/00d0d38b5450aeb877195afdfb9424f83762d178)) +- player view artist link when local playlist is playing, lyric delay adjust button alignment ([ee5c417](https://github.com/KRTirtho/spotube/commit/ee5c417ac396ef0b1796fa74a6a494181e6e0396)) +- remove windows background ([6942964](https://github.com/KRTirtho/spotube/commit/694296418787c460bb3fa63ab30f3b0eed9184dc)) +- search field ios dark icon , lyrics tabbar ios background color ([be56ad4](https://github.com/KRTirtho/spotube/commit/be56ad44773ebcd14777d80b61e26875698dc18a)) +- settings Title alignment and play button card ripple effect in other platforms ([3b6bf27](https://github.com/KRTirtho/spotube/commit/3b6bf27a984f5d4836143638396ed4b467c0eae7)) +- shuffle play logic ([65cad07](https://github.com/KRTirtho/spotube/commit/65cad07e3a6e2188c53159057f9c3d4fe89706ea)) +- small minwidth of window in desktop, linux wrong light theme accent color, search field transparent background ([5b0e22c](https://github.com/KRTirtho/spotube/commit/5b0e22c1b639f2f57d92cb70cd11f56e30e0a457)) +- tooltips of menu and adaptive pop up menu ([261aaf1](https://github.com/KRTirtho/spotube/commit/261aaf191c51bc12b28c602ee160d53d3eacf3a5)) +- update download dialog blocking the UI ([3925f74](https://github.com/KRTirtho/spotube/commit/3925f743951e51f138cc3ca865fa167c34e776ef)) +- user playlists not updating after creating/deleting, artist follow not updating after follow/unfollow ([6cc2a18](https://github.com/KRTirtho/spotube/commit/6cc2a185d0c4c19f176e6f65b8ada19ebc76af5e)) +- **windows:** windows global title bar ([bd18f19](https://github.com/KRTirtho/spotube/commit/bd18f198217538f0089d5a1c4288dd97f982661b)) ## [2.5.0](https://github.com/KRTirtho/spotube/compare/v2.4.1...v2.5.0) (2022-10-13) - ### Features -* animated transition of root PageWindowTitleBar ([ff35e06](https://github.com/KRTirtho/spotube/commit/ff35e06a6605fc7ec762e716fb7bdf6f7eb45732)) -* **auth:** new authentication flow using cookies and webview in android ([756b910](https://github.com/KRTirtho/spotube/commit/756b91007ee747c10ed10aa7060af49b555a2eaf)) -* **downloader:** replace /skip all choice for downloaded tracks ([88d7ce5](https://github.com/KRTirtho/spotube/commit/88d7ce55a59f673d60cd9e85ab062bcb1b7dcbc3)) -* implemented go_route shell/nested route ([3e498a4](https://github.com/KRTirtho/spotube/commit/3e498a4827a1118e0b23faec7cf114272f7838d4)) -* **keyboard shortcuts:** play/pause on space, seek position on left/right ([2734454](https://github.com/KRTirtho/spotube/commit/2734454717bbfb5d0621c6ea72fa755ef4fc8602)) -* **keyboard-shortcuts:** home sidebar tab navigation and close app ([8f258e7](https://github.com/KRTirtho/spotube/commit/8f258e709ada418dbeef8d272af370b1741afd9c)) -* smoother list using fl_query and waypoint ([c77b0e1](https://github.com/KRTirtho/spotube/commit/c77b0e198b215180d863747e35998a17aff92720)) -* sort tracks in playlist, album and local tracks ([cb4bd25](https://github.com/KRTirtho/spotube/commit/cb4bd25df154455d225c426cfeaaea36ac09e9b7)) -* use of smaller sized images in `TrackTile` ([0ca97b4](https://github.com/KRTirtho/spotube/commit/0ca97b495f2a9ece8356f7813fc0e37d1cdb8608)) -* volume slider mouse scroll and preference for Rotating Album Art [#255](https://github.com/KRTirtho/spotube/issues/255) ([edb6f3c](https://github.com/KRTirtho/spotube/commit/edb6f3cd1c9ee2961040b2fe7a91c48577cee4f7)) - +- animated transition of root PageWindowTitleBar ([ff35e06](https://github.com/KRTirtho/spotube/commit/ff35e06a6605fc7ec762e716fb7bdf6f7eb45732)) +- **auth:** new authentication flow using cookies and webview in android ([756b910](https://github.com/KRTirtho/spotube/commit/756b91007ee747c10ed10aa7060af49b555a2eaf)) +- **downloader:** replace /skip all choice for downloaded tracks ([88d7ce5](https://github.com/KRTirtho/spotube/commit/88d7ce55a59f673d60cd9e85ab062bcb1b7dcbc3)) +- implemented go_route shell/nested route ([3e498a4](https://github.com/KRTirtho/spotube/commit/3e498a4827a1118e0b23faec7cf114272f7838d4)) +- **keyboard shortcuts:** play/pause on space, seek position on left/right ([2734454](https://github.com/KRTirtho/spotube/commit/2734454717bbfb5d0621c6ea72fa755ef4fc8602)) +- **keyboard-shortcuts:** home sidebar tab navigation and close app ([8f258e7](https://github.com/KRTirtho/spotube/commit/8f258e709ada418dbeef8d272af370b1741afd9c)) +- smoother list using fl_query and waypoint ([c77b0e1](https://github.com/KRTirtho/spotube/commit/c77b0e198b215180d863747e35998a17aff92720)) +- sort tracks in playlist, album and local tracks ([cb4bd25](https://github.com/KRTirtho/spotube/commit/cb4bd25df154455d225c426cfeaaea36ac09e9b7)) +- use of smaller sized images in `TrackTile` ([0ca97b4](https://github.com/KRTirtho/spotube/commit/0ca97b495f2a9ece8356f7813fc0e37d1cdb8608)) +- volume slider mouse scroll and preference for Rotating Album Art [#255](https://github.com/KRTirtho/spotube/issues/255) ([edb6f3c](https://github.com/KRTirtho/spotube/commit/edb6f3cd1c9ee2961040b2fe7a91c48577cee4f7)) ### Bug Fixes -* **android:** file_picker and permission_handler failure for sdk < 33 ([139d4dc](https://github.com/KRTirtho/spotube/commit/139d4dc033d9aaa1d6882bf0f53e96a3b1e87c95)) -* cached local track is fetched from network ([abf4a57](https://github.com/KRTirtho/spotube/commit/abf4a5763a2faeedeb93d54e66c1f2482295b326)) -* categories not showing for oauth exception ([4df917e](https://github.com/KRTirtho/spotube/commit/4df917e65ee20cbcf42394cc141b1cdcdd6cc914)) -* **desktop:** maximized window size is stored and window maximized state doesn't persist ([91d5d10](https://github.com/KRTirtho/spotube/commit/91d5d1003b09530ff3bc9a0aa93e382e943977e0)) -* local audio doesn't get refreshed after getting permission ([618c6da](https://github.com/KRTirtho/spotube/commit/618c6da0ebddf3cc8e216743bbbb9220bcf40521)) -* no appropriate output when playlist is empty [#201](https://github.com/KRTirtho/spotube/issues/201) ([dbb81de](https://github.com/KRTirtho/spotube/commit/dbb81de763df60eba62ef1256a7161ea6ca59b66)) -* PlayerOverlay not hiding when not playing and unneeded bottom space in TrackTableView ([0ebac05](https://github.com/KRTirtho/spotube/commit/0ebac05a4be8e8f744a6c672d3bb9807d6f02e10)) -* **web:** not building due to metadata_god ffi ([1191bf2](https://github.com/KRTirtho/spotube/commit/1191bf232d0797aaae7eff2f5d570acd49ce61bd)) +- **android:** file_picker and permission_handler failure for sdk < 33 ([139d4dc](https://github.com/KRTirtho/spotube/commit/139d4dc033d9aaa1d6882bf0f53e96a3b1e87c95)) +- cached local track is fetched from network ([abf4a57](https://github.com/KRTirtho/spotube/commit/abf4a5763a2faeedeb93d54e66c1f2482295b326)) +- categories not showing for oauth exception ([4df917e](https://github.com/KRTirtho/spotube/commit/4df917e65ee20cbcf42394cc141b1cdcdd6cc914)) +- **desktop:** maximized window size is stored and window maximized state doesn't persist ([91d5d10](https://github.com/KRTirtho/spotube/commit/91d5d1003b09530ff3bc9a0aa93e382e943977e0)) +- local audio doesn't get refreshed after getting permission ([618c6da](https://github.com/KRTirtho/spotube/commit/618c6da0ebddf3cc8e216743bbbb9220bcf40521)) +- no appropriate output when playlist is empty [#201](https://github.com/KRTirtho/spotube/issues/201) ([dbb81de](https://github.com/KRTirtho/spotube/commit/dbb81de763df60eba62ef1256a7161ea6ca59b66)) +- PlayerOverlay not hiding when not playing and unneeded bottom space in TrackTableView ([0ebac05](https://github.com/KRTirtho/spotube/commit/0ebac05a4be8e8f744a6c672d3bb9807d6f02e10)) +- **web:** not building due to metadata_god ffi ([1191bf2](https://github.com/KRTirtho/spotube/commit/1191bf232d0797aaae7eff2f5d570acd49ce61bd)) ## [2.4.1](https://github.com/KRTirtho/spotube/compare/v2.4.0...v2.4.1) (2022-09-13) - ### Features -* add macos audio metadata tags support ([5866b0f](https://github.com/KRTirtho/spotube/commit/5866b0fcd661cf32060bb1485ea81634fbb9b90a)) -* remove macos bounds for reading and writing audio metadata ([16064f6](https://github.com/KRTirtho/spotube/commit/16064f68e882b091401ace4b895e387f46635800)) -* **search:** horizontal swipe scroll support for Desktop platform ([d5ff927](https://github.com/KRTirtho/spotube/commit/d5ff927c7273b6e72c5d775ee777f2cbd0d6d05c)) - +- add macos audio metadata tags support ([5866b0f](https://github.com/KRTirtho/spotube/commit/5866b0fcd661cf32060bb1485ea81634fbb9b90a)) +- remove macos bounds for reading and writing audio metadata ([16064f6](https://github.com/KRTirtho/spotube/commit/16064f68e882b091401ace4b895e387f46635800)) +- **search:** horizontal swipe scroll support for Desktop platform ([d5ff927](https://github.com/KRTirtho/spotube/commit/d5ff927c7273b6e72c5d775ee777f2cbd0d6d05c)) ### Bug Fixes -* **artist-page:** SpotubeMarqueeText used in ArtistCard crashes the app ([4279541](https://github.com/KRTirtho/spotube/commit/427954150ab65b250e79fc844fc864abff5b6972)) -* **layout:** Fix adaptive UI not working correctly by providing a overriding option ([8c7adde](https://github.com/KRTirtho/spotube/commit/8c7adde890105e0267b71994b7928277f84553e5)) -* **local-track:** throwing exception when downloadLocation is empty ([1a3556d](https://github.com/KRTirtho/spotube/commit/1a3556d39e8473cadb6143192c48465dc6485599)) +- **artist-page:** SpotubeMarqueeText used in ArtistCard crashes the app ([4279541](https://github.com/KRTirtho/spotube/commit/427954150ab65b250e79fc844fc864abff5b6972)) +- **layout:** Fix adaptive UI not working correctly by providing a overriding option ([8c7adde](https://github.com/KRTirtho/spotube/commit/8c7adde890105e0267b71994b7928277f84553e5)) +- **local-track:** throwing exception when downloadLocation is empty ([1a3556d](https://github.com/KRTirtho/spotube/commit/1a3556d39e8473cadb6143192c48465dc6485599)) ## [2.4.0](https://github.com/KRTirtho/spotube/compare/v2.3.0...v2.4.0) (2022-09-09) - ### Features -* Ability to change download location added ([816707c](https://github.com/KRTirtho/spotube/commit/816707c643f8d60d25bc08fd4c8005daa2ba9e63)) -* add download multi tracks support for mobile platform ([0476bf7](https://github.com/KRTirtho/spotube/commit/0476bf7ceece034a927d1df6099d8b33036f8a9b)) -* add download queue for desktop & initial playlist download support ([08f913e](https://github.com/KRTirtho/spotube/commit/08f913e9761d0f5c447af9dfb6eedb44b675498c)) -* add download tab on library ([8d77b69](https://github.com/KRTirtho/spotube/commit/8d77b6900a81aab020e19397e788964b0ac499ff)) -* add web support although nothing works just as expected ([2818ed5](https://github.com/KRTirtho/spotube/commit/2818ed5c9dadb9185a52762599c1dd0acd81e6bf)) -* **broken:** Broken Warning! Initial Local Audio Player ([c3bf511](https://github.com/KRTirtho/spotube/commit/c3bf5119ebb7c17e8c32f149598508674b0acd39)) -* **download:** track table view multi select improvement, tap to play track support, existing track replace confirmation dialog and bulk download confirmation dialog ([e217553](https://github.com/KRTirtho/spotube/commit/e21755322f2cd5f1fba00c5c8cd5c5d1f79e459d)) -* **local-tracks:** complete support for local tracks ([e206f16](https://github.com/KRTirtho/spotube/commit/e206f16723ac989ad58006c1b3c90c6691d8cab3)) -* **mpris:** MPRIS metadata are now updated in realtime ([d9addcd](https://github.com/KRTirtho/spotube/commit/d9addcda8e9562803bd73016148fab22560ee050)) -* **playback:** add repeat track support [#166](https://github.com/KRTirtho/spotube/issues/166) ([cae9993](https://github.com/KRTirtho/spotube/commit/cae99934299bd197c68f626d6c10158d449770b9)) -* **synced-lyrics:** animated active text size ([531fae6](https://github.com/KRTirtho/spotube/commit/531fae64f94b21551a7a0da363a9ab0d44f5d3b1)) -* **ui:** adaptive TrackTile actions & Setting ListTile ([615d5ce](https://github.com/KRTirtho/spotube/commit/615d5ce901eb0512e84a120b7309c9053238ee36)) - +- Ability to change download location added ([816707c](https://github.com/KRTirtho/spotube/commit/816707c643f8d60d25bc08fd4c8005daa2ba9e63)) +- add download multi tracks support for mobile platform ([0476bf7](https://github.com/KRTirtho/spotube/commit/0476bf7ceece034a927d1df6099d8b33036f8a9b)) +- add download queue for desktop & initial playlist download support ([08f913e](https://github.com/KRTirtho/spotube/commit/08f913e9761d0f5c447af9dfb6eedb44b675498c)) +- add download tab on library ([8d77b69](https://github.com/KRTirtho/spotube/commit/8d77b6900a81aab020e19397e788964b0ac499ff)) +- add web support although nothing works just as expected ([2818ed5](https://github.com/KRTirtho/spotube/commit/2818ed5c9dadb9185a52762599c1dd0acd81e6bf)) +- **broken:** Broken Warning! Initial Local Audio Player ([c3bf511](https://github.com/KRTirtho/spotube/commit/c3bf5119ebb7c17e8c32f149598508674b0acd39)) +- **download:** track table view multi select improvement, tap to play track support, existing track replace confirmation dialog and bulk download confirmation dialog ([e217553](https://github.com/KRTirtho/spotube/commit/e21755322f2cd5f1fba00c5c8cd5c5d1f79e459d)) +- **local-tracks:** complete support for local tracks ([e206f16](https://github.com/KRTirtho/spotube/commit/e206f16723ac989ad58006c1b3c90c6691d8cab3)) +- **mpris:** MPRIS metadata are now updated in realtime ([d9addcd](https://github.com/KRTirtho/spotube/commit/d9addcda8e9562803bd73016148fab22560ee050)) +- **playback:** add repeat track support [#166](https://github.com/KRTirtho/spotube/issues/166) ([cae9993](https://github.com/KRTirtho/spotube/commit/cae99934299bd197c68f626d6c10158d449770b9)) +- **synced-lyrics:** animated active text size ([531fae6](https://github.com/KRTirtho/spotube/commit/531fae64f94b21551a7a0da363a9ab0d44f5d3b1)) +- **ui:** adaptive TrackTile actions & Setting ListTile ([615d5ce](https://github.com/KRTirtho/spotube/commit/615d5ce901eb0512e84a120b7309c9053238ee36)) ### Bug Fixes -* **adaptive-list-tile:** dialog content not updating when content has changed ([a1d4230](https://github.com/KRTirtho/spotube/commit/a1d423090c854ebe319a0fa03fd6e5c4007b1387)) -* album & playlist card, player view and album view play button logic ([55852bd](https://github.com/KRTirtho/spotube/commit/55852bd15bc709d61fbba8cbea01ceca791d154c)) -* **docs:** indentions ([4a291d5](https://github.com/KRTirtho/spotube/commit/4a291d5f20dabe68f3ed64071624dcbed8327329)) -* **downloader:** downloaded track is corrupted for tagging ([2ab1fba](https://github.com/KRTirtho/spotube/commit/2ab1fba3d64147e3c5cf34756dce1cf6046d410a)) -* **downloader:** flutter downloader exception on desktop platform and too much width of TrackTile index no. ([d668760](https://github.com/KRTirtho/spotube/commit/d6687603d148ad936530cca4d09e128a59b79b19)) -* dropped flutter_downloader deps due to slow download speed and UserDownloads not showing for anonymous ([307a8e2](https://github.com/KRTirtho/spotube/commit/307a8e21df1e39123a1dca4c1b063eab50359581)) -* flutter_downloader manifest configuration breaking android support ([f3a0f78](https://github.com/KRTirtho/spotube/commit/f3a0f78fb92ff7ee38b5a9ef9954575d4282f954)) -* login screen not using safearea and no dialog bg-color found on light mode in AdaptivePopupMenuButton ([92bc611](https://github.com/KRTirtho/spotube/commit/92bc611c5e901dcabf34086be9287ac20317259a)) -* **performance:** always running marquee text causes high GPU usage [#175](https://github.com/KRTirtho/spotube/issues/175) and UserArtist overflow on smaller displays ([a23ce61](https://github.com/KRTirtho/spotube/commit/a23ce614467b4297f495b824f0958ff07c21ae92)) -* **playback:** shuffle button sometimes gets stuck and stops working [#183](https://github.com/KRTirtho/spotube/issues/183) ([4240433](https://github.com/KRTirtho/spotube/commit/4240433e3dde6ab948d2674e07e41c27c1f6eac8)) -* **player-overlay:** flickering when a track is changed or navigated to another page ([e48b67c](https://github.com/KRTirtho/spotube/commit/e48b67cd47ae54ad9268aead268e444836a67b0d)) -* **sidebar:** user image url ([747efc6](https://github.com/KRTirtho/spotube/commit/747efc6ee66bc6c7c917cc02bd134968a0781701)) -* **synced-lyrics:** active lyrics contrast ratio ([aba1ba9](https://github.com/KRTirtho/spotube/commit/aba1ba932592923720a36395c057f78820dafecf)) -* tabbar overflow in small screen, artist card too small title and synced lyrics contrast increased ([585de8c](https://github.com/KRTirtho/spotube/commit/585de8c1def9750826568317109b242a5e18f28c)) +- **adaptive-list-tile:** dialog content not updating when content has changed ([a1d4230](https://github.com/KRTirtho/spotube/commit/a1d423090c854ebe319a0fa03fd6e5c4007b1387)) +- album & playlist card, player view and album view play button logic ([55852bd](https://github.com/KRTirtho/spotube/commit/55852bd15bc709d61fbba8cbea01ceca791d154c)) +- **docs:** indentions ([4a291d5](https://github.com/KRTirtho/spotube/commit/4a291d5f20dabe68f3ed64071624dcbed8327329)) +- **downloader:** downloaded track is corrupted for tagging ([2ab1fba](https://github.com/KRTirtho/spotube/commit/2ab1fba3d64147e3c5cf34756dce1cf6046d410a)) +- **downloader:** flutter downloader exception on desktop platform and too much width of TrackTile index no. ([d668760](https://github.com/KRTirtho/spotube/commit/d6687603d148ad936530cca4d09e128a59b79b19)) +- dropped flutter_downloader deps due to slow download speed and UserDownloads not showing for anonymous ([307a8e2](https://github.com/KRTirtho/spotube/commit/307a8e21df1e39123a1dca4c1b063eab50359581)) +- flutter_downloader manifest configuration breaking android support ([f3a0f78](https://github.com/KRTirtho/spotube/commit/f3a0f78fb92ff7ee38b5a9ef9954575d4282f954)) +- login screen not using safearea and no dialog bg-color found on light mode in AdaptivePopupMenuButton ([92bc611](https://github.com/KRTirtho/spotube/commit/92bc611c5e901dcabf34086be9287ac20317259a)) +- **performance:** always running marquee text causes high GPU usage [#175](https://github.com/KRTirtho/spotube/issues/175) and UserArtist overflow on smaller displays ([a23ce61](https://github.com/KRTirtho/spotube/commit/a23ce614467b4297f495b824f0958ff07c21ae92)) +- **playback:** shuffle button sometimes gets stuck and stops working [#183](https://github.com/KRTirtho/spotube/issues/183) ([4240433](https://github.com/KRTirtho/spotube/commit/4240433e3dde6ab948d2674e07e41c27c1f6eac8)) +- **player-overlay:** flickering when a track is changed or navigated to another page ([e48b67c](https://github.com/KRTirtho/spotube/commit/e48b67cd47ae54ad9268aead268e444836a67b0d)) +- **sidebar:** user image url ([747efc6](https://github.com/KRTirtho/spotube/commit/747efc6ee66bc6c7c917cc02bd134968a0781701)) +- **synced-lyrics:** active lyrics contrast ratio ([aba1ba9](https://github.com/KRTirtho/spotube/commit/aba1ba932592923720a36395c057f78820dafecf)) +- tabbar overflow in small screen, artist card too small title and synced lyrics contrast increased ([585de8c](https://github.com/KRTirtho/spotube/commit/585de8c1def9750826568317109b242a5e18f28c)) # v2.3.0 ### New + - Playback Cache Support. So unfinished playlist and tracks remains cached & starts automatically when application is launched again - Login Screen guided tutorial about how to obtain Client ID & Client Secret - Signed Android Application so now longer need to uninstall the old version for installing the new one @@ -813,8 +914,8 @@ All notable changes to this project will be documented in this file. See [standa - New Blur background design adapted to multiple components including Floating Player, Player View & Lyrics Tab - New HighContrast Color Scheme addition which reduces battery consumption on OLED or AMOLED display devices (https://github.com/KRTirtho/spotube/issues/137) - ### Improved + - Loading screens & animations. Now uses Skeleton Loading - Playlist & Album Pages now show Album Art & extra metadata as Header with vibrant gradient background in a Sliver - Playback is now more consistent & the API is simpler. Also its the single source of truth for AudioPlayback instead of the AudioServiceHandler @@ -823,6 +924,7 @@ All notable changes to this project will be documented in this file. See [standa - Track match Cache support for previously played tracks. This dramatically reduces track change latency & load on the YouTube search engine too ### Bug Fixes + - API rate limits inside TrackTile for multiple Follow queries at once - Player doesn't stop when Application is exits or closed - First Track of Playlist doesn't load sometimes @@ -832,9 +934,11 @@ All notable changes to this project will be documented in this file. See [standa # v2.2.1 ### Improved + - Page transitions defaulted to material you design -### Bug fixes +### Bug fixes + - Mini Player flickering on random state updates - Track More Options not showing when not logged in - Wrong link to Client ID & Client Secret tutorial in Login page @@ -843,6 +947,7 @@ All notable changes to this project will be documented in this file. See [standa # v2.2.0 ### New + - Update checker - Share options for playlists & track - Android Skip to Next/Previous track from notification/lockscreen (https://github.com/KRTirtho/spotube/issues/91) @@ -855,10 +960,13 @@ All notable changes to this project will be documented in this file. See [standa - M1 Mac support via MacOS Universal Binary (untested) (https://github.com/KRTirtho/spotube/pull/87) ### Improved -- Authentication is now persistent (no more re-login) + +- Authentication is now persistent (no more re-login) - Settings Page. Shows application details in About Dialog - Playlist Create Dialog Scrollable + ### Bug fixes + - private playlists of current user aren't shown fix (https://github.com/KRTirtho/spotube/issues/92) - refresh token error causing re-login (culprit: internal lib spotify-dart) - Typo in Login instructions URL @@ -866,6 +974,7 @@ All notable changes to this project will be documented in this file. See [standa # v2.1.0 ### New + - Synced Lyrics (with fallback genius lyrics) - Playlist create/delete - Add/Remove tracks to own playlists @@ -874,6 +983,7 @@ All notable changes to this project will be documented in this file. See [standa - Customize Marketplace location ### Improved + - Spotify track to youtube track algorithm - Genius lyrics matching algorithm - Download track. Checks if already exists & replaces on user command @@ -881,14 +991,16 @@ All notable changes to this project will be documented in this file. See [standa - Bigger Title display (replaced word-break with Marquee Text for better visibility) (https://github.com/KRTirtho/spotube/pull/47) ### Bug fixes + - Sequential playlist playback not working with latest webkit2gtk (https://github.com/KRTirtho/spotube/issues/46) - Theme modification state doesn't persist (https://github.com/KRTirtho/spotube/issues/54) - Wrong URI path for "Login with Spotify" tutorial (https://github.com/KRTirtho/spotube/issues/69) -- Card shadow showing in the background of TitleBar & Searchbar +- Card shadow showing in the background of TitleBar & Searchbar -# v2.0.0 +# v2.0.0 ### New + - Android Support https://github.com/KRTirtho/spotube/issues/24 - Responsive UI (Mobile, Tablet) - Anonymous/Guest Account @@ -901,14 +1013,16 @@ All notable changes to this project will be documented in this file. See [standa - Android NavigationPanel controls (OS media controls of Android) ### Improved + - Search - now scrolls & paginates for Playlists & Albums - Authentication - allows guest accounts making authentication optional -- Lyrics - can be fetched without requiring GeniusAccessToken. This makes geniusAccessToken optional +- Lyrics - can be fetched without requiring GeniusAccessToken. This makes geniusAccessToken optional - UI snappiness & faster load times - Simpler logic, faster calculations & better caching (flutter_hooks) - shared state management - uses riverpod & hooks combination ### Bug fixes + - Can't play any song in macos https://github.com/KRTirtho/spotube/issues/23 - Downloaded tracks can't be played as they're WebAudio (.weba) instead of MP3 - delay while changing Playlist/Single tracks @@ -916,24 +1030,28 @@ All notable changes to this project will be documented in this file. See [standa # v1.2.0 ### New -- Global custom reconfigurable *hotkey* support for playback controls (play-pause/next/previous) + +- Global custom reconfigurable _hotkey_ support for playback controls (play-pause/next/previous) - Credit section in the Settings page with important links + ### Improved + - Macos support - Genius (Lyrics Provider) access_token can be saved in the Login page too - Better theme for dropdown-buttons ### Bug fixes + - broken authentication IPC on Mac OS (https://github.com/KRTirtho/spotube/pull/18) - Mac OS's global appmenu's default APP_NAME replaced with Spotube - location of back button on macOS (https://github.com/KRTirtho/spotube/pull/21) - windows titlebar buttons appears on Mac OS - genius access_token not loading on initial app start - # v1.1.0 ### New + - MacOS support https://github.com/KRTirtho/spotube/pull/7 - Download currently playing track to `/home//Downloads/Spotube` (Linux, MacOS) or `C:\Users\\Downloads\Spotube` (Windows) - Play playlist from any song (index) instead of only the first track @@ -949,27 +1067,33 @@ All notable changes to this project will be documented in this file. See [standa - Click to open artist-profile/album everywhere in the application ### Improved + - UserLibrary album & artist tab - PlaylistView simplified layout with `ListView` instead of `TableView` - Control Theme from settings manually - `PageWindowTitleBar` now acts as `appBar` ### Bug fixes + - Unsafe access to album art/artist/user Images with `.first` or `.last` causing accessing empty List error -- `url_launcher`'s unstable `canLaunch` method blocks OAuth login in certain *nix OSs +- `url_launcher`'s unstable `canLaunch` method blocks OAuth login in certain \*nix OSs - Refresh token gets revoked & doesn't get renewed automatically + # v1.0.1 ### Improved + - Placeholder avatar for User section powered by dicebear.com ### Bug fixes + - No fallback/placeholder image causing undefined behavior (#2) - Unsafe access to empty List with List.first/List.last # v1.0.0 ### New + - Complete re-write in Flutter/Dart (799e13c) - mpv & youtube-dl runtime dependencies dropped (07b1891) - just_audio (libwinmedia + libwebkit2gtk-4.0-dev) + youtube_explode based playback & streaming @@ -977,15 +1101,17 @@ All notable changes to this project will be documented in this file. See [standa - inno_setup based windows/win32 GUI installer (dbf8a34) ### Improved + - Lower RAM & CPU usage. 2x less RAM usage & 20% less CPU usage - Faster playback & smooth track change with proper shuffling support - Automatic Dark mode support (system) - 54% smaller bundle size (after compression) -- Available through package managers in Linux (Debian, Arch, Flatpak & AppImage) +- Available through package managers in Linux (Debian, Arch, Flatpak & AppImage) # v0.0.3 ### New + - Automated installer for Windows (now doesn't require manual mpv-player install) - Playback caching - Retry button for ManualLyricDialog @@ -993,23 +1119,28 @@ All notable changes to this project will be documented in this file. See [standa - Redirect to youtube video by clicking on the title of the track ### Improved + - Inapp Shortcuts.Now it doesn't interfere while typing in a input box in Search page ### Bug fixes + - Cached image didn't get deleted after exiting certain cache limit fix. Cache gets recreated after exiting the limit # v0.0.2 ### New + - Lyric Seek - Support for images in playlist cards - Infinite Query/Pagination support for Home & Genre pages - Settings for configuring local configuration ### Improved + - Home Page Layout. Fixes the jiggering of Playlist Links on hover ### Bug Fixes + - `access_token not found` Error after OAuth Login with Spotify credentials (used to need a restart of the app to load the access_token) - Volume level wasn't cached even after changing volume @@ -1018,8 +1149,9 @@ All notable changes to this project will be documented in this file. See [standa Spotube v0.0.1 - initial release of the open source software for playing Spotify music using Youtube public API ### New + - Local playback handling - Playback Queue - Save to Liked Tracks/Playlists - Bypass API rate limitation on basic usage using personal developer Apps for spotify API -- Youtube search & get handled using scrape-yt \ No newline at end of file +- Youtube search & get handled using scrape-yt diff --git a/CONTRIBUTION.md b/CONTRIBUTION.md index d4746a1a..db77a5f1 100644 --- a/CONTRIBUTION.md +++ b/CONTRIBUTION.md @@ -119,7 +119,7 @@ Enhancement suggestions are tracked as [GitHub issues](https://github.com/KRTirt Do the following: -- Download the latest Flutter SDK (>=3.16.0) & enable desktop support +- Install [Dart](https://dart.dev/get-dart) and [fvm](https://fvm.app/documentation/getting-started/installation) - Install Development dependencies in linux - Debian (>=12/Bookworm)/Ubuntu ```bash @@ -138,11 +138,11 @@ Do the following: - Create a `.env` in root of the project following the `.env.example` template - Now run the following to bootstrap the project ```bash - flutter pub get && dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns + fvm flutter pub get && fvm dart run build_runner build --delete-conflicting-outputs ``` - Finally run these following commands in the root of the project to start the Spotube Locally ```bash - flutter run -d )> + fvm flutter run -d )> ``` Do debugging/testing/build etc then submit to us with PR against the development branch (dev) & we'll review your code diff --git a/LICENSE b/LICENSE index 0bce62d7..11aea461 100644 --- a/LICENSE +++ b/LICENSE @@ -1,12 +1,12 @@ BSD-4-Clause License -Copyright (c) 2023 Kingkor Roy Tirtho. All rights reserved. +Copyright (c) 2025 Kingkor Roy Tirtho. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. All advertising materials mentioning features or use of this software must display the following acknowledgement: -This product includes software developed by Kingkor Roy Tirtho. + This product includes software developed by Kingkor Roy Tirtho. 4. Neither the name of the Software nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. -THIS SOFTWARE IS PROVIDED BY KINGKOR ROY TIRTHO AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL KINGKOR ROY TIRTHO AND CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + THIS SOFTWARE IS PROVIDED BY KINGKOR ROY TIRTHO AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL KINGKOR ROY TIRTHO AND CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/Makefile b/Makefile index 25ac3a6d..49ae034a 100644 --- a/Makefile +++ b/Makefile @@ -8,7 +8,7 @@ tar: mkdir -p $(TEMP_DIR)\ && cp -r $(BUNDLE_DIR)/* $(TEMP_DIR)\ && cp linux/spotube.desktop $(TEMP_DIR)\ - && cp assets/spotube-logo.png $(TEMP_DIR)\ + && cp assets/branding/spotube-logo.png $(TEMP_DIR)\ && cp linux/com.github.KRTirtho.Spotube.appdata.xml $(TEMP_DIR)\ && tar -cJf build/spotube-linux-${VERSION}-${PKG_ARCH}.tar.xz -C $(TEMP_DIR) .\ && rm -rf $(TEMP_DIR) @@ -45,4 +45,14 @@ gensums: sh -c scripts/gensums.sh migrate: - dart run drift_dev make-migrations \ No newline at end of file + dart run drift_dev make-migrations + +dmg: + flutter build macos &&\ + if [ -f dist/Spotube-macos-universal.dmg ];\ + then rm dist/Spotube-macos-universal.dmg;\ + fi &&\ + appdmg appdmg.json dist/Spotube-macos-universal.dmg + +changelog: + git-cliff --unreleased \ No newline at end of file diff --git a/README.md b/README.md index deeba646..1043fabc 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,8 @@
- Spotube Logo + Spotube Logo -An open source, cross-platform Spotify client compatible across multiple platforms
-utilizing Spotify's data API and YouTube, Piped.video or JioSaavn as an audio source,
-eliminating the need for Spotify Premium +A cross-platform extensible open-source music streaming platform.
+Bring your own music metadata/playlist/audio-source with plugins created by community or by yourself. A small step towards the decentralized music streaming era! Btw it's not just another Electron app 😉 @@ -19,31 +18,24 @@ Btw it's not just another Electron app 😉 --- -![Spotube Desktop](assets/spotube-screenshot.png) +![Spotube Desktop](assets/branding/spotube-screenshot.png) -![Spotube Mobile](assets/mobile-screenshots/combined.png) +![Spotube Mobile](assets/branding/mobile-screenshots/combined.jpg)
## 🌃 Features -- 🚫 No ads, thanks to the use of public & free Spotify and YT Music APIs¹ -- ⬇️ Freely downloadable tracks -- 🖥️ 📱 Cross-platform support -- 🪶 Small size & less data usage -- 🕵️ Anonymous/guest login -- 🕒 Time synced lyrics -- ✋ No telemetry, diagnostics or user data collection -- 🚀 Native performance -- 📖 Open source/libre software -- 🔉 Playback control is done locally, not on the server - -**¹** It is still **recommended** to support creators by engaging with their YouTube channels/Spotify tracks (or preferably by buying their merch/concert tickets/physical media). - -### ❌ Unsupported features - -- 🗣️ **Spotify Shows & Podcasts:** Shows and Podcasts will **never be supported** because the audio tracks are _only_ available on Spotify and accessing them would require Spotify Premium. -- 🎧 **Spotify Listen Along:** [Coming soon!](https://github.com/KRTirtho/spotube/issues/8) +- 🧩 Plugin powered, supports any platform or custom music service through plugins. +- 🗺️ Community driven plugins for popular platforms or create your own. +- ⬇️ Freely downloadable tracks with tagged metadata. +- 🖥️ 📱 Cross-platform support. +- 🪶 Small size & less data usage. +- 🕒 Time synced lyrics regardless of the plugin support. +- ✋ No telemetry, diagnostics or user data collection. +- 🚀 Native performance. +- 📖 Open source/libre software. +- 🔉 Playback control is done locally, not on the server. ## 📜 ⬇️ Installation guide @@ -66,17 +58,13 @@ This handy table lists all the methods you can use to install Spotube: MacOS - MacOS Download + MacOS Download Android - - Get it on Google Play - -
APK download @@ -110,7 +98,7 @@ This handy table lists all the methods you can use to install Spotube: AppImage - AppImage's lacking stability led to it's temporal removal. More information at https://github.com/KRTirtho/spotube/issues/1082 + AppImage's lacking stability led to it's temporary removal. More information at https://github.com/KRTirtho/spotube/issues/1082 Debian/Ubuntu @@ -189,9 +177,7 @@ You can compile Spotube's source code by [following these instructions](CONTRIBU ## 👥 The Spotube team - [Kingkor Roy Tirtho](https://github.com/KRTirtho) - The Founder, Maintainer and Lead Developer -- [RaptaG](https://github.com/RaptaG) - The GitHub Moderator and Community Manager - [Owen Connor](https://github.com/owencz1998) - The Cool Discord Moderator -- [Meenbeese](https://github.com/meenbeese) - The Android Developer - [Piotr Rogowski](https://github.com/karniv00l) - The MacOS Developer - [Rusty Apple](https://github.com/RustyApple) - The Mysterious Unknown Guy @@ -199,7 +185,7 @@ You can compile Spotube's source code by [following these instructions](CONTRIBU Spotube is open source and licensed under the [BSD-4-Clause](/LICENSE) License. -If you are concerned, you can [read the reason of choosing this license](https://dev.to/krtirtho/choosing-open-source-license-wisely-1m3p). +If you are curious, you can [read the reason of choosing this license](https://dev.to/krtirtho/choosing-open-source-license-wisely-1m3p).
@@ -207,37 +193,39 @@ If you are concerned, you can [read the reason of choosing this license](https:/ ### Services + 1. [Flutter](https://flutter.dev) - Flutter transforms the app development process. Build, test, and deploy beautiful mobile, web, desktop, and embedded apps from a single codebase -1. [Spotify API](https://developer.spotify.com/documentation/web-api) - The Spotify Web API is a RESTful API that provides access to Spotify data +1. [MPV](https://mpv.io) - mpv is a free (as in freedom) media player for the command line. It supports a wide variety of media file formats, audio and video codecs, and subtitle types. +1. [Musicbrainz](https://musicbrainz.org) - MusicBrainz is a MetaBrainz project that aims to create a collaborative music database that is similar to the freedb project. +1. [Listenbrainz](https://listenbrainz.org) - ListenBrainz is a open-source project by the MetaBrainz Foundation that allows users to crowdsource and publicly store their digital music listening data. 1. [Piped](https://piped-docs.kavin.rocks/) - Piped is a privacy friendly alternative YouTube frontend, which is efficient and scalable by design. -1. [YouTube](https://youtube.com/) - YouTube is an American online video-sharing platform headquartered in San Bruno, California. Three former PayPal employees—Chad Hurley, Steve Chen, and Jawed Karim—created the service in February 2005 -1. [JioSaavn](https://www.jiosaavn.com) - JioSaavn is an Indian online music streaming service and a digital distributor of Bollywood, English and other regional Indian music across the world. Since it was founded in 2007 as Saavn, the company has acquired rights to over 5 crore (50 million) music tracks in 15 languages -1. [SongLink](https://song.link) - SongLink is a free smart link service that helps you share music with your audience. It's a one-stop-shop for creating smart links for music, podcasts, and other audio content -1. [LRCLib](https://lrclib.net/) - A public synced lyric API +1. [Invidious](https://invidious.io/) - Invidious is an open source alternative front-end to YouTube. +1. [yt-dlp](https://github.com/yt-dlp/yt-dlp) - A feature-rich command-line audio/video downloader. +1. [NewPipeExtractor](https://github.com/TeamNewPipe/NewPipeExtractor) - NewPipe's core library for extracting data from streaming sites. +1. [YouTubeExplodeDart](https://github.com/Hexer10/youtube_explode_dart) - A port in dart of the youtube explode library. Supports several API functions without the need of Youtube API Key. +1. [LRCLib](https://lrclib.net/) - A public synced lyric API. 1. [Linux](https://www.linux.org) - Linux is a family of open-source Unix-like operating systems based on the Linux kernel, an operating system kernel first released on September 17, 1991, by Linus Torvalds. Linux is typically packaged in a Linux distribution 1. [AUR](https://aur.archlinux.org) - AUR stands for Arch User Repository. It is a community-driven repository for Arch-based Linux distributions users 1. [Flatpak](https://flatpak.org) - Flatpak is a utility for software deployment and package management for Linux 1. [SponsorBlock](https://sponsor.ajay.app) - SponsorBlock is an open-source crowdsourced browser extension and open API for skipping sponsor segments in YouTube videos. 1. [Inno Setup](https://jrsoftware.org/isinfo.php) - Inno Setup is a free installer for Windows programs by Jordan Russell and Martijn Laan 1. [F-Droid](https://f-droid.org) - F-Droid is an installable catalogue of FOSS (Free and Open Source Software) applications for the Android platform. The client makes it easy to browse, install, and keep track of updates on your device -1. [LastFM](https://last.fm) - Last.fm is a music streaming and discovery platform that helps users discover and share new music. It tracks users' music listening habits across many devices and platforms. +1. [LastFM](https://last.fm) - Last.fm is a music streaming and discovery platform that helps users discover and share new music. It tracks users' music listening habits across many devices and platforms. ### Dependencies + 1. [app_links](https://github.com/llfbandit/app_links) - Android App Links, Deep Links, iOs Universal Links and Custom URL schemes handler for Flutter (desktop included). 1. [args](https://pub.dev/packages/args) - Library for defining parsers for parsing raw command-line arguments into a set of options and values using GNU and POSIX style options. -1. [async](https://pub.dev/packages/async) - Utility functions and classes related to the 'dart:async' library. 1. [audio_service](https://pub.dev/packages/audio_service) - Flutter plugin to play audio in the background while the screen is off. 1. [audio_service_mpris](https://github.com/bdrazhzhov/audio-service-mpris) - audio_service platform interface supporting Media Player Remote Interfacing Specification. 1. [audio_session](https://github.com/ryanheise/audio_session) - Sets the iOS audio session category and Android audio attributes for your app, and manages your app's audio focus, mixing and ducking behaviour. +1. [auto_route](https://github.com/Milad-Akarie/auto_route_library) - AutoRoute is a declarative routing solution, where everything needed for navigation is automatically generated for you. 1. [auto_size_text](https://github.com/leisim/auto_size_text) - Flutter widget that automatically resizes text to fit perfectly within its bounds. 1. [bonsoir](https://bonsoir.skyost.eu) - A Zeroconf library that allows you to discover network services and to broadcast your own. Based on Apple Bonjour and Android NSD. -1. [buttons_tabbar](https://afonsoraposo.com) - A Flutter package that implements a TabBar where each label is a toggle button. 1. [cached_network_image](https://github.com/Baseflow/flutter_cached_network_image) - Flutter library to load and cache network images. Can also be used with placeholder and error widgets. -1. [collection](https://pub.dev/packages/collection) - Collections and utilities functions and classes related to collections. -1. [curved_navigation_bar](https://github.com/rafalbednarczuk/curved_navigation_bar) - Stunning Animating Curved Shape Navigation Bar. Adjustable color, background color, animation curve, animation duration. +1. [connectivity_plus](https://github.com/fluttercommunity/plus_plugins) - Flutter plugin for discovering the state of the network (WiFi & mobile/cellular) connectivity on Android and iOS. 1. [device_info_plus](https://github.com/fluttercommunity/plus_plugins) - Flutter plugin providing detailed information about the device (make, model, etc.), and Android or iOS version the app is running on. 1. [dio](https://github.com/cfug/dio) - A powerful HTTP networking package,supports Interceptors,Aborting and canceling a request,Custom adapters, Transformers, etc. -1. [disable_battery_optimization](https://github.com/pvsvamsi/Disable-Battery-Optimizations) - Flutter plugin to check and disable battery optimizations. Also shows custom steps to disable the optimizations in devices like mi, xiaomi, samsung, oppo, huawei, oneplus etc 1. [drift](https://drift.simonbinder.eu/) - Drift is a reactive library to store relational data in Dart and Flutter applications. 1. [duration](https://github.com/desktop-dart/duration) - Utilities to make working with 'Duration's easier. Formats duration in human readable form and also parses duration in human readable form to Dart's Duration. 1. [encrypt](https://pub.dev/packages/encrypt) - A set of high-level APIs over PointyCastle for two-way cryptography. @@ -245,40 +233,31 @@ If you are concerned, you can [read the reason of choosing this license](https:/ 1. [file_picker](https://github.com/miguelpruivo/plugins_flutter_file_picker) - A package that allows you to use a native file explorer to pick single or multiple absolute file paths, with extension filtering support. 1. [file_selector](https://pub.dev/packages/file_selector) - Flutter plugin for opening and saving files, or selecting directories, using native file selection UI. 1. [fluentui_system_icons](https://github.com/microsoft/fluentui-system-icons/tree/main) - Fluent UI System Icons are a collection of familiar, friendly and modern icons from Microsoft. -1. [flutter_broadcasts](https://pub.dev/packages/flutter_broadcasts) - A plugin for sending and receiving broadcasts with Android intents and iOS notifications. 1. [flutter_cache_manager](https://github.com/Baseflow/flutter_cache_manager/tree/develop/flutter_cache_manager) - Generic cache manager for flutter. Saves web files on the storages of the device and saves the cache info using sqflite. 1. [flutter_discord_rpc](https://pub.dev/packages/flutter_discord_rpc) - Discord RPC support for Flutter desktop platforms 1. [flutter_displaymode](https://github.com/ajinasokan/flutter_displaymode) - A Flutter plugin to set display mode (resolution, refresh rate) on Android platform. Allows to enable high refresh rate on supported devices. 1. [flutter_feather_icons](https://github.com/muj-programmer/flutter_feather_icons) - Feather is a collection of simply beautiful open source icons. Each icon is designed on a 24x24 grid with an emphasis on simplicity, consistency and usability. +1. [flutter_form_builder](https://github.com/flutter-form-builder-ecosystem) - This package helps in creation of forms in Flutter by removing the boilerplate code, reusing validation, react to changes, and collect final user input. 1. [flutter_hooks](https://github.com/rrousselGit/flutter_hooks) - A flutter implementation of React hooks. It adds a new kind of widget with enhanced code reuse. 1. [flutter_inappwebview](https://inappwebview.dev/) - A Flutter plugin that allows you to add an inline webview, to use an headless webview, and to open an in-app browser window. 1. [flutter_native_splash](https://pub.dev/packages/flutter_native_splash) - Customize Flutter's default white native splash screen with background color and splash image. Supports dark mode, full screen, and more. 1. [flutter_riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze. -1. [flutter_secure_storage](https://pub.dev/packages/flutter_secure_storage) - Flutter Secure Storage provides API to store data in secure storage. Keychain is used in iOS, KeyStore based solution is used in Android. 1. [flutter_sharing_intent](https://github.com/bhagat-techind/flutter_sharing_intent.git) - A flutter plugin that allow flutter apps to receive photos, videos, text, urls or any other file types from another app. -1. [flutter_svg](https://pub.dev/packages/flutter_svg) - An SVG rendering and widget library for Flutter, which allows painting and displaying Scalable Vector Graphics 1.1 files. -1. [form_validator](https://github.com/TheMisir/form-validator) - Simplest form validation library for flutter's form field widgets +1. [flutter_undraw](https://github.com/KRTirtho/flutter_undraw) - Undraw.co Illustrations for Flutter with customization options +1. [form_builder_validators](https://github.com/flutter-form-builder-ecosystem) - Form Builder Validators set of validators for FlutterFormBuilder. Provides common validators and a way to make your own. 1. [freezed_annotation](https://pub.dev/packages/freezed_annotation) - Annotations for the freezed code-generator. This package does nothing without freezed too. 1. [fuzzywuzzy](https://github.com/sphericalkat/dart-fuzzywuzzy) - An implementation of the popular fuzzywuzzy package in Dart, to suit all your fuzzy string matching/searching needs! -1. [gap](https://github.com/letsar/gap) - Flutter widgets for easily adding gaps inside Flex widgets such as Columns and Rows or scrolling views. -1. [go_router](https://pub.dev/packages/go_router) - A declarative router for Flutter based on Navigation 2 supporting deep linking, data-driven routes and more -1. [google_fonts](https://pub.dev/packages/google_fonts) - A Flutter package to use fonts from fonts.google.com. Supports HTTP fetching, caching, and asset bundling. -1. [hive](https://github.com/hivedb/hive/tree/master/hive) - Lightweight and blazing fast key-value database written in pure Dart. Strongly encrypted using AES-256. -1. [hive_flutter](https://github.com/hivedb/hive/tree/master/hive_flutter) - Extension for Hive. Makes it easier to use Hive in Flutter apps. +1. [home_widget](https://pub.dev/packages/home_widget) - A plugin to provide a common interface for creating HomeScreen Widgets for Android and iOS. 1. [hooks_riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze. 1. [html](https://pub.dev/packages/html) - APIs for parsing and manipulating HTML content outside the browser. 1. [html_unescape](https://github.com/filiph/html_unescape) - A small library for un-escaping HTML. Supports all Named Character References, Decimal Character References and Hexadecimal Character References. 1. [http](https://pub.dev/packages/http) - A composable, multi-platform, Future-based API for HTTP requests. 1. [image_picker](https://pub.dev/packages/image_picker) - Flutter plugin for selecting images from the Android and iOS image library, and taking new pictures with the camera. 1. [intl](https://pub.dev/packages/intl) - Contains code to deal with internationalized/localized messages, date and number formatting and parsing, bi-directional text, and other internationalization issues. -1. [invidious](https://pub.dev/packages/invidious) - Invidious API client for Dart and Flutter. -1. [jiosaavn](https://github.com/KRTirtho/jiosaavn) - Unofficial API client for jiosaavn.com -1. [json_annotation](https://pub.dev/packages/json_annotation) - Classes and helper functions that support JSON code generation via the `json_serializable` package. 1. [local_notifier](https://github.com/leanflutter/local_notifier) - This plugin allows Flutter desktop apps to displaying local notifications. 1. [logger](https://pub.dev/packages/logger) - Small, easy to use and extensible logger which prints beautiful logs. +1. [logging](https://pub.dev/packages/logging) - Provides APIs for debugging and error logging, similar to loggers in other languages, such as the Closure JS Logger and java.util.logging.Logger. 1. [lrc](https://pub.dev/packages/lrc) - A Dart-only package that creates, parses, and handles LRC, which is a format that stores song lyrics. -1. [media_kit](https://github.com/media-kit/media-kit) - A cross-platform video player & audio player for Flutter & Dart. Performant, stable, feature-proof & modular. -1. [media_kit_libs_audio](https://github.com/media-kit/media-kit.git) - package:media_kit audio (only) playback native libraries for all platforms. 1. [metadata_god](https://pub.dev/packages/metadata_god) - Plugin for retrieving and writing audio tags/metadata from audio files 1. [mime](https://pub.dev/packages/mime) - Utilities for handling media (MIME) types, including determining a type from a file extension and file contents. 1. [open_file](https://pub.dev/packages/open_file) - A plug-in that can call native APP to open files with string result in flutter, support iOS(UTI) / android(intent) / PC(ffi) / web(dart:html) @@ -287,25 +266,20 @@ If you are concerned, you can [read the reason of choosing this license](https:/ 1. [path](https://pub.dev/packages/path) - A string-based path manipulation library. All of the path operations you know and love, with solid support for Windows, POSIX (Linux and Mac OS X), and the web. 1. [path_provider](https://pub.dev/packages/path_provider) - Flutter plugin for getting commonly used locations on host platform file systems, such as the temp and app data directories. 1. [permission_handler](https://pub.dev/packages/permission_handler) - Permission plugin for Flutter. This plugin provides a cross-platform (iOS, Android) API to request and check permissions. -1. [piped_client](https://github.com/KRTirtho/piped_client) - API Client for piped.video -1. [popover](https://github.com/minikin/popover) - A popover is a transient view that appears above other content onscreen when you tap a control or in an area. 1. [riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze. 1. [scroll_to_index](https://github.com/quire-io/scroll-to-index) - Scroll to a specific child of any scrollable widget in Flutter +1. [shadcn_flutter](https://github.com/sunarya-thito/shadcn_flutter) - Beautifully designed components from Shadcn/UI is now available for Flutter 1. [shared_preferences](https://pub.dev/packages/shared_preferences) - Flutter plugin for reading and writing simple key-value pairs. Wraps NSUserDefaults on iOS and SharedPreferences on Android. 1. [shelf](https://pub.dev/packages/shelf) - A model for web server middleware that encourages composition and easy reuse. 1. [shelf_router](https://pub.dev/packages/shelf_router) - A convenient request router for the shelf web-framework, with support for URL-parameters, nested routers and routers generated from source annotations. 1. [shelf_web_socket](https://pub.dev/packages/shelf_web_socket) - A shelf handler that wires up a listener for every connection. -1. [sidebarx](https://github.com/Frezyx/sidebarx) - flutter multiplatform navigation sidebar / side navigationbar / drawer widget 1. [simple_icons](https://teavelopment.com/) - The Simple Icon pack available as Flutter Icons. Provides over 1500 Free SVG icons for popular brands. 1. [skeletonizer](https://github.com/Milad-Akarie/skeletonizer) - Converts already built widgets into skeleton loaders with no extra effort. +1. [sliding_up_panel](https://github.com/akshathjain/sliding_up_panel) - A draggable Flutter widget that makes implementing a SlidingUpPanel much easier! 1. [sliver_tools](https://github.com/Kavantix) - A set of useful sliver tools that are missing from the flutter framework 1. [smtc_windows](https://pub.dev/packages/smtc_windows) - Windows `SystemMediaTransportControls` implementation for Flutter giving access to Windows OS Media Control applet. -1. [spotify](https://github.com/rinukkusu/spotify-dart) - An incomplete dart library for interfacing with the Spotify Web API. 1. [sqlite3](https://github.com/simolus3/sqlite3.dart/tree/main/sqlite3) - Provides lightweight yet convenient bindings to SQLite by using dart:ffi 1. [sqlite3_flutter_libs](https://github.com/simolus3/sqlite3.dart/tree/main/sqlite3_flutter_libs) - Flutter plugin to include native sqlite3 libraries with your app -1. [stroke_text](https://github.com/MohamedAbd0/stroke_text) - A Simple Flutter plugin for applying stroke (border) style to a text widget -1. [system_theme](https://github.com/bdlukaa/system_theme/tree/master/system_theme) - A plugin to get the current system theme info. Supports Android, Web, Windows, Linux and macOS -1. [test](https://pub.dev/packages/test) - A full featured library for writing and running Dart tests across platforms. 1. [timezone](https://pub.dev/packages/timezone) - Time zone database and time zone aware DateTime. 1. [titlebar_buttons](https://github.com/gtk-flutter/titlebar_buttons) - A package which provides most of the titlebar buttons from windows, linux and macos. 1. [tray_manager](https://github.com/leanflutter/tray_manager) - This plugin allows Flutter desktop apps to defines system tray. @@ -317,28 +291,44 @@ If you are concerned, you can [read the reason of choosing this license](https:/ 1. [web_socket_channel](https://pub.dev/packages/web_socket_channel) - StreamChannel wrappers for WebSockets. Provides a cross-platform WebSocketChannel API, a cross-platform implementation of that API that communicates over an underlying StreamChannel. 1. [wikipedia_api](https://github.com/KRTirtho/wikipedia_api) - Wikipedia API for dart and flutter 1. [win32_registry](https://pub.dev/packages/win32_registry) - A package that provides a friendly Dart API for accessing the Windows Registry. -1. [window_manager](https://github.com/leanflutter/window_manager) - This plugin allows Flutter desktop apps to resizing and repositioning the window. +1. [window_manager](https://leanflutter.dev) - This plugin allows Flutter desktop apps to resizing and repositioning the window. 1. [youtube_explode_dart](https://github.com/Hexer10/youtube_explode_dart) - A port in dart of the youtube explode library. Supports several API functions without the need of Youtube API Key. +1. [http_parser](https://pub.dev/packages/http_parser) - A platform-independent package for parsing and serializing HTTP formats. +1. [collection](https://pub.dev/packages/collection) - Collections and utilities functions and classes related to collections. +1. [archive](https://pub.dev/packages/archive) - Provides encoders and decoders for various archive and compression formats such as zip, tar, bzip2, gzip, and zlib. +1. [hetu_script](https://github.com/hetu-script/hetu-script) - Hetu is a lightweight scripting language for embedding in Flutter apps. +1. [get_it](https://github.com/flutter-it/get_it) - Simple direct Service Locator that allows to decouple the interface from a concrete implementation and to access the concrete implementation from everywhere in your App" +1. [flutter_markdown_plus](https://pub.dev/packages/flutter_markdown_plus) - A Markdown renderer for Flutter. Create rich text output, including text styles, tables, links, and more, from plain text data formatted with simple Markdown tags. +1. [pub_semver](https://pub.dev/packages/pub_semver) - Versions and version constraints implementing pub's versioning policy. This is very similar to vanilla semver, with a few corner cases. +1. [change_case](https://github.com/mrgnhnt96/change_case) - An extension on String for the missing methods for camelCase, PascalCase, Capital Case, snake_case, param-case, CONSTANT_CASE and others. +1. [flutter_secure_storage](https://pub.dev/packages/flutter_secure_storage) - Flutter Secure Storage provides API to store data in secure storage. Keychain is used in iOS, KeyStore based solution is used in Android. 1. [build_runner](https://pub.dev/packages/build_runner) - A build system for Dart code generation and modular compilation. -1. [crypto](https://pub.dev/packages/crypto) - Implementations of SHA, MD5, and HMAC cryptographic functions. 1. [envied_generator](https://github.com/petercinibulk/envied) - Generator for the Envied package. See https://pub.dev/packages/envied. 1. [flutter_gen_runner](https://github.com/FlutterGen/flutter_gen) - The Flutter code generator for your assets, fonts, colors, … — Get rid of all String-based APIs. 1. [flutter_launcher_icons](https://github.com/fluttercommunity/flutter_launcher_icons) - A package which simplifies the task of updating your Flutter app's launcher icon. 1. [flutter_lints](https://pub.dev/packages/flutter_lints) - Recommended lints for Flutter apps, packages, and plugins to encourage good coding practices. -1. [hive_generator](https://github.com/hivedb/hive/tree/master/hive_generator) - Extension for Hive. Automatically generates TypeAdapters to store any class. 1. [json_serializable](https://pub.dev/packages/json_serializable) - Automatically generate code for converting to and from JSON by annotating Dart classes. 1. [freezed](https://pub.dev/packages/freezed) - Code generation for immutable classes that has a simple syntax/API without compromising on the features. -1. [custom_lint](https://pub.dev/packages/custom_lint) - Lint rules are a powerful way to improve the maintainability of a project. Custom Lint allows package authors and developers to easily write custom lint rules. -1. [riverpod_lint](https://riverpod.dev) - Riverpod_lint is a developer tool for users of Riverpod, designed to help stop common issues and simplify repetitive tasks. 1. [process_run](https://github.com/tekartik/process_run.dart/blob/master/packages/process_run) - Process run helpers for Linux/Win/Mac and which like feature for finding executables. 1. [pubspec_parse](https://pub.dev/packages/pubspec_parse) - Simple package for parsing pubspec.yaml files with a type-safe API and rich error reporting. 1. [pub_api_client](https://github.com/leoafarias/pub_api_client) - An API Client for Pub to interact with public package information. -1. [xml](https://github.com/renggli/dart-xml) - A lightweight library for parsing, traversing, querying, transforming and building XML documents. 1. [io](https://pub.dev/packages/io) - Utilities for the Dart VM Runtime including support for ANSI colors, file copying, and standard exit code values. 1. [drift_dev](https://drift.simonbinder.eu/) - Dev-dependency for users of drift. Contains the generator and development tools. +1. [test](https://pub.dev/packages/test) - A full featured library for writing and running Dart tests across platforms. +1. [auto_route_generator](https://github.com/Milad-Akarie/auto_route_library) - AutoRoute is a declarative routing solution, where everything needed for navigation is automatically generated for you. 1. [desktop_webview_window](https://github.com/MixinNetwork/flutter-plugins/tree/main/packages/desktop_webview_window) - Show a webview window on your flutter desktop application. +1. [disable_battery_optimization](https://github.com/pvsvamsi/Disable-Battery-Optimizations) - Flutter plugin to check and disable battery optimizations. Also shows custom steps to disable the optimizations in devices like mi, xiaomi, samsung, oppo, huawei, oneplus etc 1. [draggable_scrollbar](https://github.com/fluttercommunity/flutter-draggable-scrollbar) - A scrollbar that can be dragged for quickly navigation through a vertical list. Additional option is showing label next to scrollthumb with information about current item. +1. [flutter_broadcasts](https://github.com/KRTirtho/flutter_broadcasts.git) - A plugin for sending and receiving broadcasts with Android intents and iOS notifications. 1. [scrobblenaut](https://github.com/Nebulino/Scrobblenaut) - A deadly simple LastFM API Wrapper for Dart. So deadly simple that it's gonna hit the mark. +1. [yt_dlp_dart](https://github.com/KRTirtho/yt_dlp_dart.git) - A starting point for Dart libraries or applications. +1. [flutter_new_pipe_extractor](https://github.com/KRTirtho/flutter_new_pipe_extractor) - NewPipeExtractor binding for Flutter (Android only) +1. [hetu_std](https://github.com/hetu-community/hetu_std.git) - A sample command-line application. +1. [hetu_otp_util](https://github.com/hetu-community/hetu_otp_util.git) - A sample command-line application. +1. [hetu_spotube_plugin](https://github.com/KRTirtho/hetu_spotube_plugin) - A new Flutter package project. +1. [media_kit](https://github.com/media-kit/media-kit) - A cross-platform video player & audio player for Flutter & Dart. Performant, stable, feature-proof & modular. +1. [media_kit_libs_audio](https://github.com/media-kit/media-kit.git) - package:media_kit audio (only) playback native libraries for all platforms. +
-

© Copyright Spotube 2024

+

© Copyright Spotube 2025

diff --git a/analysis_options.yaml b/analysis_options.yaml index 1eda286e..af222653 100644 --- a/analysis_options.yaml +++ b/analysis_options.yaml @@ -32,8 +32,6 @@ linter: analyzer: errors: invalid_annotation_target: ignore - plugins: - - custom_lint exclude: - "**.freezed.dart" - "**.g.dart" diff --git a/android/.gitignore b/android/.gitignore index 6f568019..2391a77e 100644 --- a/android/.gitignore +++ b/android/.gitignore @@ -11,3 +11,4 @@ GeneratedPluginRegistrant.java key.properties **/*.keystore **/*.jks +.kotlin \ No newline at end of file diff --git a/android/app/build.gradle b/android/app/build.gradle index 8ec1872e..7319c6a8 100644 --- a/android/app/build.gradle +++ b/android/app/build.gradle @@ -2,6 +2,7 @@ plugins { id "com.android.application" id "kotlin-android" id "dev.flutter.flutter-gradle-plugin" + id "org.jetbrains.kotlin.plugin.compose" } def localProperties = new Properties() @@ -28,12 +29,17 @@ if (keystorePropertiesFile.exists()) { keystoreProperties.load(new FileInputStream(keystorePropertiesFile)) } -android { - compileSdkVersion 34 +def composeVersion = "1.4.8" - ndkVersion "25.1.8937393" +android { + namespace "oss.krtirtho.spotube" + + compileSdkVersion 36 + + ndkVersion = "29.0.14206865" compileOptions { + coreLibraryDesugaringEnabled true sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 } @@ -46,10 +52,18 @@ android { main.java.srcDirs += 'src/main/kotlin' } + buildFeatures { + compose true + } + + composeOptions { + kotlinCompilerExtensionVersion "$composeVersion" // Correlates with org.jetbrains.kotlin.android plugin in settings.gradle + } + defaultConfig { applicationId "oss.krtirtho.spotube" minSdkVersion 24 - targetSdkVersion 34 + targetSdkVersion 35 versionCode flutterVersionCode.toInteger() versionName flutterVersionName multiDexEnabled true @@ -63,6 +77,7 @@ android { storePassword keystoreProperties['storePassword'] } } + buildTypes { release { signingConfig signingConfigs.release @@ -96,15 +111,30 @@ android { } } + packagingOptions { + resources.excludes += "DebugProbesKt.bin" + } } flutter { source '../..' } +def glanceVersion = "1.1.1" dependencies { - implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.5.1' + coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:2.1.4' + implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.5.1' // other deps so just ignore implementation 'com.android.support:multidex:2.0.1' + + implementation "androidx.glance:glance-appwidget:$glanceVersion" + implementation "androidx.glance:glance-appwidget-preview:$glanceVersion" + implementation "androidx.glance:glance-preview:$glanceVersion" + implementation "androidx.glance:glance-material3:$glanceVersion" + implementation "androidx.glance:glance-material:$glanceVersion" + implementation "androidx.work:work-runtime-ktx:2.8.1" + + implementation "org.jetbrains.kotlinx:kotlinx-serialization-json:1.7.3" + implementation 'com.google.code.gson:gson:2.11.0' } \ No newline at end of file diff --git a/android/app/proguard-rules.pro b/android/app/proguard-rules.pro index 116bc22f..1f5a556c 100644 --- a/android/app/proguard-rules.pro +++ b/android/app/proguard-rules.pro @@ -1 +1,59 @@ --keep class androidx.lifecycle.DefaultLifecycleObserver \ No newline at end of file +#Flutter Wrapper +# -keep class io.flutter.app.** { *; } +-keep class io.flutter.plugin.** { *; } +-keep class io.flutter.util.** { *; } +-keep class io.flutter.view.** { *; } +# -keep class io.flutter.** { *; } +-keep class io.flutter.plugins.** { *; } +-keep class de.prosiebensat1digital.** { *; } + +-keep class androidx.lifecycle.DefaultLifecycleObserver + +-keepnames class kotlinx.serialization.** { *; } +-keepnames class oss.krtirtho.spotube.glance.models.** { *; } +-keep @kotlinx.serialization.Serializable class * +-keepclassmembers class ** { + @kotlinx.serialization.* ; +} + +## We don't need beans +-dontwarn java.beans.BeanDescriptor +-dontwarn java.beans.BeanInfo +-dontwarn java.beans.IntrospectionException +-dontwarn java.beans.Introspector +-dontwarn java.beans.PropertyDescriptor + +## Rules for NewPipeExtractor +-keep class org.schabi.newpipe.extractor.timeago.patterns.** { *; } +-keep class org.mozilla.javascript.** { *; } +-keep class org.mozilla.classfile.ClassFileWriter +-dontwarn org.mozilla.javascript.tools.** + +-dontwarn javax.script.AbstractScriptEngine +-dontwarn javax.script.Bindings +-dontwarn javax.script.Compilable +-dontwarn javax.script.CompiledScript +-dontwarn javax.script.Invocable +-dontwarn javax.script.ScriptContext +-dontwarn javax.script.ScriptEngine +-dontwarn javax.script.ScriptEngineFactory +-dontwarn javax.script.ScriptException +-dontwarn javax.script.SimpleBindings +-dontwarn jdk.dynalink.CallSiteDescriptor +-dontwarn jdk.dynalink.DynamicLinker +-dontwarn jdk.dynalink.DynamicLinkerFactory +-dontwarn jdk.dynalink.NamedOperation +-dontwarn jdk.dynalink.Namespace +-dontwarn jdk.dynalink.NamespaceOperation +-dontwarn jdk.dynalink.Operation +-dontwarn jdk.dynalink.RelinkableCallSite +-dontwarn jdk.dynalink.StandardNamespace +-dontwarn jdk.dynalink.StandardOperation +-dontwarn jdk.dynalink.linker.GuardedInvocation +-dontwarn jdk.dynalink.linker.GuardingDynamicLinker +-dontwarn jdk.dynalink.linker.LinkRequest +-dontwarn jdk.dynalink.linker.LinkerServices +-dontwarn jdk.dynalink.linker.TypeBasedGuardingDynamicLinker +-dontwarn jdk.dynalink.linker.support.CompositeTypeBasedGuardingDynamicLinker +-dontwarn jdk.dynalink.linker.support.Guards +-dontwarn jdk.dynalink.support.ChainedCallSite \ No newline at end of file diff --git a/android/app/src/debug/AndroidManifest.xml b/android/app/src/debug/AndroidManifest.xml index 1041f6ca..400c91e8 100644 --- a/android/app/src/debug/AndroidManifest.xml +++ b/android/app/src/debug/AndroidManifest.xml @@ -1,7 +1,19 @@ - - - - + + + + + + \ No newline at end of file diff --git a/android/app/src/main/AndroidManifest.xml b/android/app/src/main/AndroidManifest.xml index 64c32e28..a005257e 100644 --- a/android/app/src/main/AndroidManifest.xml +++ b/android/app/src/main/AndroidManifest.xml @@ -1,4 +1,4 @@ - + @@ -17,38 +17,36 @@ + android:usesCleartextTraffic="true"> + android:value="false" /> --> + android:windowSoftInputMode="adjustResize"> + Specifies an Android theme to apply to this Activity as soon as + the Android process has started. This theme is visible to the user + while the Flutter UI initializes. After that, this theme continues + to determine the Window background behind the Flutter UI. + --> + android:resource="@style/NormalTheme" /> + @@ -56,12 +54,9 @@ + - @@ -72,23 +67,28 @@ + - - + + + + - + - @@ -96,11 +96,40 @@ - + + + + + + + + + + + + + + + + + - + This is used by the Flutter tool to generate GeneratedPluginRegistrant.java --> + \ No newline at end of file diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/HomePlayerWidget.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/HomePlayerWidget.kt new file mode 100644 index 00000000..a20af959 --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/HomePlayerWidget.kt @@ -0,0 +1,207 @@ +package oss.krtirtho.spotube.glance + +import HomeWidgetGlanceState +import HomeWidgetGlanceStateDefinition +import android.R +import android.content.Context +import android.graphics.drawable.Icon +import android.net.Uri +import android.util.Log +import androidx.compose.runtime.Composable +import androidx.compose.ui.unit.DpSize +import androidx.compose.ui.unit.dp +import androidx.glance.GlanceId +import androidx.glance.GlanceModifier +import androidx.glance.GlanceTheme +import androidx.glance.Image +import androidx.glance.ImageProvider +import androidx.glance.LocalSize +import androidx.glance.action.ActionParameters +import androidx.glance.action.actionParametersOf +import androidx.glance.action.clickable +import androidx.glance.background +import androidx.glance.appwidget.GlanceAppWidget +import androidx.glance.appwidget.SizeMode +import androidx.glance.appwidget.action.ActionCallback +import androidx.glance.appwidget.action.actionRunCallback +import androidx.glance.appwidget.background +import androidx.glance.appwidget.components.CircleIconButton +import androidx.glance.appwidget.components.Scaffold +import androidx.glance.appwidget.cornerRadius +import androidx.glance.appwidget.provideContent +import androidx.glance.background +import androidx.glance.currentState +import androidx.glance.layout.Alignment +import androidx.glance.layout.Box +import androidx.glance.layout.Column +import androidx.glance.layout.ContentScale +import androidx.glance.layout.Row +import androidx.glance.layout.Spacer +import androidx.glance.layout.fillMaxSize +import androidx.glance.layout.fillMaxWidth +import androidx.glance.layout.padding +import androidx.glance.layout.size +import androidx.glance.preview.ExperimentalGlancePreviewApi +import androidx.glance.preview.Preview +import androidx.glance.state.GlanceStateDefinition +import com.google.gson.Gson +import es.antonborri.home_widget.HomeWidgetBackgroundIntent +import es.antonborri.home_widget.actionStartActivity +import oss.krtirtho.spotube.MainActivity +import oss.krtirtho.spotube.glance.models.Track +import oss.krtirtho.spotube.glance.widgets.FlutterAssetImageProvider +import oss.krtirtho.spotube.glance.widgets.TrackDetailsView +import oss.krtirtho.spotube.glance.widgets.TrackProgress + +val gson = Gson() +val serverAddressKey = ActionParameters.Key("serverAddress") + +class Breakpoints { + companion object { + val SMALL_SQUARE = DpSize(100.dp, 100.dp) + val HORIZONTAL_RECTANGLE = DpSize(250.dp, 100.dp) + val BIG_SQUARE = DpSize(250.dp, 250.dp) + } +} + +class HomePlayerWidget : GlanceAppWidget() { + + override val sizeMode = SizeMode.Responsive( + setOf( + Breakpoints.SMALL_SQUARE, + Breakpoints.HORIZONTAL_RECTANGLE, + Breakpoints.BIG_SQUARE + ) + ) + + override val stateDefinition: GlanceStateDefinition<*>? + get() = HomeWidgetGlanceStateDefinition() + + override suspend fun provideGlance(context: Context, id: GlanceId) { + provideContent { + GlanceContent(context, currentState()) + } + } + + + @OptIn(ExperimentalGlancePreviewApi::class) + @Preview(widthDp = 100, heightDp = 100) + @Composable + private fun GlanceContent(context: Context, currentState: HomeWidgetGlanceState) { + val prefs = currentState.preferences + val size = LocalSize.current + + val activeTrackStr = prefs.getString("activeTrack", null) + + val isPlaying = prefs.getBoolean("isPlaying", false) + val playbackServerAddress = prefs.getString("playbackServerAddress", null) ?: "" + + var activeTrack: Track? = null + if (activeTrackStr != null) { + activeTrack = gson.fromJson(activeTrackStr, Track::class.java) + } + + + val playIcon = Icon.createWithResource(context, R.drawable.ic_media_play); + val pauseIcon = Icon.createWithResource(context, R.drawable.ic_media_pause); + val previousIcon = Icon.createWithResource(context, R.drawable.ic_media_previous); + val nextIcon = Icon.createWithResource(context, R.drawable.ic_media_next); + + GlanceTheme { + Box( + modifier = GlanceModifier + .fillMaxSize() + .cornerRadius(8.dp) + .background( + color = GlanceTheme.colors.surface.getColor(context) + ) + .clickable { + actionStartActivity(context) + } + , + ) { + Box( + modifier = GlanceModifier + .background( + color = + GlanceTheme.colors.surface.getColor(context) + .copy(alpha = 0.5f), + ) + .fillMaxSize(), + ) {} + Column( + modifier = GlanceModifier.padding(top = 10.dp, start = 10.dp, end = 10.dp) + ) { + Row(verticalAlignment = Alignment.Vertical.CenterVertically) { + TrackDetailsView(activeTrack) + } + Spacer(modifier = GlanceModifier.size(6.dp)) + if (size != Breakpoints.SMALL_SQUARE) { + TrackProgress(prefs) + } + Spacer(modifier = GlanceModifier.size(6.dp)) + Row( + modifier = GlanceModifier.fillMaxWidth(), + horizontalAlignment = Alignment.Horizontal.CenterHorizontally + ) { + CircleIconButton( + imageProvider = ImageProvider(previousIcon), + contentDescription = "Previous", + onClick = actionRunCallback( + parameters = actionParametersOf(serverAddressKey to playbackServerAddress) + ) + ) + Spacer(modifier = GlanceModifier.size(6.dp)) + CircleIconButton( + imageProvider = + if (isPlaying) ImageProvider(pauseIcon) + else ImageProvider(playIcon), + contentDescription = "Play/Pause", + onClick = actionRunCallback( + parameters = actionParametersOf(serverAddressKey to playbackServerAddress) + ) + ) + Spacer(modifier = GlanceModifier.size(6.dp)) + CircleIconButton( + imageProvider = ImageProvider(nextIcon), + contentDescription = "Previous", + onClick = actionRunCallback( + parameters = actionParametersOf( + serverAddressKey to playbackServerAddress + ) + ) + ) + } + } + } + } + } +} + +class PlayPauseAction : InteractiveAction("toggle-playback") +class NextAction : InteractiveAction("next") +class PreviousAction : InteractiveAction("previous") + + +abstract class InteractiveAction(val command: String) : ActionCallback { + override suspend fun onAction( + context: Context, + glanceId: GlanceId, + parameters: ActionParameters + ) { + val serverAddress = parameters[serverAddressKey] ?: "" + + Log.d("HomePlayerWidget", "Sending command $command to $serverAddress") + + if (serverAddress == null || serverAddress.isEmpty()) { + return + } + + + val backgroundIntent = HomeWidgetBackgroundIntent.getBroadcast( + context, + Uri.parse("spotube://playback/$command?serverAddress=$serverAddress") + ) + backgroundIntent.send() + } +} diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/HomePlayerWidgetReceiver.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/HomePlayerWidgetReceiver.kt new file mode 100644 index 00000000..2d23c64f --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/HomePlayerWidgetReceiver.kt @@ -0,0 +1,7 @@ +package oss.krtirtho.spotube.glance + +import HomeWidgetGlanceWidgetReceiver + +class HomePlayerWidgetReceiver : HomeWidgetGlanceWidgetReceiver() { + override val glanceAppWidget = HomePlayerWidget() +} diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/AlbumSimple.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/AlbumSimple.kt new file mode 100644 index 00000000..4edd69f6 --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/AlbumSimple.kt @@ -0,0 +1,40 @@ +package oss.krtirtho.spotube.glance.models + +import com.google.gson.annotations.SerializedName +import kotlinx.serialization.Serializable + +@Serializable +data class AlbumSimple( + @SerializedName("album_type") + val albumType: AlbumType?, + + @SerializedName("available_markets") + val availableMarkets: List?, + + val href: String?, + val id: String?, + val images: List?, + val name: String?, + + @SerializedName("release_date") + val releaseDate: String?, + + @SerializedName("release_date_precision") + val releaseDatePrecision: DatePrecision?, + + val type: String?, + val uri: String?, +) + +@Serializable +enum class AlbumType { + album, + single, + compilation +} + +enum class DatePrecision { + year, + month, + day +} \ No newline at end of file diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Artist.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Artist.kt new file mode 100644 index 00000000..ef43ecc8 --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Artist.kt @@ -0,0 +1,25 @@ +package oss.krtirtho.spotube.glance.models + +import com.google.gson.annotations.SerializedName +import kotlinx.serialization.Serializable + +@Serializable +data class Artist( + val href: String?, + val id: String?, + val name: String?, + val type: String?, + val uri: String?, + + val followers: Followers?, + val genres: List?, + val images: List?, + + @SerializedName("popularity") + val popularity: Int? +) + +@Serializable +data class Followers( + val total: Int? +) diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Image.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Image.kt new file mode 100644 index 00000000..de7d5521 --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Image.kt @@ -0,0 +1,10 @@ +package oss.krtirtho.spotube.glance.models + +import kotlinx.serialization.Serializable + +@Serializable +data class Image( + val height: Int?, + val width: Int?, + val path: String, +) \ No newline at end of file diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Track.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Track.kt new file mode 100644 index 00000000..717b790f --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/models/Track.kt @@ -0,0 +1,37 @@ +package oss.krtirtho.spotube.glance.models + +import com.google.gson.annotations.SerializedName +import kotlinx.serialization.Serializable +import kotlin.time.Duration.Companion.milliseconds + +@Serializable +data class Track( + val album: AlbumSimple?, val artists: List?, + + @SerializedName("available_markets") val availableMarkets: List?, + + @SerializedName("disc_number") val discNumber: Int?, + + @SerializedName("duration_ms") val durationMs: Int, + + val explicit: Boolean?, val href: String?, val id: String?, + + @SerializedName("is_playable") val isPlayable: Boolean?, + + val name: String?, + + @SerializedName("popularity") val popularity: Int?, + + @SerializedName("preview_url") val previewUrl: String?, + + @SerializedName("track_number") val trackNumber: Int?, + + val type: String?, val uri: String? +) { + val duration: kotlin.time.Duration + get() = durationMs.toLong().milliseconds +} + +enum class Market { + AD, AE, AF, AG, AI, AL, AM, AO, AQ, AR, AS, AT, AU, AW, AX, AZ, BA, BB, BD, BE, BF, BG, BH, BI, BJ, BL, BM, BN, BO, BQ, BR, BS, BT, BV, BW, BY, BZ, CA, CC, CD, CF, CG, CH, CI, CK, CL, CM, CN, CO, CR, CU, CV, CW, CX, CY, CZ, DE, DJ, DK, DM, DO, DZ, EC, EE, EG, EH, ER, ES, ET, FI, FJ, FK, FM, FO, FR, GA, GB, GD, GE, GF, GG, GH, GI, GL, GM, GN, GP, GQ, GR, GS, GT, GU, GW, GY, HK, HM, HN, HR, HT, HU, ID, IE, IL, IM, IN, IO, IQ, IR, IS, IT, JE, JM, JO, JP, KE, KG, KH, KI, KM, KN, KP, KR, KW, KY, KZ, LA, LB, LC, LI, LK, LR, LS, LT, LU, LV, LY, MA, MC, MD, ME, MF, MG, MH, MK, ML, MM, MN, MO, MP, MQ, MR, MS, MT, MU, MV, MW, MX, MY, MZ, NA, NC, NE, NF, NG, NI, NL, NO, NP, NR, NU, NZ, OM, PA, PE, PF, PG, PH, PK, PL, PM, PN, PR, PS, PT, PW, PY, QA, RE, RO, RS, RU, RW, SA, SB, SC, SD, SE, SG, SH, SI, SJ, SK, SL, SM, SN, SO, SR, SS, ST, SV, SX, SY, SZ, TC, TD, TF, TG, TH, TJ, TK, TL, TM, TN, TO, TR, TT, TV, TW, TZ, UA, UG, UM, US, UY, UZ, VA, VC, VE, VG, VI, VN, VU, WF, WS, XK, YE, YT, ZA, ZM, ZW, +} diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/Base64ImageProvider.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/Base64ImageProvider.kt new file mode 100644 index 00000000..79339cea --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/Base64ImageProvider.kt @@ -0,0 +1,14 @@ +package oss.krtirtho.spotube.glance.widgets + +import android.graphics.BitmapFactory +import android.util.Base64 +import androidx.glance.ImageProvider + +@Suppress("FunctionName") +fun Base64ImageProvider(base64: String): ImageProvider { + var bytes = Base64.decode(base64, Base64.DEFAULT); + + var bitmap = BitmapFactory.decodeByteArray(bytes, 0, bytes.size); + + return ImageProvider(bitmap) +} \ No newline at end of file diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/FlutterAssetImageProvider.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/FlutterAssetImageProvider.kt new file mode 100644 index 00000000..ad51ca3c --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/FlutterAssetImageProvider.kt @@ -0,0 +1,14 @@ +package oss.krtirtho.spotube.glance.widgets + +import android.content.Context +import android.graphics.BitmapFactory +import androidx.glance.ImageProvider + +@Suppress("FunctionName") +fun FlutterAssetImageProvider(context: Context, path: String): ImageProvider { + var inputStream = context.assets.open("flutter_assets/$path") + + return ImageProvider( + BitmapFactory.decodeStream(inputStream) + ) +} \ No newline at end of file diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/TrackDetailsView.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/TrackDetailsView.kt new file mode 100644 index 00000000..fdfe8e4b --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/TrackDetailsView.kt @@ -0,0 +1,78 @@ +package oss.krtirtho.spotube.glance.widgets + +import android.graphics.BitmapFactory +import android.net.Uri +import android.util.Log +import androidx.compose.runtime.Composable +import androidx.compose.ui.unit.dp +import androidx.compose.ui.unit.sp +import androidx.glance.GlanceModifier +import androidx.glance.GlanceTheme +import androidx.glance.Image +import androidx.glance.ImageProvider +import androidx.glance.LocalContext +import androidx.glance.LocalSize +import androidx.glance.appwidget.cornerRadius +import androidx.glance.layout.Alignment +import androidx.glance.layout.Row +import androidx.glance.layout.Column +import androidx.glance.layout.ContentScale +import androidx.glance.layout.Spacer +import androidx.glance.layout.size +import androidx.glance.text.FontWeight +import androidx.glance.text.Text +import androidx.glance.text.TextStyle +import oss.krtirtho.spotube.glance.Breakpoints +import oss.krtirtho.spotube.glance.models.Track + +@Composable +fun TrackDetailsView(activeTrack: Track?) { + val context = LocalContext.current + + val size = LocalSize.current + + val artistStr = activeTrack?.artists?.map { it.name }?.joinToString(", ") ?: "" + val imgLocalPath = activeTrack?.album?.images?.get(0)?.path; + val title = activeTrack?.name ?: "" + + + Image( + provider = + if (imgLocalPath == null) + ImageProvider( + BitmapFactory.decodeResource( + context.resources, + android.R.drawable.ic_delete + ) + ) + else ImageProvider(BitmapFactory.decodeFile(imgLocalPath)), + contentDescription = "Album Art", + modifier = GlanceModifier.cornerRadius(8.dp) + .size( + if (size.height < 200.dp) 50.dp + else 100.dp + ), + contentScale = ContentScale.Fit + ) + Spacer(modifier = GlanceModifier.size(6.dp)) + Column { + Text( + text = title, + style = TextStyle( + fontSize = 16.sp, + fontWeight = FontWeight.Bold, + color = GlanceTheme.colors.onBackground + ), + ) + if (size != Breakpoints.SMALL_SQUARE) { + Spacer(modifier = GlanceModifier.size(6.dp)) + Text( + text = artistStr, + style = TextStyle( + fontSize = 14.sp, + color = GlanceTheme.colors.onBackground + ), + ) + } + } +} \ No newline at end of file diff --git a/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/TrackProgress.kt b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/TrackProgress.kt new file mode 100644 index 00000000..b54059b1 --- /dev/null +++ b/android/app/src/main/kotlin/oss/krtirtho/spotube/glance/widgets/TrackProgress.kt @@ -0,0 +1,77 @@ +package oss.krtirtho.spotube.glance.widgets + +import android.content.SharedPreferences +import androidx.compose.runtime.Composable +import androidx.compose.ui.unit.dp +import androidx.glance.GlanceModifier +import androidx.glance.GlanceTheme +import androidx.glance.LocalSize +import androidx.glance.appwidget.LinearProgressIndicator +import androidx.glance.layout.Column +import androidx.glance.layout.Row +import androidx.glance.layout.Spacer +import androidx.glance.layout.fillMaxWidth +import androidx.glance.layout.size +import androidx.glance.text.Text +import androidx.glance.text.TextStyle +import kotlin.math.max +import kotlin.time.Duration +import kotlin.time.Duration.Companion.seconds +import oss.krtirtho.spotube.glance.Breakpoints + +fun Duration.format(): String { + return this.toComponents { hour, minutes, seconds, nanoseconds -> + var paddedSeconds = seconds.toString().padStart(2, '0') + var paddedMinutes = minutes.toString().padStart(2, '0') + var paddedHour = hour.toString().padStart(2, '0') + if (hour == 0L) { + "$paddedMinutes:$paddedSeconds" + } else { + "$paddedHour:$paddedMinutes:$paddedSeconds" + } + } +} + +@Composable +fun TrackProgress(prefs: SharedPreferences) { + val size = LocalSize.current + val position = prefs.getInt("position", 0).seconds + var duration = prefs.getInt("duration", 0).seconds + + var progress = position.inWholeSeconds.toFloat() / max(duration.inWholeSeconds.toFloat(), 1.0f) + + var textStyle = + TextStyle( + color = GlanceTheme.colors.onBackground, + ) + + if (size == Breakpoints.HORIZONTAL_RECTANGLE) { + Row(modifier = GlanceModifier.fillMaxWidth()) { + Text(text = position.format(), style = textStyle) + Spacer(modifier = GlanceModifier.size(6.dp)) + LinearProgressIndicator( + progress = progress, + modifier = GlanceModifier.defaultWeight(), + color = GlanceTheme.colors.primary, + backgroundColor = GlanceTheme.colors.primaryContainer, + ) + Spacer(modifier = GlanceModifier.size(6.dp)) + Text(text = duration.format(), style = textStyle) + } + } else { + Column(modifier = GlanceModifier.fillMaxWidth()) { + LinearProgressIndicator( + progress = progress, + modifier = GlanceModifier.fillMaxWidth(), + color = GlanceTheme.colors.primary, + backgroundColor = GlanceTheme.colors.primaryContainer, + ) + Spacer(modifier = GlanceModifier.size(6.dp)) + Row(modifier = GlanceModifier.fillMaxWidth()) { + Text(text = position.format(), style = textStyle) + Spacer(modifier = GlanceModifier.defaultWeight()) + Text(text = duration.format(), style = textStyle) + } + } + } +} diff --git a/android/app/src/main/res/drawable-hdpi-v31/android12branding.png b/android/app/src/main/res/drawable-hdpi-v31/android12branding.png index fff3f148..22a9b8d3 100644 Binary files a/android/app/src/main/res/drawable-hdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-hdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-hdpi/android12splash.png b/android/app/src/main/res/drawable-hdpi/android12splash.png index 0d4e1c68..adeebcd1 100644 Binary files a/android/app/src/main/res/drawable-hdpi/android12splash.png and b/android/app/src/main/res/drawable-hdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-hdpi/branding.png b/android/app/src/main/res/drawable-hdpi/branding.png index fff3f148..22a9b8d3 100644 Binary files a/android/app/src/main/res/drawable-hdpi/branding.png and b/android/app/src/main/res/drawable-hdpi/branding.png differ diff --git a/android/app/src/main/res/drawable-hdpi/ic_launcher_foreground.png b/android/app/src/main/res/drawable-hdpi/ic_launcher_foreground.png index a65caa1b..204ffe94 100644 Binary files a/android/app/src/main/res/drawable-hdpi/ic_launcher_foreground.png and b/android/app/src/main/res/drawable-hdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/main/res/drawable-hdpi/splash.png b/android/app/src/main/res/drawable-hdpi/splash.png index 0979b5e4..87eebe5c 100644 Binary files a/android/app/src/main/res/drawable-hdpi/splash.png and b/android/app/src/main/res/drawable-hdpi/splash.png differ diff --git a/android/app/src/main/res/drawable-mdpi-v31/android12branding.png b/android/app/src/main/res/drawable-mdpi-v31/android12branding.png index 5a197da3..3ff2a2da 100644 Binary files a/android/app/src/main/res/drawable-mdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-mdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-mdpi/android12splash.png b/android/app/src/main/res/drawable-mdpi/android12splash.png index 5d9760d4..72c22614 100644 Binary files a/android/app/src/main/res/drawable-mdpi/android12splash.png and b/android/app/src/main/res/drawable-mdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-mdpi/branding.png b/android/app/src/main/res/drawable-mdpi/branding.png index 5a197da3..3ff2a2da 100644 Binary files a/android/app/src/main/res/drawable-mdpi/branding.png and b/android/app/src/main/res/drawable-mdpi/branding.png differ diff --git a/android/app/src/main/res/drawable-mdpi/ic_launcher_foreground.png b/android/app/src/main/res/drawable-mdpi/ic_launcher_foreground.png index 3ad0f47e..e7a75c9a 100644 Binary files a/android/app/src/main/res/drawable-mdpi/ic_launcher_foreground.png and b/android/app/src/main/res/drawable-mdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/main/res/drawable-mdpi/splash.png b/android/app/src/main/res/drawable-mdpi/splash.png index c5e4aca0..6e04efdc 100644 Binary files a/android/app/src/main/res/drawable-mdpi/splash.png and b/android/app/src/main/res/drawable-mdpi/splash.png differ diff --git a/android/app/src/main/res/drawable-night-hdpi-v31/android12branding.png b/android/app/src/main/res/drawable-night-hdpi-v31/android12branding.png index fff3f148..22a9b8d3 100644 Binary files a/android/app/src/main/res/drawable-night-hdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-night-hdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-night-hdpi/android12splash.png b/android/app/src/main/res/drawable-night-hdpi/android12splash.png index 0d4e1c68..adeebcd1 100644 Binary files a/android/app/src/main/res/drawable-night-hdpi/android12splash.png and b/android/app/src/main/res/drawable-night-hdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-night-mdpi-v31/android12branding.png b/android/app/src/main/res/drawable-night-mdpi-v31/android12branding.png index 5a197da3..3ff2a2da 100644 Binary files a/android/app/src/main/res/drawable-night-mdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-night-mdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-night-mdpi/android12splash.png b/android/app/src/main/res/drawable-night-mdpi/android12splash.png index 5d9760d4..72c22614 100644 Binary files a/android/app/src/main/res/drawable-night-mdpi/android12splash.png and b/android/app/src/main/res/drawable-night-mdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-night-xhdpi-v31/android12branding.png b/android/app/src/main/res/drawable-night-xhdpi-v31/android12branding.png index efc7e6c8..8e2bb197 100644 Binary files a/android/app/src/main/res/drawable-night-xhdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-night-xhdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-night-xhdpi/android12splash.png b/android/app/src/main/res/drawable-night-xhdpi/android12splash.png index 8e5ceb0a..5adba278 100644 Binary files a/android/app/src/main/res/drawable-night-xhdpi/android12splash.png and b/android/app/src/main/res/drawable-night-xhdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-night-xxhdpi-v31/android12branding.png b/android/app/src/main/res/drawable-night-xxhdpi-v31/android12branding.png index 593877e0..d301093a 100644 Binary files a/android/app/src/main/res/drawable-night-xxhdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-night-xxhdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-night-xxhdpi/android12splash.png b/android/app/src/main/res/drawable-night-xxhdpi/android12splash.png index fb89da37..4e294a2d 100644 Binary files a/android/app/src/main/res/drawable-night-xxhdpi/android12splash.png and b/android/app/src/main/res/drawable-night-xxhdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-night-xxxhdpi-v31/android12branding.png b/android/app/src/main/res/drawable-night-xxxhdpi-v31/android12branding.png index 9d233302..42b0bdc2 100644 Binary files a/android/app/src/main/res/drawable-night-xxxhdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-night-xxxhdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-night-xxxhdpi/android12splash.png b/android/app/src/main/res/drawable-night-xxxhdpi/android12splash.png index 38e4a12c..dcc70377 100644 Binary files a/android/app/src/main/res/drawable-night-xxxhdpi/android12splash.png and b/android/app/src/main/res/drawable-night-xxxhdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-v21/background.png b/android/app/src/main/res/drawable-v21/background.png index 203fc77a..4bebb9de 100644 Binary files a/android/app/src/main/res/drawable-v21/background.png and b/android/app/src/main/res/drawable-v21/background.png differ diff --git a/android/app/src/main/res/drawable-v21/launch_background.xml b/android/app/src/main/res/drawable-v21/launch_background.xml index 52e8749e..5367a886 100644 --- a/android/app/src/main/res/drawable-v21/launch_background.xml +++ b/android/app/src/main/res/drawable-v21/launch_background.xml @@ -6,7 +6,7 @@ - + diff --git a/android/app/src/main/res/drawable-xhdpi-v31/android12branding.png b/android/app/src/main/res/drawable-xhdpi-v31/android12branding.png index efc7e6c8..8e2bb197 100644 Binary files a/android/app/src/main/res/drawable-xhdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-xhdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-xhdpi/android12splash.png b/android/app/src/main/res/drawable-xhdpi/android12splash.png index 8e5ceb0a..5adba278 100644 Binary files a/android/app/src/main/res/drawable-xhdpi/android12splash.png and b/android/app/src/main/res/drawable-xhdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-xhdpi/branding.png b/android/app/src/main/res/drawable-xhdpi/branding.png index efc7e6c8..8e2bb197 100644 Binary files a/android/app/src/main/res/drawable-xhdpi/branding.png and b/android/app/src/main/res/drawable-xhdpi/branding.png differ diff --git a/android/app/src/main/res/drawable-xhdpi/ic_launcher_foreground.png b/android/app/src/main/res/drawable-xhdpi/ic_launcher_foreground.png index 8774c0ff..b608dca7 100644 Binary files a/android/app/src/main/res/drawable-xhdpi/ic_launcher_foreground.png and b/android/app/src/main/res/drawable-xhdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/main/res/drawable-xhdpi/splash.png b/android/app/src/main/res/drawable-xhdpi/splash.png index 41222c6c..51a669aa 100644 Binary files a/android/app/src/main/res/drawable-xhdpi/splash.png and b/android/app/src/main/res/drawable-xhdpi/splash.png differ diff --git a/android/app/src/main/res/drawable-xxhdpi-v31/android12branding.png b/android/app/src/main/res/drawable-xxhdpi-v31/android12branding.png index 593877e0..d301093a 100644 Binary files a/android/app/src/main/res/drawable-xxhdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-xxhdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-xxhdpi/android12splash.png b/android/app/src/main/res/drawable-xxhdpi/android12splash.png index fb89da37..4e294a2d 100644 Binary files a/android/app/src/main/res/drawable-xxhdpi/android12splash.png and b/android/app/src/main/res/drawable-xxhdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-xxhdpi/branding.png b/android/app/src/main/res/drawable-xxhdpi/branding.png index 593877e0..d301093a 100644 Binary files a/android/app/src/main/res/drawable-xxhdpi/branding.png and b/android/app/src/main/res/drawable-xxhdpi/branding.png differ diff --git a/android/app/src/main/res/drawable-xxhdpi/ic_launcher_foreground.png b/android/app/src/main/res/drawable-xxhdpi/ic_launcher_foreground.png index a379c006..b1e93458 100644 Binary files a/android/app/src/main/res/drawable-xxhdpi/ic_launcher_foreground.png and b/android/app/src/main/res/drawable-xxhdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/main/res/drawable-xxhdpi/splash.png b/android/app/src/main/res/drawable-xxhdpi/splash.png index 5ab19d32..cc79cb85 100644 Binary files a/android/app/src/main/res/drawable-xxhdpi/splash.png and b/android/app/src/main/res/drawable-xxhdpi/splash.png differ diff --git a/android/app/src/main/res/drawable-xxxhdpi-v31/android12branding.png b/android/app/src/main/res/drawable-xxxhdpi-v31/android12branding.png index 9d233302..42b0bdc2 100644 Binary files a/android/app/src/main/res/drawable-xxxhdpi-v31/android12branding.png and b/android/app/src/main/res/drawable-xxxhdpi-v31/android12branding.png differ diff --git a/android/app/src/main/res/drawable-xxxhdpi/android12splash.png b/android/app/src/main/res/drawable-xxxhdpi/android12splash.png index 38e4a12c..dcc70377 100644 Binary files a/android/app/src/main/res/drawable-xxxhdpi/android12splash.png and b/android/app/src/main/res/drawable-xxxhdpi/android12splash.png differ diff --git a/android/app/src/main/res/drawable-xxxhdpi/branding.png b/android/app/src/main/res/drawable-xxxhdpi/branding.png index 9d233302..42b0bdc2 100644 Binary files a/android/app/src/main/res/drawable-xxxhdpi/branding.png and b/android/app/src/main/res/drawable-xxxhdpi/branding.png differ diff --git a/android/app/src/main/res/drawable-xxxhdpi/ic_launcher_foreground.png b/android/app/src/main/res/drawable-xxxhdpi/ic_launcher_foreground.png index 44621df8..d178ba12 100644 Binary files a/android/app/src/main/res/drawable-xxxhdpi/ic_launcher_foreground.png and b/android/app/src/main/res/drawable-xxxhdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/main/res/drawable-xxxhdpi/splash.png b/android/app/src/main/res/drawable-xxxhdpi/splash.png index a6d40b89..f526b26d 100644 Binary files a/android/app/src/main/res/drawable-xxxhdpi/splash.png and b/android/app/src/main/res/drawable-xxxhdpi/splash.png differ diff --git a/android/app/src/main/res/drawable/background.png b/android/app/src/main/res/drawable/background.png index 203fc77a..4bebb9de 100644 Binary files a/android/app/src/main/res/drawable/background.png and b/android/app/src/main/res/drawable/background.png differ diff --git a/android/app/src/main/res/drawable/launch_background.xml b/android/app/src/main/res/drawable/launch_background.xml index 52e8749e..5367a886 100644 --- a/android/app/src/main/res/drawable/launch_background.xml +++ b/android/app/src/main/res/drawable/launch_background.xml @@ -6,7 +6,7 @@ - + diff --git a/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml b/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml index f606c4d8..c79c58a3 100644 --- a/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml +++ b/android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml @@ -1,6 +1,9 @@ - - + + + diff --git a/android/app/src/main/res/mipmap-hdpi/ic_launcher.png b/android/app/src/main/res/mipmap-hdpi/ic_launcher.png index d26a3a6e..c9e5cfad 100644 Binary files a/android/app/src/main/res/mipmap-hdpi/ic_launcher.png and b/android/app/src/main/res/mipmap-hdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-mdpi/ic_launcher.png b/android/app/src/main/res/mipmap-mdpi/ic_launcher.png index 63d16c60..159b302f 100644 Binary files a/android/app/src/main/res/mipmap-mdpi/ic_launcher.png and b/android/app/src/main/res/mipmap-mdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png b/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png index fe2957ad..9c81fb60 100644 Binary files a/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png and b/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png b/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png index 85c1d793..3450fb00 100644 Binary files a/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png and b/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png index f22fa5cf..f94fd407 100644 Binary files a/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png and b/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/android/app/src/main/res/xml/home_player_widget_config.xml b/android/app/src/main/res/xml/home_player_widget_config.xml new file mode 100644 index 00000000..c8ec7048 --- /dev/null +++ b/android/app/src/main/res/xml/home_player_widget_config.xml @@ -0,0 +1,7 @@ + + diff --git a/android/app/src/nightly/res/drawable-hdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-hdpi-v31/android12branding.png index 9e62c813..22a9b8d3 100644 Binary files a/android/app/src/nightly/res/drawable-hdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-hdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-hdpi/android12splash.png b/android/app/src/nightly/res/drawable-hdpi/android12splash.png index 98fd87f4..1d1cb853 100644 Binary files a/android/app/src/nightly/res/drawable-hdpi/android12splash.png and b/android/app/src/nightly/res/drawable-hdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-hdpi/branding.png b/android/app/src/nightly/res/drawable-hdpi/branding.png index 9e62c813..22a9b8d3 100644 Binary files a/android/app/src/nightly/res/drawable-hdpi/branding.png and b/android/app/src/nightly/res/drawable-hdpi/branding.png differ diff --git a/android/app/src/nightly/res/drawable-hdpi/ic_launcher_foreground.png b/android/app/src/nightly/res/drawable-hdpi/ic_launcher_foreground.png index 20a57ba2..7373eec1 100644 Binary files a/android/app/src/nightly/res/drawable-hdpi/ic_launcher_foreground.png and b/android/app/src/nightly/res/drawable-hdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/nightly/res/drawable-hdpi/splash.png b/android/app/src/nightly/res/drawable-hdpi/splash.png index 07c7024a..e3869211 100644 Binary files a/android/app/src/nightly/res/drawable-hdpi/splash.png and b/android/app/src/nightly/res/drawable-hdpi/splash.png differ diff --git a/android/app/src/nightly/res/drawable-mdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-mdpi-v31/android12branding.png index 80579983..3ff2a2da 100644 Binary files a/android/app/src/nightly/res/drawable-mdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-mdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-mdpi/android12splash.png b/android/app/src/nightly/res/drawable-mdpi/android12splash.png index a86a2222..3f2a1a0a 100644 Binary files a/android/app/src/nightly/res/drawable-mdpi/android12splash.png and b/android/app/src/nightly/res/drawable-mdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-mdpi/branding.png b/android/app/src/nightly/res/drawable-mdpi/branding.png index 80579983..3ff2a2da 100644 Binary files a/android/app/src/nightly/res/drawable-mdpi/branding.png and b/android/app/src/nightly/res/drawable-mdpi/branding.png differ diff --git a/android/app/src/nightly/res/drawable-mdpi/ic_launcher_foreground.png b/android/app/src/nightly/res/drawable-mdpi/ic_launcher_foreground.png index 43f5e0fe..8d5a9656 100644 Binary files a/android/app/src/nightly/res/drawable-mdpi/ic_launcher_foreground.png and b/android/app/src/nightly/res/drawable-mdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/nightly/res/drawable-mdpi/splash.png b/android/app/src/nightly/res/drawable-mdpi/splash.png index 86d3fe74..d8f7cc0e 100644 Binary files a/android/app/src/nightly/res/drawable-mdpi/splash.png and b/android/app/src/nightly/res/drawable-mdpi/splash.png differ diff --git a/android/app/src/nightly/res/drawable-night-hdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-night-hdpi-v31/android12branding.png index 9e62c813..22a9b8d3 100644 Binary files a/android/app/src/nightly/res/drawable-night-hdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-night-hdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-night-hdpi/android12splash.png b/android/app/src/nightly/res/drawable-night-hdpi/android12splash.png index 98fd87f4..1d1cb853 100644 Binary files a/android/app/src/nightly/res/drawable-night-hdpi/android12splash.png and b/android/app/src/nightly/res/drawable-night-hdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-night-mdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-night-mdpi-v31/android12branding.png index 80579983..3ff2a2da 100644 Binary files a/android/app/src/nightly/res/drawable-night-mdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-night-mdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-night-mdpi/android12splash.png b/android/app/src/nightly/res/drawable-night-mdpi/android12splash.png index a86a2222..3f2a1a0a 100644 Binary files a/android/app/src/nightly/res/drawable-night-mdpi/android12splash.png and b/android/app/src/nightly/res/drawable-night-mdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-night-xhdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-night-xhdpi-v31/android12branding.png index 0bcf138d..8e2bb197 100644 Binary files a/android/app/src/nightly/res/drawable-night-xhdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-night-xhdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-night-xhdpi/android12splash.png b/android/app/src/nightly/res/drawable-night-xhdpi/android12splash.png index ad3f39d0..ba73b0e2 100644 Binary files a/android/app/src/nightly/res/drawable-night-xhdpi/android12splash.png and b/android/app/src/nightly/res/drawable-night-xhdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-night-xxhdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-night-xxhdpi-v31/android12branding.png index c7d01776..d301093a 100644 Binary files a/android/app/src/nightly/res/drawable-night-xxhdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-night-xxhdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-night-xxhdpi/android12splash.png b/android/app/src/nightly/res/drawable-night-xxhdpi/android12splash.png index 133fb647..5e5cdc45 100644 Binary files a/android/app/src/nightly/res/drawable-night-xxhdpi/android12splash.png and b/android/app/src/nightly/res/drawable-night-xxhdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-night-xxxhdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-night-xxxhdpi-v31/android12branding.png index 5477b799..42b0bdc2 100644 Binary files a/android/app/src/nightly/res/drawable-night-xxxhdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-night-xxxhdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-night-xxxhdpi/android12splash.png b/android/app/src/nightly/res/drawable-night-xxxhdpi/android12splash.png index fa5a8c92..adff9bec 100644 Binary files a/android/app/src/nightly/res/drawable-night-xxxhdpi/android12splash.png and b/android/app/src/nightly/res/drawable-night-xxxhdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-v21/background.png b/android/app/src/nightly/res/drawable-v21/background.png index 203fc77a..4bebb9de 100644 Binary files a/android/app/src/nightly/res/drawable-v21/background.png and b/android/app/src/nightly/res/drawable-v21/background.png differ diff --git a/android/app/src/nightly/res/drawable-v21/launch_background.xml b/android/app/src/nightly/res/drawable-v21/launch_background.xml index 52e8749e..5367a886 100644 --- a/android/app/src/nightly/res/drawable-v21/launch_background.xml +++ b/android/app/src/nightly/res/drawable-v21/launch_background.xml @@ -6,7 +6,7 @@ - + diff --git a/android/app/src/nightly/res/drawable-xhdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-xhdpi-v31/android12branding.png index 0bcf138d..8e2bb197 100644 Binary files a/android/app/src/nightly/res/drawable-xhdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-xhdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-xhdpi/android12splash.png b/android/app/src/nightly/res/drawable-xhdpi/android12splash.png index ad3f39d0..ba73b0e2 100644 Binary files a/android/app/src/nightly/res/drawable-xhdpi/android12splash.png and b/android/app/src/nightly/res/drawable-xhdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-xhdpi/branding.png b/android/app/src/nightly/res/drawable-xhdpi/branding.png index 0bcf138d..8e2bb197 100644 Binary files a/android/app/src/nightly/res/drawable-xhdpi/branding.png and b/android/app/src/nightly/res/drawable-xhdpi/branding.png differ diff --git a/android/app/src/nightly/res/drawable-xhdpi/ic_launcher_foreground.png b/android/app/src/nightly/res/drawable-xhdpi/ic_launcher_foreground.png index 4cf86d25..f4f416f7 100644 Binary files a/android/app/src/nightly/res/drawable-xhdpi/ic_launcher_foreground.png and b/android/app/src/nightly/res/drawable-xhdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/nightly/res/drawable-xhdpi/splash.png b/android/app/src/nightly/res/drawable-xhdpi/splash.png index dbb0ea02..17a2c373 100644 Binary files a/android/app/src/nightly/res/drawable-xhdpi/splash.png and b/android/app/src/nightly/res/drawable-xhdpi/splash.png differ diff --git a/android/app/src/nightly/res/drawable-xxhdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-xxhdpi-v31/android12branding.png index c7d01776..d301093a 100644 Binary files a/android/app/src/nightly/res/drawable-xxhdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-xxhdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-xxhdpi/android12splash.png b/android/app/src/nightly/res/drawable-xxhdpi/android12splash.png index 133fb647..5e5cdc45 100644 Binary files a/android/app/src/nightly/res/drawable-xxhdpi/android12splash.png and b/android/app/src/nightly/res/drawable-xxhdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-xxhdpi/branding.png b/android/app/src/nightly/res/drawable-xxhdpi/branding.png index c7d01776..d301093a 100644 Binary files a/android/app/src/nightly/res/drawable-xxhdpi/branding.png and b/android/app/src/nightly/res/drawable-xxhdpi/branding.png differ diff --git a/android/app/src/nightly/res/drawable-xxhdpi/ic_launcher_foreground.png b/android/app/src/nightly/res/drawable-xxhdpi/ic_launcher_foreground.png index 95fa3443..9f88e976 100644 Binary files a/android/app/src/nightly/res/drawable-xxhdpi/ic_launcher_foreground.png and b/android/app/src/nightly/res/drawable-xxhdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/nightly/res/drawable-xxhdpi/splash.png b/android/app/src/nightly/res/drawable-xxhdpi/splash.png index 12eb5531..db53f016 100644 Binary files a/android/app/src/nightly/res/drawable-xxhdpi/splash.png and b/android/app/src/nightly/res/drawable-xxhdpi/splash.png differ diff --git a/android/app/src/nightly/res/drawable-xxxhdpi-v31/android12branding.png b/android/app/src/nightly/res/drawable-xxxhdpi-v31/android12branding.png index 5477b799..42b0bdc2 100644 Binary files a/android/app/src/nightly/res/drawable-xxxhdpi-v31/android12branding.png and b/android/app/src/nightly/res/drawable-xxxhdpi-v31/android12branding.png differ diff --git a/android/app/src/nightly/res/drawable-xxxhdpi/android12splash.png b/android/app/src/nightly/res/drawable-xxxhdpi/android12splash.png index fa5a8c92..adff9bec 100644 Binary files a/android/app/src/nightly/res/drawable-xxxhdpi/android12splash.png and b/android/app/src/nightly/res/drawable-xxxhdpi/android12splash.png differ diff --git a/android/app/src/nightly/res/drawable-xxxhdpi/branding.png b/android/app/src/nightly/res/drawable-xxxhdpi/branding.png index 5477b799..42b0bdc2 100644 Binary files a/android/app/src/nightly/res/drawable-xxxhdpi/branding.png and b/android/app/src/nightly/res/drawable-xxxhdpi/branding.png differ diff --git a/android/app/src/nightly/res/drawable-xxxhdpi/ic_launcher_foreground.png b/android/app/src/nightly/res/drawable-xxxhdpi/ic_launcher_foreground.png index 3de8a2ee..7e2bb4c3 100644 Binary files a/android/app/src/nightly/res/drawable-xxxhdpi/ic_launcher_foreground.png and b/android/app/src/nightly/res/drawable-xxxhdpi/ic_launcher_foreground.png differ diff --git a/android/app/src/nightly/res/drawable-xxxhdpi/splash.png b/android/app/src/nightly/res/drawable-xxxhdpi/splash.png index 68e806f4..a74c95a4 100644 Binary files a/android/app/src/nightly/res/drawable-xxxhdpi/splash.png and b/android/app/src/nightly/res/drawable-xxxhdpi/splash.png differ diff --git a/android/app/src/nightly/res/drawable/background.png b/android/app/src/nightly/res/drawable/background.png index 203fc77a..4bebb9de 100644 Binary files a/android/app/src/nightly/res/drawable/background.png and b/android/app/src/nightly/res/drawable/background.png differ diff --git a/android/app/src/nightly/res/drawable/ic_launcher_monochrome.xml b/android/app/src/nightly/res/drawable/ic_launcher_monochrome.xml new file mode 100644 index 00000000..8aae0e6c --- /dev/null +++ b/android/app/src/nightly/res/drawable/ic_launcher_monochrome.xml @@ -0,0 +1,27 @@ + + + + + diff --git a/android/app/src/nightly/res/drawable/launch_background.xml b/android/app/src/nightly/res/drawable/launch_background.xml index 52e8749e..5367a886 100644 --- a/android/app/src/nightly/res/drawable/launch_background.xml +++ b/android/app/src/nightly/res/drawable/launch_background.xml @@ -6,7 +6,7 @@ - + diff --git a/android/app/src/nightly/res/mipmap-anydpi-v26/ic_launcher.xml b/android/app/src/nightly/res/mipmap-anydpi-v26/ic_launcher.xml index 5f349f7f..c79c58a3 100644 --- a/android/app/src/nightly/res/mipmap-anydpi-v26/ic_launcher.xml +++ b/android/app/src/nightly/res/mipmap-anydpi-v26/ic_launcher.xml @@ -1,5 +1,9 @@ - + + + diff --git a/android/app/src/nightly/res/mipmap-hdpi/ic_launcher.png b/android/app/src/nightly/res/mipmap-hdpi/ic_launcher.png index a826bb73..8164cc67 100644 Binary files a/android/app/src/nightly/res/mipmap-hdpi/ic_launcher.png and b/android/app/src/nightly/res/mipmap-hdpi/ic_launcher.png differ diff --git a/android/app/src/nightly/res/mipmap-mdpi/ic_launcher.png b/android/app/src/nightly/res/mipmap-mdpi/ic_launcher.png index 3743861d..bff95e97 100644 Binary files a/android/app/src/nightly/res/mipmap-mdpi/ic_launcher.png and b/android/app/src/nightly/res/mipmap-mdpi/ic_launcher.png differ diff --git a/android/app/src/nightly/res/mipmap-xhdpi/ic_launcher.png b/android/app/src/nightly/res/mipmap-xhdpi/ic_launcher.png index 1be1daa7..df515ed1 100644 Binary files a/android/app/src/nightly/res/mipmap-xhdpi/ic_launcher.png and b/android/app/src/nightly/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/android/app/src/nightly/res/mipmap-xxhdpi/ic_launcher.png b/android/app/src/nightly/res/mipmap-xxhdpi/ic_launcher.png index 3a8a7832..e58ef25f 100644 Binary files a/android/app/src/nightly/res/mipmap-xxhdpi/ic_launcher.png and b/android/app/src/nightly/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/android/app/src/nightly/res/mipmap-xxxhdpi/ic_launcher.png b/android/app/src/nightly/res/mipmap-xxxhdpi/ic_launcher.png index 781c9c1a..d39a5e64 100644 Binary files a/android/app/src/nightly/res/mipmap-xxxhdpi/ic_launcher.png and b/android/app/src/nightly/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/android/app/src/profile/AndroidManifest.xml b/android/app/src/profile/AndroidManifest.xml index 1041f6ca..a32d12af 100644 --- a/android/app/src/profile/AndroidManifest.xml +++ b/android/app/src/profile/AndroidManifest.xml @@ -1,7 +1,6 @@ - - - - + + \ No newline at end of file diff --git a/android/build.gradle b/android/build.gradle index bc157bd1..8f31e8ca 100644 --- a/android/build.gradle +++ b/android/build.gradle @@ -15,4 +15,4 @@ subprojects { tasks.register("clean", Delete) { delete rootProject.buildDir -} +} \ No newline at end of file diff --git a/android/gradle/wrapper/gradle-wrapper.properties b/android/gradle/wrapper/gradle-wrapper.properties index 02e5f581..bf6b7385 100644 --- a/android/gradle/wrapper/gradle-wrapper.properties +++ b/android/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,6 @@ -#Fri Jun 23 08:50:38 CEST 2017 +#Fri Dec 13 21:53:13 BDT 2024 distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-7.6-all.zip diff --git a/android/settings.gradle b/android/settings.gradle index 89651748..53d34a77 100644 --- a/android/settings.gradle +++ b/android/settings.gradle @@ -18,8 +18,9 @@ pluginManagement { plugins { id "dev.flutter.flutter-plugin-loader" version "1.0.0" - id "com.android.application" version "7.2.1" apply false - id "org.jetbrains.kotlin.android" version "1.8.22" apply false + id "com.android.application" version '8.7.0' apply false + id "org.jetbrains.kotlin.android" version "2.1.0" apply false + id "org.jetbrains.kotlin.plugin.compose" version "2.1.0" apply false } -include ":app" \ No newline at end of file +include ':app' \ No newline at end of file diff --git a/appdmg.json b/appdmg.json index eb9b5236..6e365f23 100644 --- a/appdmg.json +++ b/appdmg.json @@ -1,6 +1,6 @@ { "title": "Spotube", - "icon": "assets/spotube-logo-macos.png", + "icon": "assets/branding/spotube-logo-macos.png", "contents": [ { "x": 448, @@ -15,4 +15,4 @@ "path": "build/macos/Build/Products/Release/Spotube.app" } ] -} \ No newline at end of file +} diff --git a/assets/bengali-patterns-bg.jpg b/assets/bengali-patterns-bg.jpg deleted file mode 100644 index 513557a3..00000000 Binary files a/assets/bengali-patterns-bg.jpg and /dev/null differ diff --git a/assets/branding.png b/assets/branding/branding.png similarity index 100% rename from assets/branding.png rename to assets/branding/branding.png diff --git a/assets/branding/mobile-screenshots/android-1.jpg b/assets/branding/mobile-screenshots/android-1.jpg new file mode 100644 index 00000000..16debbd7 Binary files /dev/null and b/assets/branding/mobile-screenshots/android-1.jpg differ diff --git a/assets/branding/mobile-screenshots/android-2.jpg b/assets/branding/mobile-screenshots/android-2.jpg new file mode 100644 index 00000000..d594227e Binary files /dev/null and b/assets/branding/mobile-screenshots/android-2.jpg differ diff --git a/assets/branding/mobile-screenshots/android-3.jpg b/assets/branding/mobile-screenshots/android-3.jpg new file mode 100644 index 00000000..ed2d61f2 Binary files /dev/null and b/assets/branding/mobile-screenshots/android-3.jpg differ diff --git a/assets/branding/mobile-screenshots/android-4.jpg b/assets/branding/mobile-screenshots/android-4.jpg new file mode 100644 index 00000000..36a3549f Binary files /dev/null and b/assets/branding/mobile-screenshots/android-4.jpg differ diff --git a/assets/branding/mobile-screenshots/android-5.jpg b/assets/branding/mobile-screenshots/android-5.jpg new file mode 100644 index 00000000..29be481b Binary files /dev/null and b/assets/branding/mobile-screenshots/android-5.jpg differ diff --git a/assets/branding/mobile-screenshots/combined.jpg b/assets/branding/mobile-screenshots/combined.jpg new file mode 100644 index 00000000..03740580 Binary files /dev/null and b/assets/branding/mobile-screenshots/combined.jpg differ diff --git a/assets/spotube-hero-banner.png b/assets/branding/spotube-hero-banner.png similarity index 100% rename from assets/spotube-hero-banner.png rename to assets/branding/spotube-hero-banner.png diff --git a/assets/branding/spotube-logo-foreground.png b/assets/branding/spotube-logo-foreground.png new file mode 100644 index 00000000..997d3209 Binary files /dev/null and b/assets/branding/spotube-logo-foreground.png differ diff --git a/assets/branding/spotube-logo-item.png b/assets/branding/spotube-logo-item.png new file mode 100644 index 00000000..476dda29 Binary files /dev/null and b/assets/branding/spotube-logo-item.png differ diff --git a/assets/branding/spotube-logo-light.png b/assets/branding/spotube-logo-light.png new file mode 100644 index 00000000..8c86d621 Binary files /dev/null and b/assets/branding/spotube-logo-light.png differ diff --git a/assets/branding/spotube-logo-macos.png b/assets/branding/spotube-logo-macos.png new file mode 100644 index 00000000..2694b530 Binary files /dev/null and b/assets/branding/spotube-logo-macos.png differ diff --git a/assets/branding/spotube-logo.bmp b/assets/branding/spotube-logo.bmp new file mode 100644 index 00000000..f95df092 Binary files /dev/null and b/assets/branding/spotube-logo.bmp differ diff --git a/assets/branding/spotube-logo.ico b/assets/branding/spotube-logo.ico new file mode 100644 index 00000000..562110b9 Binary files /dev/null and b/assets/branding/spotube-logo.ico differ diff --git a/assets/branding/spotube-logo.png b/assets/branding/spotube-logo.png new file mode 100644 index 00000000..d46d9ce5 Binary files /dev/null and b/assets/branding/spotube-logo.png differ diff --git a/assets/branding/spotube-logo_android12.png b/assets/branding/spotube-logo_android12.png new file mode 100644 index 00000000..4af89de4 Binary files /dev/null and b/assets/branding/spotube-logo_android12.png differ diff --git a/assets/branding/spotube-nightly-item.png b/assets/branding/spotube-nightly-item.png new file mode 100644 index 00000000..61ef1c5d Binary files /dev/null and b/assets/branding/spotube-nightly-item.png differ diff --git a/assets/branding/spotube-nightly-logo-foreground.png b/assets/branding/spotube-nightly-logo-foreground.png new file mode 100644 index 00000000..def278cf Binary files /dev/null and b/assets/branding/spotube-nightly-logo-foreground.png differ diff --git a/assets/branding/spotube-nightly-logo-foreground.svg b/assets/branding/spotube-nightly-logo-foreground.svg new file mode 100644 index 00000000..de2bf370 --- /dev/null +++ b/assets/branding/spotube-nightly-logo-foreground.svg @@ -0,0 +1,190 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/assets/branding/spotube-nightly-logo.png b/assets/branding/spotube-nightly-logo.png new file mode 100644 index 00000000..269aaabc Binary files /dev/null and b/assets/branding/spotube-nightly-logo.png differ diff --git a/assets/branding/spotube-nightly-logo_android12.png b/assets/branding/spotube-nightly-logo_android12.png new file mode 100644 index 00000000..e5683399 Binary files /dev/null and b/assets/branding/spotube-nightly-logo_android12.png differ diff --git a/assets/branding/spotube-screenshot.png b/assets/branding/spotube-screenshot.png new file mode 100644 index 00000000..c708e240 Binary files /dev/null and b/assets/branding/spotube-screenshot.png differ diff --git a/assets/branding/spotube-tall-capsule.png b/assets/branding/spotube-tall-capsule.png new file mode 100644 index 00000000..53bff791 Binary files /dev/null and b/assets/branding/spotube-tall-capsule.png differ diff --git a/assets/branding/spotube-wide-capsule-large.png b/assets/branding/spotube-wide-capsule-large.png new file mode 100644 index 00000000..41614c80 Binary files /dev/null and b/assets/branding/spotube-wide-capsule-large.png differ diff --git a/assets/branding/spotube-wide-capsule-small.png b/assets/branding/spotube-wide-capsule-small.png new file mode 100644 index 00000000..e0717dc5 Binary files /dev/null and b/assets/branding/spotube-wide-capsule-small.png differ diff --git a/assets/branding/spotube_banner.png b/assets/branding/spotube_banner.png new file mode 100644 index 00000000..c9450265 Binary files /dev/null and b/assets/branding/spotube_banner.png differ diff --git a/assets/empty_box.png b/assets/empty_box.png deleted file mode 100644 index 24e95b23..00000000 Binary files a/assets/empty_box.png and /dev/null differ diff --git a/assets/fonts/Cookie-Regular.ttf b/assets/fonts/Cookie-Regular.ttf new file mode 100644 index 00000000..e3ac74eb Binary files /dev/null and b/assets/fonts/Cookie-Regular.ttf differ diff --git a/assets/fonts/Ubuntu_Mono/UFL.txt b/assets/fonts/Ubuntu_Mono/UFL.txt new file mode 100644 index 00000000..6e722c88 --- /dev/null +++ b/assets/fonts/Ubuntu_Mono/UFL.txt @@ -0,0 +1,96 @@ +------------------------------- +UBUNTU FONT LICENCE Version 1.0 +------------------------------- + +PREAMBLE +This licence allows the licensed fonts to be used, studied, modified and +redistributed freely. The fonts, including any derivative works, can be +bundled, embedded, and redistributed provided the terms of this licence +are met. The fonts and derivatives, however, cannot be released under +any other licence. The requirement for fonts to remain under this +licence does not require any document created using the fonts or their +derivatives to be published under this licence, as long as the primary +purpose of the document is not to be a vehicle for the distribution of +the fonts. + +DEFINITIONS +"Font Software" refers to the set of files released by the Copyright +Holder(s) under this licence and clearly marked as such. This may +include source files, build scripts and documentation. + +"Original Version" refers to the collection of Font Software components +as received under this licence. + +"Modified Version" refers to any derivative made by adding to, deleting, +or substituting -- in part or in whole -- any of the components of the +Original Version, by changing formats or by porting the Font Software to +a new environment. + +"Copyright Holder(s)" refers to all individuals and companies who have a +copyright ownership of the Font Software. + +"Substantially Changed" refers to Modified Versions which can be easily +identified as dissimilar to the Font Software by users of the Font +Software comparing the Original Version with the Modified Version. + +To "Propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification and with or without charging +a redistribution fee), making available to the public, and in some +countries other activities as well. + +PERMISSION & CONDITIONS +This licence does not grant any rights under trademark law and all such +rights are reserved. + +Permission is hereby granted, free of charge, to any person obtaining a +copy of the Font Software, to propagate the Font Software, subject to +the below conditions: + +1) Each copy of the Font Software must contain the above copyright +notice and this licence. These can be included either as stand-alone +text files, human-readable headers or in the appropriate machine- +readable metadata fields within text or binary files as long as those +fields can be easily viewed by the user. + +2) The font name complies with the following: +(a) The Original Version must retain its name, unmodified. +(b) Modified Versions which are Substantially Changed must be renamed to +avoid use of the name of the Original Version or similar names entirely. +(c) Modified Versions which are not Substantially Changed must be +renamed to both (i) retain the name of the Original Version and (ii) add +additional naming elements to distinguish the Modified Version from the +Original Version. The name of such Modified Versions must be the name of +the Original Version, with "derivative X" where X represents the name of +the new work, appended to that name. + +3) The name(s) of the Copyright Holder(s) and any contributor to the +Font Software shall not be used to promote, endorse or advertise any +Modified Version, except (i) as required by this licence, (ii) to +acknowledge the contribution(s) of the Copyright Holder(s) or (iii) with +their explicit written permission. + +4) The Font Software, modified or unmodified, in part or in whole, must +be distributed entirely under this licence, and must not be distributed +under any other licence. The requirement for fonts to remain under this +licence does not affect any document created using the Font Software, +except any version of the Font Software extracted from a document +created using the Font Software may only be distributed under this +licence. + +TERMINATION +This licence becomes null and void if any of the above conditions are +not met. + +DISCLAIMER +THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF +COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE +COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL +DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM OTHER +DEALINGS IN THE FONT SOFTWARE. diff --git a/assets/fonts/Ubuntu_Mono/UbuntuMono-Bold.ttf b/assets/fonts/Ubuntu_Mono/UbuntuMono-Bold.ttf new file mode 100644 index 00000000..01ad81bf Binary files /dev/null and b/assets/fonts/Ubuntu_Mono/UbuntuMono-Bold.ttf differ diff --git a/assets/fonts/Ubuntu_Mono/UbuntuMono-BoldItalic.ttf b/assets/fonts/Ubuntu_Mono/UbuntuMono-BoldItalic.ttf new file mode 100644 index 00000000..731884eb Binary files /dev/null and b/assets/fonts/Ubuntu_Mono/UbuntuMono-BoldItalic.ttf differ diff --git a/assets/fonts/Ubuntu_Mono/UbuntuMono-Italic.ttf b/assets/fonts/Ubuntu_Mono/UbuntuMono-Italic.ttf new file mode 100644 index 00000000..b89338d4 Binary files /dev/null and b/assets/fonts/Ubuntu_Mono/UbuntuMono-Italic.ttf differ diff --git a/assets/fonts/Ubuntu_Mono/UbuntuMono-Regular.ttf b/assets/fonts/Ubuntu_Mono/UbuntuMono-Regular.ttf new file mode 100644 index 00000000..4977028d Binary files /dev/null and b/assets/fonts/Ubuntu_Mono/UbuntuMono-Regular.ttf differ diff --git a/assets/album-placeholder.png b/assets/images/album-placeholder.png similarity index 100% rename from assets/album-placeholder.png rename to assets/images/album-placeholder.png diff --git a/assets/images/bengali-patterns-bg.jpg b/assets/images/bengali-patterns-bg.jpg new file mode 100644 index 00000000..a4090a01 Binary files /dev/null and b/assets/images/bengali-patterns-bg.jpg differ diff --git a/assets/images/liked-tracks.jpg b/assets/images/liked-tracks.jpg new file mode 100644 index 00000000..71e010dc Binary files /dev/null and b/assets/images/liked-tracks.jpg differ diff --git a/assets/images/logos/dab-music.png b/assets/images/logos/dab-music.png new file mode 100644 index 00000000..e09d3410 Binary files /dev/null and b/assets/images/logos/dab-music.png differ diff --git a/assets/images/logos/invidious.jpg b/assets/images/logos/invidious.jpg new file mode 100644 index 00000000..3a54ace1 Binary files /dev/null and b/assets/images/logos/invidious.jpg differ diff --git a/assets/jiosaavn.png b/assets/images/logos/jiosaavn.png similarity index 100% rename from assets/jiosaavn.png rename to assets/images/logos/jiosaavn.png diff --git a/assets/placeholder.png b/assets/images/placeholder.png similarity index 100% rename from assets/placeholder.png rename to assets/images/placeholder.png diff --git a/assets/user-placeholder.png b/assets/images/user-placeholder.png similarity index 100% rename from assets/user-placeholder.png rename to assets/images/user-placeholder.png diff --git a/assets/invidious.jpg b/assets/invidious.jpg deleted file mode 100644 index 12c5f107..00000000 Binary files a/assets/invidious.jpg and /dev/null differ diff --git a/assets/liked-tracks.jpg b/assets/liked-tracks.jpg deleted file mode 100644 index 62dad65e..00000000 Binary files a/assets/liked-tracks.jpg and /dev/null differ diff --git a/assets/logos/songlink-transparent.png b/assets/logos/songlink-transparent.png deleted file mode 100644 index 6b7064c9..00000000 Binary files a/assets/logos/songlink-transparent.png and /dev/null differ diff --git a/assets/logos/songlink.png b/assets/logos/songlink.png deleted file mode 100644 index 43d823a5..00000000 Binary files a/assets/logos/songlink.png and /dev/null differ diff --git a/assets/mobile-screenshots/android-1.jpg b/assets/mobile-screenshots/android-1.jpg deleted file mode 100644 index ae1ef8ac..00000000 Binary files a/assets/mobile-screenshots/android-1.jpg and /dev/null differ diff --git a/assets/mobile-screenshots/android-2.jpg b/assets/mobile-screenshots/android-2.jpg deleted file mode 100644 index b6668d2b..00000000 Binary files a/assets/mobile-screenshots/android-2.jpg and /dev/null differ diff --git a/assets/mobile-screenshots/android-3.jpg b/assets/mobile-screenshots/android-3.jpg deleted file mode 100644 index 87619b21..00000000 Binary files a/assets/mobile-screenshots/android-3.jpg and /dev/null differ diff --git a/assets/mobile-screenshots/android-4.jpg b/assets/mobile-screenshots/android-4.jpg deleted file mode 100644 index 2d1e58e2..00000000 Binary files a/assets/mobile-screenshots/android-4.jpg and /dev/null differ diff --git a/assets/mobile-screenshots/android-5.jpg b/assets/mobile-screenshots/android-5.jpg deleted file mode 100644 index fc4b2c9a..00000000 Binary files a/assets/mobile-screenshots/android-5.jpg and /dev/null differ diff --git a/assets/mobile-screenshots/combined.png b/assets/mobile-screenshots/combined.png deleted file mode 100644 index 23c761f9..00000000 Binary files a/assets/mobile-screenshots/combined.png and /dev/null differ diff --git a/assets/plugins/spotube-plugin-musicbrainz-listenbrainz/plugin.smplug b/assets/plugins/spotube-plugin-musicbrainz-listenbrainz/plugin.smplug new file mode 100644 index 00000000..41be05a4 Binary files /dev/null and b/assets/plugins/spotube-plugin-musicbrainz-listenbrainz/plugin.smplug differ diff --git a/assets/plugins/spotube-plugin-youtube-audio/plugin.smplug b/assets/plugins/spotube-plugin-youtube-audio/plugin.smplug new file mode 100644 index 00000000..55aa2895 Binary files /dev/null and b/assets/plugins/spotube-plugin-youtube-audio/plugin.smplug differ diff --git a/assets/spotube-logo-foreground.jpg b/assets/spotube-logo-foreground.jpg deleted file mode 100644 index 0af774c9..00000000 Binary files a/assets/spotube-logo-foreground.jpg and /dev/null differ diff --git a/assets/spotube-logo-macos.png b/assets/spotube-logo-macos.png deleted file mode 100644 index b14a7691..00000000 Binary files a/assets/spotube-logo-macos.png and /dev/null differ diff --git a/assets/spotube-logo.bmp b/assets/spotube-logo.bmp deleted file mode 100644 index c3503e85..00000000 Binary files a/assets/spotube-logo.bmp and /dev/null differ diff --git a/assets/spotube-logo.ico b/assets/spotube-logo.ico deleted file mode 100644 index 84906308..00000000 Binary files a/assets/spotube-logo.ico and /dev/null differ diff --git a/assets/spotube-logo.png b/assets/spotube-logo.png deleted file mode 100644 index b24a8c23..00000000 Binary files a/assets/spotube-logo.png and /dev/null differ diff --git a/assets/spotube-logo.svg b/assets/spotube-logo.svg deleted file mode 100644 index 5cd88f8e..00000000 --- a/assets/spotube-logo.svg +++ /dev/null @@ -1,349 +0,0 @@ - - diff --git a/assets/spotube-logo_android12.png b/assets/spotube-logo_android12.png deleted file mode 100644 index f04e25b0..00000000 Binary files a/assets/spotube-logo_android12.png and /dev/null differ diff --git a/assets/spotube-nightly-logo-foreground.jpg b/assets/spotube-nightly-logo-foreground.jpg deleted file mode 100644 index a0c849b6..00000000 Binary files a/assets/spotube-nightly-logo-foreground.jpg and /dev/null differ diff --git a/assets/spotube-nightly-logo.png b/assets/spotube-nightly-logo.png deleted file mode 100644 index ea7a8b20..00000000 Binary files a/assets/spotube-nightly-logo.png and /dev/null differ diff --git a/assets/spotube-nightly-logo.svg b/assets/spotube-nightly-logo.svg deleted file mode 100644 index 7601108e..00000000 --- a/assets/spotube-nightly-logo.svg +++ /dev/null @@ -1,359 +0,0 @@ - - diff --git a/assets/spotube-nightly-logo_android12.png b/assets/spotube-nightly-logo_android12.png deleted file mode 100644 index 1a5bf4f1..00000000 Binary files a/assets/spotube-nightly-logo_android12.png and /dev/null differ diff --git a/assets/spotube-screenshot.png b/assets/spotube-screenshot.png deleted file mode 100644 index 44567ae6..00000000 Binary files a/assets/spotube-screenshot.png and /dev/null differ diff --git a/assets/spotube-tall-capsule.png b/assets/spotube-tall-capsule.png deleted file mode 100644 index 43fb8229..00000000 Binary files a/assets/spotube-tall-capsule.png and /dev/null differ diff --git a/assets/spotube-wide-capsule-large.png b/assets/spotube-wide-capsule-large.png deleted file mode 100644 index 09a93d83..00000000 Binary files a/assets/spotube-wide-capsule-large.png and /dev/null differ diff --git a/assets/spotube-wide-capsule-small.png b/assets/spotube-wide-capsule-small.png deleted file mode 100644 index 17566550..00000000 Binary files a/assets/spotube-wide-capsule-small.png and /dev/null differ diff --git a/assets/spotube_banner.png b/assets/spotube_banner.png deleted file mode 100644 index b2be4539..00000000 Binary files a/assets/spotube_banner.png and /dev/null differ diff --git a/assets/success.png b/assets/success.png deleted file mode 100644 index 65cdba35..00000000 Binary files a/assets/success.png and /dev/null differ diff --git a/assets/tutorial/step-1.png b/assets/tutorial/step-1.png deleted file mode 100644 index 1182f054..00000000 Binary files a/assets/tutorial/step-1.png and /dev/null differ diff --git a/assets/tutorial/step-2.png b/assets/tutorial/step-2.png deleted file mode 100644 index af4616b0..00000000 Binary files a/assets/tutorial/step-2.png and /dev/null differ diff --git a/assets/tutorial/step-3.png b/assets/tutorial/step-3.png deleted file mode 100644 index ddbea140..00000000 Binary files a/assets/tutorial/step-3.png and /dev/null differ diff --git a/aur-struct/.SRCINFO b/aur-struct/.SRCINFO index 4c07a045..0460932b 100644 --- a/aur-struct/.SRCINFO +++ b/aur-struct/.SRCINFO @@ -1,8 +1,8 @@ pkgbase = spotube-bin -pkgdesc = Open source Spotify client that doesn't require Premium nor uses Electron! Available for both desktop & mobile! -pkgver = 3.7.1 -pkgrel = 2 -url = https://github.com/KRTirtho/spotube/ +pkgdesc = Open source Music client that doesn't require Premium nor uses Electron! Available for both desktop & mobile! +pkgver = 4.0.0 +pkgrel = 1 +url = https://spotube.krtirtho.dev arch = x86_64 license = BSD-4-Clause depends = mpv @@ -12,6 +12,7 @@ depends = jsoncpp depends = libnotify depends = xdg-user-dirs depends = webkit2gtk-4.1 +optdepends = yt-dlp-git source = https://github.com/KRTirtho/spotube/releases/download/v3.7.1/spotube-linux-3.7.1-x86_64.tar.xz md5sums = 475b1ae9b08f27743a4d4749391ae3db diff --git a/aur-struct/PKGBUILD b/aur-struct/PKGBUILD index d7e1052b..4f884f0d 100644 --- a/aur-struct/PKGBUILD +++ b/aur-struct/PKGBUILD @@ -3,15 +3,15 @@ pkgname=spotube-bin pkgver=%{{SPOTUBE_VERSION}}% pkgrel=%{{PKGREL}}% epoch= -pkgdesc="Open source Spotify client that doesn't require Premium nor uses Electron! Available for both desktop & mobile!" +pkgdesc="Open source Music client that doesn't require Premium nor uses Electron! Available for both desktop & mobile!" arch=(x86_64) -url="https://github.com/KRTirtho/spotube/" +url="https://spotube.krtirtho.dev" license=('BSD-4-Clause') groups=() depends=('mpv' 'libappindicator-gtk3' 'libsecret' 'jsoncpp' 'libnotify' 'xdg-user-dirs' 'webkit2gtk-4.1') makedepends=() checkdepends=() -optdepends=() +optdepends=('yt-dlp-git') provides=() conflicts=() replaces=() diff --git a/build.yaml b/build.yaml index 17d5bc50..76771f22 100644 --- a/build.yaml +++ b/build.yaml @@ -4,6 +4,16 @@ targets: exclude: - bin/*.dart builders: + auto_route_generator:auto_route_generator: # this for @RoutePage + options: + enable_cached_builds: true + generate_for: + - lib/pages/**/*.dart + auto_route_generator:auto_router_generator: # this for @AutoRouterConfig + options: + enable_cached_builds: true + generate_for: + - lib/collections/routes.dart json_serializable: options: any_map: true diff --git a/choco-struct/spotube.nuspec b/choco-struct/spotube.nuspec index 1cef4354..40941c08 100644 --- a/choco-struct/spotube.nuspec +++ b/choco-struct/spotube.nuspec @@ -1,5 +1,6 @@ - + @@ -12,34 +13,39 @@ spotube (Install) Kingkor Roy Tirtho - https://github.com/KRTirtho/spotube/ - https://rawcdn.githack.com/KRTirtho/spotube/7edb0bb834eb18c05551e30a891720a6abf53dbe/assets/spotube-logo.png + https://spotube.krtirtho.dev + + https://rawcdn.githack.com/KRTirtho/spotube/7edb0bb834eb18c05551e30a891720a6abf53dbe/assets/branding/spotube-logo.png 2022 Spotube https://github.com/KRTirtho/spotube/blob/master/LICENSE true https://github.com/KRTirtho/spotube - https://github.com/KRTirtho/spotube#readme + https://spotube.krtirtho.dev https://github.com/KRTirtho/spotube/issues/new - spotube music audio spotify youtube flutter - 🎧 Open source Spotify client that doesn't require Premium nor uses Electron! Available for both desktop & mobile! + spotube music audio youtube flutter + 🎧 Open source music client that doesn't require Premium nor uses Electron! Available + for both desktop & mobile! - Spotube is a Flutter based lightweight spotify client. It utilizes the power - of Spotify & Youtube's public API & creates a hazardless, performant & resource - friendly User Experience + Spotube is a Flutter based lightweight music client. It utilizes the power + of music metadata providers & Youtube's public API & creates a hazardless, performant + & resource + friendly User Experience - # Features - - Open source/libre software - - Anonymous/guest login - - Cross platform support - - No telemetry, diagnostics or user data collection - - Lightweight & resource-friendly - - Native performance (Thanks to Flutter+Skia) - - Playback control is done locally instead of on the server - - Small size & less data usage - - No Spotify or YouTube ads since it uses all public & free APIs (It is still recommended to support the creators by watching/liking/subscribing to the artists' YouTube channels or liking their tracks on Spotify. Purchasing Spotify Premium is usually the best way to support their valuable creations.) - - Time synced lyrics - - Downloadable tracks + # Features + - Open source/libre software + - Anonymous/guest login + - Cross platform support + - No telemetry, diagnostics or user data collection + - Lightweight & resource-friendly + - Native performance (Thanks to Flutter+Skia) + - Playback control is done locally instead of on the server + - Small size & less data usage + - No ads since it uses all public & free APIs (It is still recommended + to support the creators by watching/liking/subscribing to the artists' YouTube channels or + liking their tracks on different music platforms.) + - Time synced lyrics + - Downloadable tracks https://github.com/KRTirtho/spotube/releases/tag/v%{{SPOTUBE_VERSION}}% diff --git a/choco-struct/tools/LICENSE.txt b/choco-struct/tools/LICENSE.txt index 6d460a42..1a285107 100644 --- a/choco-struct/tools/LICENSE.txt +++ b/choco-struct/tools/LICENSE.txt @@ -2,7 +2,7 @@ BSD 4-Clause License -Copyright (c) 2022 Kingkor Roy Tirtho. All rights reserved. +Copyright (c) 2025 Kingkor Roy Tirtho. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: diff --git a/cli/commands/build.dart b/cli/commands/build.dart index fdf35a95..e0c254ff 100644 --- a/cli/commands/build.dart +++ b/cli/commands/build.dart @@ -3,7 +3,6 @@ import 'package:args/command_runner.dart'; import 'build/android.dart'; import 'build/ios.dart'; import 'build/linux.dart'; -import 'build/linux_arm.dart'; import 'build/macos.dart'; import 'build/windows.dart'; @@ -18,8 +17,13 @@ class BuildCommand extends Command { addSubcommand(AndroidBuildCommand()); addSubcommand(IosBuildCommand()); addSubcommand(LinuxBuildCommand()); - addSubcommand(LinuxArmBuildCommand()); addSubcommand(MacosBuildCommand()); addSubcommand(WindowsBuildCommand()); + argParser.addOption( + "arch", + abbr: "a", + defaultsTo: "x86", + allowed: ["x86", "arm64", "all"], + ); } } diff --git a/cli/commands/build/android.dart b/cli/commands/build/android.dart index 4216553a..b9edeb84 100644 --- a/cli/commands/build/android.dart +++ b/cli/commands/build/android.dart @@ -2,9 +2,7 @@ import 'dart:async'; import 'dart:io'; import 'package:args/command_runner.dart'; -import 'package:collection/collection.dart'; import 'package:path/path.dart'; -import 'package:xml/xml.dart'; import '../../core/env.dart'; import 'common.dart'; @@ -24,39 +22,6 @@ class AndroidBuildCommand extends Command with BuildCommandCommonSteps { "flutter build apk --flavor ${CliEnv.channel.name}", ); - await dotEnvFile.writeAsString( - "\nENABLE_UPDATE_CHECK=0" - "\nHIDE_DONATIONS=1", - mode: FileMode.append, - ); - - final androidManifestFile = File( - join(cwd.path, "android", "app", "src", "main", "AndroidManifest.xml")); - - final androidManifestXml = - XmlDocument.parse(await androidManifestFile.readAsString()); - - final deletingElement = - androidManifestXml.findAllElements("meta-data").firstWhereOrNull( - (el) => - el.getAttribute("android:name") == - "com.google.android.gms.car.application", - ); - - deletingElement?.parent?.children.remove(deletingElement); - - await androidManifestFile.writeAsString( - androidManifestXml.toXmlString(pretty: true), - ); - - await shell.run( - """ - dart run build_runner clean - dart run build_runner build --delete-conflicting-outputs - flutter build appbundle --flavor ${CliEnv.channel.name} - """, - ); - final ogApkFile = File( join( "build", @@ -71,22 +36,6 @@ class AndroidBuildCommand extends Command with BuildCommandCommonSteps { join(cwd.path, "build", "Spotube-android-all-arch.apk"), ); - final ogAppbundleFile = File( - join( - cwd.path, - "build", - "app", - "outputs", - "bundle", - "${CliEnv.channel.name}Release", - "app-${CliEnv.channel.name}-release.aab", - ), - ); - - await ogAppbundleFile.copy( - join(cwd.path, "build", "Spotube-playstore-all-arch.aab"), - ); - stdout.writeln("✅ Built Android Apk and Appbundle"); } } diff --git a/cli/commands/build/common.dart b/cli/commands/build/common.dart index 4c7e3e51..c30197f5 100644 --- a/cli/commands/build/common.dart +++ b/cli/commands/build/common.dart @@ -59,8 +59,10 @@ mixin BuildCommandCommonSteps on Command { """ flutter pub get dart run build_runner build --delete-conflicting-outputs - dart pub global activate flutter_distributor + dart pub global activate fastforge """, ); } + + String get architecture => parent?.argResults?.option("arch") as String; } diff --git a/cli/commands/build/linux.dart b/cli/commands/build/linux.dart index a218720c..3ca792ea 100644 --- a/cli/commands/build/linux.dart +++ b/cli/commands/build/linux.dart @@ -37,23 +37,31 @@ class LinuxBuildCommand extends Command with BuildCommandCommonSteps { await bootstrap(); await shell.run( - """ - flutter_distributor package --platform=linux --targets=deb - flutter_distributor package --platform=linux --targets=rpm - """, + "fastforge package --platform=linux --targets=deb,appimage", ); + if (architecture == "x86") { + await shell.run( + "fastforge package --platform=linux --targets=rpm", + ); + } final tempDir = join(Directory.systemTemp.path, "spotube-tar"); - - final bundleDirPath = - join(cwd.path, "build", "linux", "x64", "release", "bundle"); + final bundleArchName = architecture == "x86" ? "x86_64" : "aarch64"; + final bundleDirPath = join( + cwd.path, + "build", + "linux", + architecture == "x86" ? "x64" : architecture, + "release", + "bundle", + ); final tarFile = File(join( cwd.path, "dist", "spotube-linux-" "${CliEnv.channel == BuildChannel.nightly ? "nightly" : versionWithoutBuildNumber}" - "-x86_64.tar.xz", + "-$bundleArchName.tar.xz", )); await copyPath(bundleDirPath, tempDir); @@ -65,7 +73,7 @@ class LinuxBuildCommand extends Command with BuildCommandCommonSteps { ).copy( join(tempDir, "com.github.KRTirtho.Spotube.appdata.xml"), ); - await File(join(cwd.path, "assets", "spotube-logo.png")).copy( + await File(join(cwd.path, "assets", "branding", "spotube-logo.png")).copy( join(tempDir, "spotube-logo.png"), ); @@ -81,25 +89,48 @@ class LinuxBuildCommand extends Command with BuildCommandCommonSteps { "spotube-${pubspec.version}-linux.deb", ), ); + await ogDeb.copy( + join( + cwd.path, + "dist", + "Spotube-linux-$bundleArchName.deb", + ), + ); + await ogDeb.delete(); - final ogRpm = File( + if (architecture == "x86") { + final ogRpm = File( + join( + cwd.path, + "dist", + pubspec.version.toString(), + "spotube-${pubspec.version}-linux.rpm", + ), + ); + + await ogRpm.copy( + join(cwd.path, "dist", "Spotube-linux-$bundleArchName.rpm"), + ); + + await ogRpm.delete(); + } + + final ogAppImage = File( join( cwd.path, "dist", pubspec.version.toString(), - "spotube-${pubspec.version}-linux.rpm", + "spotube-${pubspec.version}-linux.AppImage", ), ); - - await ogDeb.copy( - join(cwd.path, "dist", "Spotube-linux-x86_64.deb"), + await ogAppImage.copy( + join( + cwd.path, + "dist", + "Spotube-linux-$bundleArchName.AppImage", + ), ); - await ogRpm.copy( - join(cwd.path, "dist", "Spotube-linux-x86_64.rpm"), - ); - - await ogDeb.delete(); - await ogRpm.delete(); + await ogAppImage.delete(); stdout.writeln("✅ Linux building done"); } diff --git a/cli/commands/build/linux_arm.dart b/cli/commands/build/linux_arm.dart deleted file mode 100644 index a09f0980..00000000 --- a/cli/commands/build/linux_arm.dart +++ /dev/null @@ -1,37 +0,0 @@ -import 'dart:async'; - -import 'package:args/command_runner.dart'; -import 'package:path/path.dart'; - -import '../../core/env.dart'; -import 'common.dart'; - -class LinuxArmBuildCommand extends Command with BuildCommandCommonSteps { - @override - String get description => "Build Linux Arm"; - - @override - String get name => "linux_arm"; - - @override - FutureOr? run() async { - await bootstrap(); - - await shell.run( - "docker buildx build --platform=linux/arm64 " - "-f ${join(cwd.path, ".github", "Dockerfile")} ${cwd.path} " - "--build-arg FLUTTER_VERSION=${CliEnv.flutterVersion} " - "--build-arg BUILD_VERSION=${CliEnv.channel == BuildChannel.nightly ? "nightly" : versionWithoutBuildNumber} " - "-t krtirtho/spotube_linux_arm:latest " - "--load", - ); - - await shell.run( - """ - docker images ls - docker create --name spotube_linux_arm krtirtho/spotube_linux_arm:latest - docker cp spotube_linux_arm:/app/dist/ dist/ - """, - ); - } -} diff --git a/cli/commands/build/macos.dart b/cli/commands/build/macos.dart index e8f34b77..936f1fc8 100644 --- a/cli/commands/build/macos.dart +++ b/cli/commands/build/macos.dart @@ -21,7 +21,7 @@ class MacosBuildCommand extends Command with BuildCommandCommonSteps { """ flutter build macos appdmg appdmg.json ${join(cwd.path, "build", "Spotube-macos-universal.dmg")} - flutter_distributor package --platform=macos --targets pkg --skip-clean + fastforge package --platform=macos --targets pkg --skip-clean """, ); diff --git a/cli/commands/build/windows.dart b/cli/commands/build/windows.dart index c44ed52f..1045c11c 100644 --- a/cli/commands/build/windows.dart +++ b/cli/commands/build/windows.dart @@ -61,7 +61,7 @@ class WindowsBuildCommand extends Command with BuildCommandCommonSteps { ); await shell.run( - "flutter_distributor package --platform=windows --targets=exe --skip-clean", + "fastforge package --platform=windows --targets=exe --skip-clean", ); final ogExe = File( diff --git a/cli/commands/install-dependencies.dart b/cli/commands/install-dependencies.dart index dc519cc6..56f679f1 100644 --- a/cli/commands/install-dependencies.dart +++ b/cli/commands/install-dependencies.dart @@ -24,28 +24,37 @@ class InstallDependenciesCommand extends Command { ], mandatory: true, ); + + argParser.addOption( + "arch", + abbr: "a", + allowed: ["x86", "arm64", "all"], + defaultsTo: "x86", + ); } @override FutureOr? run() async { final shell = Shell(); + final arch = argResults?.option("arch") == "x86" ? "x86_64" : "aarch64"; + switch (argResults!.option("platform")) { case "windows": + await shell.run( + """ + choco install innosetup -y + """, + ); break; case "linux": await shell.run( """ sudo apt-get update -y - sudo apt-get install -y tar clang cmake ninja-build pkg-config libgtk-3-dev make python3-pip python3-setuptools desktop-file-utils libgdk-pixbuf2.0-dev fakeroot strace fuse libunwind-dev locate patchelf gir1.2-appindicator3-0.1 libappindicator3-1 libappindicator3-dev libsecret-1-0 libjsoncpp25 libsecret-1-dev libjsoncpp-dev libnotify-bin libnotify-dev mpv libmpv-dev libwebkit2gtk-4.1-0 libwebkit2gtk-4.1-dev libsoup-3.0-0 libsoup-3.0-dev - """, - ); - break; - case "linux_arm": - await shell.run( - """ - sudo apt-get update -y - sudo apt-get install -y pkg-config make python3-pip python3-setuptools + sudo apt-get install -y wget tar clang cmake ninja-build pkg-config libgtk-3-dev make python3-pip python3-setuptools desktop-file-utils libgdk-pixbuf2.0-dev fakeroot strace fuse libunwind-dev locate patchelf gir1.2-appindicator3-0.1 libappindicator3-1 libappindicator3-dev libsecret-1-0 libjsoncpp25 libsecret-1-dev libjsoncpp-dev libnotify-bin libnotify-dev mpv libmpv-dev libwebkit2gtk-4.1-0 libwebkit2gtk-4.1-dev libsoup-3.0-0 libsoup-3.0-dev + wget -O appimagetool "https://github.com/AppImage/appimagetool/releases/download/continuous/appimagetool-$arch.AppImage" + chmod +x appimagetool + sudo mv appimagetool /usr/local/bin/ """, ); break; diff --git a/cliff.toml b/cliff.toml new file mode 100644 index 00000000..311a7d7c --- /dev/null +++ b/cliff.toml @@ -0,0 +1,92 @@ +# git-cliff ~ configuration file +# https://git-cliff.org/docs/configuration + + +[changelog] +# A Tera template to be rendered for each release in the changelog. +# See https://keats.github.io/tera/docs/#introduction +body = """ +{% if version %}\ + ## [{{ version | trim_start_matches(pat="v") }}](/compare/v{{ previous.version | trim_start_matches(pat="v") }}...v{{ version | trim_start_matches(pat="v") }}) ({{ timestamp | date(format="%Y-%m-%d") }}) +{% else %}\ + ## [unreleased] +{% endif %}\ +{% for group, commits in commits | group_by(attribute="group") %} + ### {{ group | striptags | trim | upper_first }} + {% for commit in commits %} + - {% if commit.scope %}**{{ commit.scope }}**: {% endif %}\ + {% if commit.breaking %}[**breaking**] {% endif %}\ + {{ commit.message | upper_first }}\ + {% endfor %} +{% endfor %} +""" +# Remove leading and trailing whitespaces from the changelog's body. +trim = true +# Render body even when there are no releases to process. +render_always = true +# An array of regex based postprocessors to modify the changelog. +postprocessors = [ + # Replace the placeholder with a URL. + { pattern = '', replace = "https://github.com/KRTirtho/spotube" }, +] +# render body even when there are no releases to process +# render_always = true +# output file path +# output = "test.md" + +[git] +# Parse commits according to the conventional commits specification. +# See https://www.conventionalcommits.org +conventional_commits = true +# Exclude commits that do not match the conventional commits specification. +filter_unconventional = true +# Require all commits to be conventional. +# Takes precedence over filter_unconventional. +require_conventional = false +# Split commits on newlines, treating each line as an individual commit. +split_commits = false +# An array of regex based parsers to modify commit messages prior to further processing. +commit_preprocessors = [ + # Replace issue numbers with link templates to be updated in `changelog.postprocessors`. + { pattern = '\((\w+\s)?#([0-9]+)\)', replace = "([#${2}](/issues/${2}))" }, + # Check spelling of the commit message using https://github.com/crate-ci/typos. + # If the spelling is incorrect, it will be fixed automatically. + #{ pattern = '.*', replace_command = 'typos --write-changes -' }, +] +# Prevent commits that are breaking from being excluded by commit parsers. +protect_breaking_commits = false +# An array of regex based parsers for extracting data from the commit message. +# Assigns commits to groups. +# Optionally sets the commit's scope and can decide to exclude commits from further processing. +commit_parsers = [ + { message = "^feat", group = "Features" }, + { message = "^fix", group = "Bug Fixes" }, + { message = "^translation", group = " Translation" }, + # { message = "^perf", group = "⚡ Performance" }, + # { message = "^refactor", group = "🚜 Refactor" }, + # { message = "^style", group = "🎨 Styling" }, + # { message = "^test", group = "🧪 Testing" }, + # { message = "^chore\\(release\\): prepare for", skip = true }, + # { message = "^chore\\(deps.*\\)", skip = true }, + # { message = "^chore\\(pr\\)", skip = true }, + # { message = "^chore\\(pull\\)", skip = true }, + # { message = "^chore|^ci", group = "⚙️ Miscellaneous Tasks" }, + # { body = ".*security", group = "🛡️ Security" }, + # { message = "^revert", group = "◀️ Revert" }, + # { message = ".*", group = "💼 Other" }, +] +# Exclude commits that are not matched by any commit parser. +filter_commits = true +# An array of link parsers for extracting external references, and turning them into URLs, using regex. +link_parsers = [] +# Include only the tags that belong to the current branch. +use_branch_tags = false +# Order releases topologically instead of chronologically. +topo_order = false +# Order releases topologically instead of chronologically. +topo_order_commits = true +# Order of commits in each group/release within the changelog. +# Allowed values: newest, oldest +sort_commits = "oldest" +# Process submodules commits +recurse_submodules = false diff --git a/drift_schemas/app_db/drift_schema_v10.json b/drift_schemas/app_db/drift_schema_v10.json new file mode 100644 index 00000000..5fb86d25 --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v10.json @@ -0,0 +1 @@ +{"_meta":{"description":"This file contains a serialized version of schema entities for drift.","version":"1.2.0"},"options":{"store_date_time_values_as_text":false},"entities":[{"id":0,"references":[],"type":"table","data":{"name":"authentication_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"cookie","getter_name":"cookie","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"access_token","getter_name":"accessToken","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"expiration","getter_name":"expiration","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":1,"references":[],"type":"table","data":{"name":"blacklist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"element_type","getter_name":"elementType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(BlacklistedType.values)","dart_type_name":"BlacklistedType"}},{"name":"element_id","getter_name":"elementId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":2,"references":[],"type":"table","data":{"name":"preferences_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"album_color_sync","getter_name":"albumColorSync","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"album_color_sync\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"album_color_sync\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"amoled_dark_theme","getter_name":"amoledDarkTheme","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"amoled_dark_theme\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"amoled_dark_theme\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"check_update","getter_name":"checkUpdate","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"check_update\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"check_update\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"normalize_audio","getter_name":"normalizeAudio","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"normalize_audio\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"normalize_audio\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"show_system_tray_icon","getter_name":"showSystemTrayIcon","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"show_system_tray_icon\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"show_system_tray_icon\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"system_title_bar","getter_name":"systemTitleBar","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"system_title_bar\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"system_title_bar\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"skip_non_music","getter_name":"skipNonMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"skip_non_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"skip_non_music\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"close_behavior","getter_name":"closeBehavior","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(CloseBehavior.close.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(CloseBehavior.values)","dart_type_name":"CloseBehavior"}},{"name":"accent_color_scheme","getter_name":"accentColorScheme","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"Slate:0xff64748b\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeColorConverter()","dart_type_name":"SpotubeColor"}},{"name":"layout_mode","getter_name":"layoutMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(LayoutMode.adaptive.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(LayoutMode.values)","dart_type_name":"LayoutMode"}},{"name":"locale","getter_name":"locale","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const LocaleConverter()","dart_type_name":"Locale"}},{"name":"market","getter_name":"market","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(Market.US.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(Market.values)","dart_type_name":"Market"}},{"name":"search_mode","getter_name":"searchMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SearchMode.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SearchMode.values)","dart_type_name":"SearchMode"}},{"name":"download_location","getter_name":"downloadLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[]},{"name":"local_library_location","getter_name":"localLibraryLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"theme_mode","getter_name":"themeMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(ThemeMode.system.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(ThemeMode.values)","dart_type_name":"ThemeMode"}},{"name":"audio_source_id","getter_name":"audioSourceId","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"youtube_client_engine","getter_name":"youtubeClientEngine","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(YoutubeClientEngine.youtubeExplode.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(YoutubeClientEngine.values)","dart_type_name":"YoutubeClientEngine"}},{"name":"discord_presence","getter_name":"discordPresence","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"discord_presence\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"discord_presence\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"endless_playback","getter_name":"endlessPlayback","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"endless_playback\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"endless_playback\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"enable_connect","getter_name":"enableConnect","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"enable_connect\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"enable_connect\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"connect_port","getter_name":"connectPort","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(-1)","default_client_dart":null,"dsl_features":[]},{"name":"cache_music","getter_name":"cacheMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"cache_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"cache_music\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":3,"references":[],"type":"table","data":{"name":"scrobbler_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"username","getter_name":"username","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"password_hash","getter_name":"passwordHash","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":4,"references":[],"type":"table","data":{"name":"skip_segment_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"start","getter_name":"start","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"end","getter_name":"end","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":5,"references":[],"type":"table","data":{"name":"source_match_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_info","getter_name":"sourceInfo","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"{}\")","default_client_dart":null,"dsl_features":[]},{"name":"source_type","getter_name":"sourceType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":6,"references":[],"type":"table","data":{"name":"audio_player_state_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playing","getter_name":"playing","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"playing\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"playing\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"loop_mode","getter_name":"loopMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(PlaylistMode.values)","dart_type_name":"PlaylistMode"}},{"name":"shuffled","getter_name":"shuffled","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"shuffled\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"shuffled\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"collections","getter_name":"collections","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"tracks","getter_name":"tracks","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"[]\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeTrackObjectListConverter()","dart_type_name":"List"}},{"name":"current_index","getter_name":"currentIndex","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(0)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":7,"references":[],"type":"table","data":{"name":"history_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"type","getter_name":"type","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(HistoryEntryType.values)","dart_type_name":"HistoryEntryType"}},{"name":"item_id","getter_name":"itemId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":8,"references":[],"type":"table","data":{"name":"lyrics_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"SubtitleTypeConverter()","dart_type_name":"SubtitleSimple"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":9,"references":[],"type":"table","data":{"name":"plugins_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[{"allowed-lengths":{"min":1,"max":50}}]},{"name":"description","getter_name":"description","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"version","getter_name":"version","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"author","getter_name":"author","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"entry_point","getter_name":"entryPoint","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"apis","getter_name":"apis","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"abilities","getter_name":"abilities","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"selected_for_metadata","getter_name":"selectedForMetadata","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"selected_for_metadata\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"selected_for_metadata\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"selected_for_audio_source","getter_name":"selectedForAudioSource","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"selected_for_audio_source\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"selected_for_audio_source\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"repository","getter_name":"repository","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"plugin_api_version","getter_name":"pluginApiVersion","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('2.0.0')","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":10,"references":[1],"type":"index","data":{"on":1,"name":"unique_blacklist","sql":null,"unique":true,"columns":["element_type","element_id"]}},{"id":11,"references":[5],"type":"index","data":{"on":5,"name":"uniq_track_match","sql":null,"unique":true,"columns":["track_id","source_info","source_type"]}}]} \ No newline at end of file diff --git a/drift_schemas/app_db/drift_schema_v4.json b/drift_schemas/app_db/drift_schema_v4.json new file mode 100644 index 00000000..fc50a6f8 --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v4.json @@ -0,0 +1 @@ +{"_meta":{"description":"This file contains a serialized version of schema entities for drift.","version":"1.2.0"},"options":{"store_date_time_values_as_text":false},"entities":[{"id":0,"references":[],"type":"table","data":{"name":"authentication_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"cookie","getter_name":"cookie","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"access_token","getter_name":"accessToken","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"expiration","getter_name":"expiration","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":1,"references":[],"type":"table","data":{"name":"blacklist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"element_type","getter_name":"elementType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(BlacklistedType.values)","dart_type_name":"BlacklistedType"}},{"name":"element_id","getter_name":"elementId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":2,"references":[],"type":"table","data":{"name":"preferences_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_quality","getter_name":"audioQuality","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceQualities.high.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceQualities.values)","dart_type_name":"SourceQualities"}},{"name":"album_color_sync","getter_name":"albumColorSync","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"album_color_sync\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"album_color_sync\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"amoled_dark_theme","getter_name":"amoledDarkTheme","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"amoled_dark_theme\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"amoled_dark_theme\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"check_update","getter_name":"checkUpdate","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"check_update\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"check_update\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"normalize_audio","getter_name":"normalizeAudio","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"normalize_audio\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"normalize_audio\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"show_system_tray_icon","getter_name":"showSystemTrayIcon","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"show_system_tray_icon\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"show_system_tray_icon\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"system_title_bar","getter_name":"systemTitleBar","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"system_title_bar\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"system_title_bar\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"skip_non_music","getter_name":"skipNonMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"skip_non_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"skip_non_music\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"close_behavior","getter_name":"closeBehavior","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(CloseBehavior.close.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(CloseBehavior.values)","dart_type_name":"CloseBehavior"}},{"name":"accent_color_scheme","getter_name":"accentColorScheme","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"Blue:0xFF2196F3\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeColorConverter()","dart_type_name":"SpotubeColor"}},{"name":"layout_mode","getter_name":"layoutMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(LayoutMode.adaptive.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(LayoutMode.values)","dart_type_name":"LayoutMode"}},{"name":"locale","getter_name":"locale","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const LocaleConverter()","dart_type_name":"Locale"}},{"name":"market","getter_name":"market","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(Market.US.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(Market.values)","dart_type_name":"Market"}},{"name":"search_mode","getter_name":"searchMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SearchMode.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SearchMode.values)","dart_type_name":"SearchMode"}},{"name":"download_location","getter_name":"downloadLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[]},{"name":"local_library_location","getter_name":"localLibraryLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"piped_instance","getter_name":"pipedInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://pipedapi.kavin.rocks\")","default_client_dart":null,"dsl_features":[]},{"name":"invidious_instance","getter_name":"invidiousInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://inv.nadeko.net\")","default_client_dart":null,"dsl_features":[]},{"name":"theme_mode","getter_name":"themeMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(ThemeMode.system.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(ThemeMode.values)","dart_type_name":"ThemeMode"}},{"name":"audio_source","getter_name":"audioSource","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(AudioSource.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(AudioSource.values)","dart_type_name":"AudioSource"}},{"name":"youtube_client_engine","getter_name":"youtubeClientEngine","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(YoutubeClientEngine.youtubeExplode.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(YoutubeClientEngine.values)","dart_type_name":"YoutubeClientEngine"}},{"name":"stream_music_codec","getter_name":"streamMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.weba.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"download_music_codec","getter_name":"downloadMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.m4a.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"discord_presence","getter_name":"discordPresence","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"discord_presence\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"discord_presence\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"endless_playback","getter_name":"endlessPlayback","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"endless_playback\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"endless_playback\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"enable_connect","getter_name":"enableConnect","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"enable_connect\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"enable_connect\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"cache_music","getter_name":"cacheMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"cache_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"cache_music\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":3,"references":[],"type":"table","data":{"name":"scrobbler_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"username","getter_name":"username","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"password_hash","getter_name":"passwordHash","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":4,"references":[],"type":"table","data":{"name":"skip_segment_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"start","getter_name":"start","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"end","getter_name":"end","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":5,"references":[],"type":"table","data":{"name":"source_match_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_id","getter_name":"sourceId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_type","getter_name":"sourceType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceType.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceType.values)","dart_type_name":"SourceType"}},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":6,"references":[],"type":"table","data":{"name":"audio_player_state_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playing","getter_name":"playing","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"playing\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"playing\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"loop_mode","getter_name":"loopMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(PlaylistMode.values)","dart_type_name":"PlaylistMode"}},{"name":"shuffled","getter_name":"shuffled","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"shuffled\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"shuffled\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"collections","getter_name":"collections","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":7,"references":[6],"type":"table","data":{"name":"playlist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_player_state_id","getter_name":"audioPlayerStateId","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"REFERENCES audio_player_state_table (id)","dialectAwareDefaultConstraints":{"sqlite":"REFERENCES audio_player_state_table (id)"},"default_dart":null,"default_client_dart":null,"dsl_features":["unknown"]},{"name":"index","getter_name":"index","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":8,"references":[7],"type":"table","data":{"name":"playlist_media_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playlist_id","getter_name":"playlistId","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"REFERENCES playlist_table (id)","dialectAwareDefaultConstraints":{"sqlite":"REFERENCES playlist_table (id)"},"default_dart":null,"default_client_dart":null,"dsl_features":["unknown"]},{"name":"uri","getter_name":"uri","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"extras","getter_name":"extras","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}},{"name":"http_headers","getter_name":"httpHeaders","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":9,"references":[],"type":"table","data":{"name":"history_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"type","getter_name":"type","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(HistoryEntryType.values)","dart_type_name":"HistoryEntryType"}},{"name":"item_id","getter_name":"itemId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":10,"references":[],"type":"table","data":{"name":"lyrics_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"SubtitleTypeConverter()","dart_type_name":"SubtitleSimple"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":11,"references":[1],"type":"index","data":{"on":1,"name":"unique_blacklist","sql":null,"unique":true,"columns":["element_type","element_id"]}},{"id":12,"references":[5],"type":"index","data":{"on":5,"name":"uniq_track_match","sql":null,"unique":true,"columns":["track_id","source_id","source_type"]}}]} \ No newline at end of file diff --git a/drift_schemas/app_db/drift_schema_v5.json b/drift_schemas/app_db/drift_schema_v5.json new file mode 100644 index 00000000..eefe0205 --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v5.json @@ -0,0 +1 @@ +{"_meta":{"description":"This file contains a serialized version of schema entities for drift.","version":"1.2.0"},"options":{"store_date_time_values_as_text":false},"entities":[{"id":0,"references":[],"type":"table","data":{"name":"authentication_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"cookie","getter_name":"cookie","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"access_token","getter_name":"accessToken","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"expiration","getter_name":"expiration","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":1,"references":[],"type":"table","data":{"name":"blacklist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"element_type","getter_name":"elementType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(BlacklistedType.values)","dart_type_name":"BlacklistedType"}},{"name":"element_id","getter_name":"elementId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":2,"references":[],"type":"table","data":{"name":"preferences_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_quality","getter_name":"audioQuality","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceQualities.high.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceQualities.values)","dart_type_name":"SourceQualities"}},{"name":"album_color_sync","getter_name":"albumColorSync","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"album_color_sync\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"album_color_sync\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"amoled_dark_theme","getter_name":"amoledDarkTheme","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"amoled_dark_theme\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"amoled_dark_theme\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"check_update","getter_name":"checkUpdate","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"check_update\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"check_update\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"normalize_audio","getter_name":"normalizeAudio","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"normalize_audio\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"normalize_audio\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"show_system_tray_icon","getter_name":"showSystemTrayIcon","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"show_system_tray_icon\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"show_system_tray_icon\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"system_title_bar","getter_name":"systemTitleBar","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"system_title_bar\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"system_title_bar\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"skip_non_music","getter_name":"skipNonMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"skip_non_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"skip_non_music\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"close_behavior","getter_name":"closeBehavior","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(CloseBehavior.close.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(CloseBehavior.values)","dart_type_name":"CloseBehavior"}},{"name":"accent_color_scheme","getter_name":"accentColorScheme","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"Orange:0xFFf97315\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeColorConverter()","dart_type_name":"SpotubeColor"}},{"name":"layout_mode","getter_name":"layoutMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(LayoutMode.adaptive.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(LayoutMode.values)","dart_type_name":"LayoutMode"}},{"name":"locale","getter_name":"locale","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const LocaleConverter()","dart_type_name":"Locale"}},{"name":"market","getter_name":"market","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(Market.US.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(Market.values)","dart_type_name":"Market"}},{"name":"search_mode","getter_name":"searchMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SearchMode.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SearchMode.values)","dart_type_name":"SearchMode"}},{"name":"download_location","getter_name":"downloadLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[]},{"name":"local_library_location","getter_name":"localLibraryLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"piped_instance","getter_name":"pipedInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://pipedapi.kavin.rocks\")","default_client_dart":null,"dsl_features":[]},{"name":"invidious_instance","getter_name":"invidiousInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://inv.nadeko.net\")","default_client_dart":null,"dsl_features":[]},{"name":"theme_mode","getter_name":"themeMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(ThemeMode.system.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(ThemeMode.values)","dart_type_name":"ThemeMode"}},{"name":"audio_source","getter_name":"audioSource","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(AudioSource.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(AudioSource.values)","dart_type_name":"AudioSource"}},{"name":"youtube_client_engine","getter_name":"youtubeClientEngine","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(YoutubeClientEngine.youtubeExplode.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(YoutubeClientEngine.values)","dart_type_name":"YoutubeClientEngine"}},{"name":"stream_music_codec","getter_name":"streamMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.weba.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"download_music_codec","getter_name":"downloadMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.m4a.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"discord_presence","getter_name":"discordPresence","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"discord_presence\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"discord_presence\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"endless_playback","getter_name":"endlessPlayback","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"endless_playback\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"endless_playback\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"enable_connect","getter_name":"enableConnect","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"enable_connect\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"enable_connect\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"cache_music","getter_name":"cacheMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"cache_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"cache_music\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":3,"references":[],"type":"table","data":{"name":"scrobbler_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"username","getter_name":"username","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"password_hash","getter_name":"passwordHash","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":4,"references":[],"type":"table","data":{"name":"skip_segment_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"start","getter_name":"start","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"end","getter_name":"end","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":5,"references":[],"type":"table","data":{"name":"source_match_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_id","getter_name":"sourceId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_type","getter_name":"sourceType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceType.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceType.values)","dart_type_name":"SourceType"}},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":6,"references":[],"type":"table","data":{"name":"audio_player_state_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playing","getter_name":"playing","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"playing\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"playing\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"loop_mode","getter_name":"loopMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(PlaylistMode.values)","dart_type_name":"PlaylistMode"}},{"name":"shuffled","getter_name":"shuffled","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"shuffled\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"shuffled\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"collections","getter_name":"collections","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":7,"references":[6],"type":"table","data":{"name":"playlist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_player_state_id","getter_name":"audioPlayerStateId","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"REFERENCES audio_player_state_table (id)","dialectAwareDefaultConstraints":{"sqlite":"REFERENCES audio_player_state_table (id)"},"default_dart":null,"default_client_dart":null,"dsl_features":["unknown"]},{"name":"index","getter_name":"index","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":8,"references":[7],"type":"table","data":{"name":"playlist_media_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playlist_id","getter_name":"playlistId","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"REFERENCES playlist_table (id)","dialectAwareDefaultConstraints":{"sqlite":"REFERENCES playlist_table (id)"},"default_dart":null,"default_client_dart":null,"dsl_features":["unknown"]},{"name":"uri","getter_name":"uri","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"extras","getter_name":"extras","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}},{"name":"http_headers","getter_name":"httpHeaders","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":9,"references":[],"type":"table","data":{"name":"history_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"type","getter_name":"type","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(HistoryEntryType.values)","dart_type_name":"HistoryEntryType"}},{"name":"item_id","getter_name":"itemId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":10,"references":[],"type":"table","data":{"name":"lyrics_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"SubtitleTypeConverter()","dart_type_name":"SubtitleSimple"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":11,"references":[1],"type":"index","data":{"on":1,"name":"unique_blacklist","sql":null,"unique":true,"columns":["element_type","element_id"]}},{"id":12,"references":[5],"type":"index","data":{"on":5,"name":"uniq_track_match","sql":null,"unique":true,"columns":["track_id","source_id","source_type"]}}]} \ No newline at end of file diff --git a/drift_schemas/app_db/drift_schema_v6.json b/drift_schemas/app_db/drift_schema_v6.json new file mode 100644 index 00000000..8a646be1 --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v6.json @@ -0,0 +1 @@ +{"_meta":{"description":"This file contains a serialized version of schema entities for drift.","version":"1.2.0"},"options":{"store_date_time_values_as_text":false},"entities":[{"id":0,"references":[],"type":"table","data":{"name":"authentication_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"cookie","getter_name":"cookie","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"access_token","getter_name":"accessToken","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"expiration","getter_name":"expiration","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":1,"references":[],"type":"table","data":{"name":"blacklist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"element_type","getter_name":"elementType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(BlacklistedType.values)","dart_type_name":"BlacklistedType"}},{"name":"element_id","getter_name":"elementId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":2,"references":[],"type":"table","data":{"name":"preferences_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_quality","getter_name":"audioQuality","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceQualities.high.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceQualities.values)","dart_type_name":"SourceQualities"}},{"name":"album_color_sync","getter_name":"albumColorSync","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"album_color_sync\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"album_color_sync\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"amoled_dark_theme","getter_name":"amoledDarkTheme","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"amoled_dark_theme\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"amoled_dark_theme\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"check_update","getter_name":"checkUpdate","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"check_update\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"check_update\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"normalize_audio","getter_name":"normalizeAudio","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"normalize_audio\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"normalize_audio\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"show_system_tray_icon","getter_name":"showSystemTrayIcon","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"show_system_tray_icon\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"show_system_tray_icon\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"system_title_bar","getter_name":"systemTitleBar","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"system_title_bar\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"system_title_bar\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"skip_non_music","getter_name":"skipNonMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"skip_non_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"skip_non_music\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"close_behavior","getter_name":"closeBehavior","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(CloseBehavior.close.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(CloseBehavior.values)","dart_type_name":"CloseBehavior"}},{"name":"accent_color_scheme","getter_name":"accentColorScheme","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"Orange:0xFFf97315\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeColorConverter()","dart_type_name":"SpotubeColor"}},{"name":"layout_mode","getter_name":"layoutMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(LayoutMode.adaptive.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(LayoutMode.values)","dart_type_name":"LayoutMode"}},{"name":"locale","getter_name":"locale","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const LocaleConverter()","dart_type_name":"Locale"}},{"name":"market","getter_name":"market","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(Market.US.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(Market.values)","dart_type_name":"Market"}},{"name":"search_mode","getter_name":"searchMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SearchMode.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SearchMode.values)","dart_type_name":"SearchMode"}},{"name":"download_location","getter_name":"downloadLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[]},{"name":"local_library_location","getter_name":"localLibraryLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"piped_instance","getter_name":"pipedInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://pipedapi.kavin.rocks\")","default_client_dart":null,"dsl_features":[]},{"name":"invidious_instance","getter_name":"invidiousInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://inv.nadeko.net\")","default_client_dart":null,"dsl_features":[]},{"name":"theme_mode","getter_name":"themeMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(ThemeMode.system.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(ThemeMode.values)","dart_type_name":"ThemeMode"}},{"name":"audio_source","getter_name":"audioSource","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(AudioSource.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(AudioSource.values)","dart_type_name":"AudioSource"}},{"name":"youtube_client_engine","getter_name":"youtubeClientEngine","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(YoutubeClientEngine.youtubeExplode.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(YoutubeClientEngine.values)","dart_type_name":"YoutubeClientEngine"}},{"name":"stream_music_codec","getter_name":"streamMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.weba.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"download_music_codec","getter_name":"downloadMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.m4a.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"discord_presence","getter_name":"discordPresence","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"discord_presence\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"discord_presence\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"endless_playback","getter_name":"endlessPlayback","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"endless_playback\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"endless_playback\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"enable_connect","getter_name":"enableConnect","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"enable_connect\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"enable_connect\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"connect_port","getter_name":"connectPort","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(-1)","default_client_dart":null,"dsl_features":[]},{"name":"cache_music","getter_name":"cacheMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"cache_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"cache_music\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":3,"references":[],"type":"table","data":{"name":"scrobbler_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"username","getter_name":"username","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"password_hash","getter_name":"passwordHash","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":4,"references":[],"type":"table","data":{"name":"skip_segment_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"start","getter_name":"start","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"end","getter_name":"end","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":5,"references":[],"type":"table","data":{"name":"source_match_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_id","getter_name":"sourceId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_type","getter_name":"sourceType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceType.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceType.values)","dart_type_name":"SourceType"}},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":6,"references":[],"type":"table","data":{"name":"audio_player_state_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playing","getter_name":"playing","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"playing\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"playing\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"loop_mode","getter_name":"loopMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(PlaylistMode.values)","dart_type_name":"PlaylistMode"}},{"name":"shuffled","getter_name":"shuffled","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"shuffled\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"shuffled\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"collections","getter_name":"collections","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":7,"references":[6],"type":"table","data":{"name":"playlist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_player_state_id","getter_name":"audioPlayerStateId","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"REFERENCES audio_player_state_table (id)","dialectAwareDefaultConstraints":{"sqlite":"REFERENCES audio_player_state_table (id)"},"default_dart":null,"default_client_dart":null,"dsl_features":["unknown"]},{"name":"index","getter_name":"index","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":8,"references":[7],"type":"table","data":{"name":"playlist_media_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playlist_id","getter_name":"playlistId","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"REFERENCES playlist_table (id)","dialectAwareDefaultConstraints":{"sqlite":"REFERENCES playlist_table (id)"},"default_dart":null,"default_client_dart":null,"dsl_features":["unknown"]},{"name":"uri","getter_name":"uri","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"extras","getter_name":"extras","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}},{"name":"http_headers","getter_name":"httpHeaders","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":9,"references":[],"type":"table","data":{"name":"history_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"type","getter_name":"type","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(HistoryEntryType.values)","dart_type_name":"HistoryEntryType"}},{"name":"item_id","getter_name":"itemId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":10,"references":[],"type":"table","data":{"name":"lyrics_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"SubtitleTypeConverter()","dart_type_name":"SubtitleSimple"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":11,"references":[1],"type":"index","data":{"on":1,"name":"unique_blacklist","sql":null,"unique":true,"columns":["element_type","element_id"]}},{"id":12,"references":[5],"type":"index","data":{"on":5,"name":"uniq_track_match","sql":null,"unique":true,"columns":["track_id","source_id","source_type"]}}]} \ No newline at end of file diff --git a/drift_schemas/app_db/drift_schema_v7.json b/drift_schemas/app_db/drift_schema_v7.json new file mode 100644 index 00000000..d6644857 --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v7.json @@ -0,0 +1 @@ +{"_meta":{"description":"This file contains a serialized version of schema entities for drift.","version":"1.2.0"},"options":{"store_date_time_values_as_text":false},"entities":[{"id":0,"references":[],"type":"table","data":{"name":"authentication_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"cookie","getter_name":"cookie","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"access_token","getter_name":"accessToken","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"expiration","getter_name":"expiration","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":1,"references":[],"type":"table","data":{"name":"blacklist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"element_type","getter_name":"elementType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(BlacklistedType.values)","dart_type_name":"BlacklistedType"}},{"name":"element_id","getter_name":"elementId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":2,"references":[],"type":"table","data":{"name":"preferences_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_quality","getter_name":"audioQuality","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceQualities.high.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceQualities.values)","dart_type_name":"SourceQualities"}},{"name":"album_color_sync","getter_name":"albumColorSync","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"album_color_sync\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"album_color_sync\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"amoled_dark_theme","getter_name":"amoledDarkTheme","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"amoled_dark_theme\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"amoled_dark_theme\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"check_update","getter_name":"checkUpdate","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"check_update\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"check_update\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"normalize_audio","getter_name":"normalizeAudio","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"normalize_audio\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"normalize_audio\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"show_system_tray_icon","getter_name":"showSystemTrayIcon","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"show_system_tray_icon\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"show_system_tray_icon\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"system_title_bar","getter_name":"systemTitleBar","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"system_title_bar\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"system_title_bar\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"skip_non_music","getter_name":"skipNonMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"skip_non_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"skip_non_music\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"close_behavior","getter_name":"closeBehavior","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(CloseBehavior.close.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(CloseBehavior.values)","dart_type_name":"CloseBehavior"}},{"name":"accent_color_scheme","getter_name":"accentColorScheme","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"Orange:0xFFf97315\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeColorConverter()","dart_type_name":"SpotubeColor"}},{"name":"layout_mode","getter_name":"layoutMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(LayoutMode.adaptive.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(LayoutMode.values)","dart_type_name":"LayoutMode"}},{"name":"locale","getter_name":"locale","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const LocaleConverter()","dart_type_name":"Locale"}},{"name":"market","getter_name":"market","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(Market.US.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(Market.values)","dart_type_name":"Market"}},{"name":"search_mode","getter_name":"searchMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SearchMode.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SearchMode.values)","dart_type_name":"SearchMode"}},{"name":"download_location","getter_name":"downloadLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[]},{"name":"local_library_location","getter_name":"localLibraryLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"piped_instance","getter_name":"pipedInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://pipedapi.kavin.rocks\")","default_client_dart":null,"dsl_features":[]},{"name":"invidious_instance","getter_name":"invidiousInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://inv.nadeko.net\")","default_client_dart":null,"dsl_features":[]},{"name":"theme_mode","getter_name":"themeMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(ThemeMode.system.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(ThemeMode.values)","dart_type_name":"ThemeMode"}},{"name":"audio_source","getter_name":"audioSource","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(AudioSource.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(AudioSource.values)","dart_type_name":"AudioSource"}},{"name":"youtube_client_engine","getter_name":"youtubeClientEngine","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(YoutubeClientEngine.youtubeExplode.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(YoutubeClientEngine.values)","dart_type_name":"YoutubeClientEngine"}},{"name":"stream_music_codec","getter_name":"streamMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.weba.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"download_music_codec","getter_name":"downloadMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.m4a.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"discord_presence","getter_name":"discordPresence","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"discord_presence\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"discord_presence\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"endless_playback","getter_name":"endlessPlayback","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"endless_playback\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"endless_playback\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"enable_connect","getter_name":"enableConnect","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"enable_connect\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"enable_connect\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"connect_port","getter_name":"connectPort","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(-1)","default_client_dart":null,"dsl_features":[]},{"name":"cache_music","getter_name":"cacheMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"cache_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"cache_music\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":3,"references":[],"type":"table","data":{"name":"scrobbler_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"username","getter_name":"username","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"password_hash","getter_name":"passwordHash","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":4,"references":[],"type":"table","data":{"name":"skip_segment_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"start","getter_name":"start","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"end","getter_name":"end","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":5,"references":[],"type":"table","data":{"name":"source_match_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_id","getter_name":"sourceId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_type","getter_name":"sourceType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceType.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceType.values)","dart_type_name":"SourceType"}},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":6,"references":[],"type":"table","data":{"name":"audio_player_state_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playing","getter_name":"playing","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"playing\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"playing\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"loop_mode","getter_name":"loopMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(PlaylistMode.values)","dart_type_name":"PlaylistMode"}},{"name":"shuffled","getter_name":"shuffled","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"shuffled\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"shuffled\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"collections","getter_name":"collections","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"tracks","getter_name":"tracks","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"[]\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeTrackObjectListConverter()","dart_type_name":"List"}},{"name":"current_index","getter_name":"currentIndex","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(0)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":7,"references":[],"type":"table","data":{"name":"history_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"type","getter_name":"type","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(HistoryEntryType.values)","dart_type_name":"HistoryEntryType"}},{"name":"item_id","getter_name":"itemId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":8,"references":[],"type":"table","data":{"name":"lyrics_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"SubtitleTypeConverter()","dart_type_name":"SubtitleSimple"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":9,"references":[],"type":"table","data":{"name":"metadata_plugins_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[{"allowed-lengths":{"min":1,"max":50}}]},{"name":"description","getter_name":"description","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"version","getter_name":"version","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"author","getter_name":"author","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"entry_point","getter_name":"entryPoint","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"apis","getter_name":"apis","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"abilities","getter_name":"abilities","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"selected","getter_name":"selected","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"selected\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"selected\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"repository","getter_name":"repository","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"plugin_api_version","getter_name":"pluginApiVersion","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":10,"references":[1],"type":"index","data":{"on":1,"name":"unique_blacklist","sql":null,"unique":true,"columns":["element_type","element_id"]}},{"id":11,"references":[5],"type":"index","data":{"on":5,"name":"uniq_track_match","sql":null,"unique":true,"columns":["track_id","source_id","source_type"]}}]} \ No newline at end of file diff --git a/drift_schemas/app_db/drift_schema_v8.json b/drift_schemas/app_db/drift_schema_v8.json new file mode 100644 index 00000000..eba4c46e --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v8.json @@ -0,0 +1,1143 @@ +{ + "_meta": { + "description": "This file contains a serialized version of schema entities for drift.", + "version": "1.2.0" + }, + "options": { "store_date_time_values_as_text": false }, + "entities": [ + { + "id": 0, + "references": [], + "type": "table", + "data": { + "name": "authentication_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "cookie", + "getter_name": "cookie", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "EncryptedTextConverter()", + "dart_type_name": "DecryptedText" + } + }, + { + "name": "access_token", + "getter_name": "accessToken", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "EncryptedTextConverter()", + "dart_type_name": "DecryptedText" + } + }, + { + "name": "expiration", + "getter_name": "expiration", + "moor_type": "dateTime", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 1, + "references": [], + "type": "table", + "data": { + "name": "blacklist_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "name", + "getter_name": "name", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "element_type", + "getter_name": "elementType", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(BlacklistedType.values)", + "dart_type_name": "BlacklistedType" + } + }, + { + "name": "element_id", + "getter_name": "elementId", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 2, + "references": [], + "type": "table", + "data": { + "name": "preferences_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "audio_quality", + "getter_name": "audioQuality", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(SourceQualities.high.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(SourceQualities.values)", + "dart_type_name": "SourceQualities" + } + }, + { + "name": "album_color_sync", + "getter_name": "albumColorSync", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"album_color_sync\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"album_color_sync\" IN (0, 1))" + }, + "default_dart": "const Constant(true)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "amoled_dark_theme", + "getter_name": "amoledDarkTheme", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"amoled_dark_theme\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"amoled_dark_theme\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "check_update", + "getter_name": "checkUpdate", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"check_update\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"check_update\" IN (0, 1))" + }, + "default_dart": "const Constant(true)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "normalize_audio", + "getter_name": "normalizeAudio", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"normalize_audio\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"normalize_audio\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "show_system_tray_icon", + "getter_name": "showSystemTrayIcon", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"show_system_tray_icon\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"show_system_tray_icon\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "system_title_bar", + "getter_name": "systemTitleBar", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"system_title_bar\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"system_title_bar\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "skip_non_music", + "getter_name": "skipNonMusic", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"skip_non_music\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"skip_non_music\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "close_behavior", + "getter_name": "closeBehavior", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(CloseBehavior.close.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(CloseBehavior.values)", + "dart_type_name": "CloseBehavior" + } + }, + { + "name": "accent_color_scheme", + "getter_name": "accentColorScheme", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(\"Slate:0xff64748b\")", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const SpotubeColorConverter()", + "dart_type_name": "SpotubeColor" + } + }, + { + "name": "layout_mode", + "getter_name": "layoutMode", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(LayoutMode.adaptive.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(LayoutMode.values)", + "dart_type_name": "LayoutMode" + } + }, + { + "name": "locale", + "getter_name": "locale", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const LocaleConverter()", + "dart_type_name": "Locale" + } + }, + { + "name": "market", + "getter_name": "market", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(Market.US.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(Market.values)", + "dart_type_name": "Market" + } + }, + { + "name": "search_mode", + "getter_name": "searchMode", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(SearchMode.youtube.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(SearchMode.values)", + "dart_type_name": "SearchMode" + } + }, + { + "name": "download_location", + "getter_name": "downloadLocation", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(\"\")", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "local_library_location", + "getter_name": "localLibraryLocation", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(\"\")", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const StringListConverter()", + "dart_type_name": "List" + } + }, + { + "name": "piped_instance", + "getter_name": "pipedInstance", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(\"https://pipedapi.kavin.rocks\")", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "invidious_instance", + "getter_name": "invidiousInstance", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(\"https://inv.nadeko.net\")", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "theme_mode", + "getter_name": "themeMode", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(ThemeMode.system.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(ThemeMode.values)", + "dart_type_name": "ThemeMode" + } + }, + { + "name": "audio_source", + "getter_name": "audioSource", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(AudioSource.youtube.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(AudioSource.values)", + "dart_type_name": "AudioSource" + } + }, + { + "name": "youtube_client_engine", + "getter_name": "youtubeClientEngine", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(YoutubeClientEngine.youtubeExplode.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(YoutubeClientEngine.values)", + "dart_type_name": "YoutubeClientEngine" + } + }, + { + "name": "stream_music_codec", + "getter_name": "streamMusicCodec", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(SourceCodecs.weba.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(SourceCodecs.values)", + "dart_type_name": "SourceCodecs" + } + }, + { + "name": "download_music_codec", + "getter_name": "downloadMusicCodec", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(SourceCodecs.m4a.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(SourceCodecs.values)", + "dart_type_name": "SourceCodecs" + } + }, + { + "name": "discord_presence", + "getter_name": "discordPresence", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"discord_presence\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"discord_presence\" IN (0, 1))" + }, + "default_dart": "const Constant(true)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "endless_playback", + "getter_name": "endlessPlayback", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"endless_playback\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"endless_playback\" IN (0, 1))" + }, + "default_dart": "const Constant(true)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "enable_connect", + "getter_name": "enableConnect", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"enable_connect\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"enable_connect\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "connect_port", + "getter_name": "connectPort", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(-1)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "cache_music", + "getter_name": "cacheMusic", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"cache_music\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"cache_music\" IN (0, 1))" + }, + "default_dart": "const Constant(true)", + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 3, + "references": [], + "type": "table", + "data": { + "name": "scrobbler_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "created_at", + "getter_name": "createdAt", + "moor_type": "dateTime", + "nullable": false, + "customConstraints": null, + "default_dart": "currentDateAndTime", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "username", + "getter_name": "username", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "password_hash", + "getter_name": "passwordHash", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "EncryptedTextConverter()", + "dart_type_name": "DecryptedText" + } + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 4, + "references": [], + "type": "table", + "data": { + "name": "skip_segment_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "start", + "getter_name": "start", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "end", + "getter_name": "end", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "track_id", + "getter_name": "trackId", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "created_at", + "getter_name": "createdAt", + "moor_type": "dateTime", + "nullable": false, + "customConstraints": null, + "default_dart": "currentDateAndTime", + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 5, + "references": [], + "type": "table", + "data": { + "name": "source_match_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "track_id", + "getter_name": "trackId", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "source_id", + "getter_name": "sourceId", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "source_type", + "getter_name": "sourceType", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "Constant(SourceType.youtube.name)", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(SourceType.values)", + "dart_type_name": "SourceType" + } + }, + { + "name": "created_at", + "getter_name": "createdAt", + "moor_type": "dateTime", + "nullable": false, + "customConstraints": null, + "default_dart": "currentDateAndTime", + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 6, + "references": [], + "type": "table", + "data": { + "name": "audio_player_state_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "playing", + "getter_name": "playing", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"playing\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"playing\" IN (0, 1))" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "loop_mode", + "getter_name": "loopMode", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(PlaylistMode.values)", + "dart_type_name": "PlaylistMode" + } + }, + { + "name": "shuffled", + "getter_name": "shuffled", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"shuffled\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"shuffled\" IN (0, 1))" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "collections", + "getter_name": "collections", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const StringListConverter()", + "dart_type_name": "List" + } + }, + { + "name": "tracks", + "getter_name": "tracks", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(\"[]\")", + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const SpotubeTrackObjectListConverter()", + "dart_type_name": "List" + } + }, + { + "name": "current_index", + "getter_name": "currentIndex", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant(0)", + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 7, + "references": [], + "type": "table", + "data": { + "name": "history_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "created_at", + "getter_name": "createdAt", + "moor_type": "dateTime", + "nullable": false, + "customConstraints": null, + "default_dart": "currentDateAndTime", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "type", + "getter_name": "type", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const EnumNameConverter(HistoryEntryType.values)", + "dart_type_name": "HistoryEntryType" + } + }, + { + "name": "item_id", + "getter_name": "itemId", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "data", + "getter_name": "data", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const MapTypeConverter()", + "dart_type_name": "Map" + } + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 8, + "references": [], + "type": "table", + "data": { + "name": "lyrics_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "track_id", + "getter_name": "trackId", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "data", + "getter_name": "data", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "SubtitleTypeConverter()", + "dart_type_name": "SubtitleSimple" + } + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 9, + "references": [], + "type": "table", + "data": { + "name": "metadata_plugins_table", + "was_declared_in_moor": false, + "columns": [ + { + "name": "id", + "getter_name": "id", + "moor_type": "int", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "PRIMARY KEY AUTOINCREMENT", + "dialectAwareDefaultConstraints": { + "sqlite": "PRIMARY KEY AUTOINCREMENT" + }, + "default_dart": null, + "default_client_dart": null, + "dsl_features": ["auto-increment"] + }, + { + "name": "name", + "getter_name": "name", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [{ "allowed-lengths": { "min": 1, "max": 50 } }] + }, + { + "name": "description", + "getter_name": "description", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "version", + "getter_name": "version", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "author", + "getter_name": "author", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "entry_point", + "getter_name": "entryPoint", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "apis", + "getter_name": "apis", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const StringListConverter()", + "dart_type_name": "List" + } + }, + { + "name": "abilities", + "getter_name": "abilities", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [], + "type_converter": { + "dart_expr": "const StringListConverter()", + "dart_type_name": "List" + } + }, + { + "name": "selected", + "getter_name": "selected", + "moor_type": "bool", + "nullable": false, + "customConstraints": null, + "defaultConstraints": "CHECK (\"selected\" IN (0, 1))", + "dialectAwareDefaultConstraints": { + "sqlite": "CHECK (\"selected\" IN (0, 1))" + }, + "default_dart": "const Constant(false)", + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "repository", + "getter_name": "repository", + "moor_type": "string", + "nullable": true, + "customConstraints": null, + "default_dart": null, + "default_client_dart": null, + "dsl_features": [] + }, + { + "name": "plugin_api_version", + "getter_name": "pluginApiVersion", + "moor_type": "string", + "nullable": false, + "customConstraints": null, + "default_dart": "const Constant('1.0.0')", + "default_client_dart": null, + "dsl_features": [] + } + ], + "is_virtual": false, + "without_rowid": false, + "constraints": [] + } + }, + { + "id": 10, + "references": [1], + "type": "index", + "data": { + "on": 1, + "name": "unique_blacklist", + "sql": null, + "unique": true, + "columns": ["element_type", "element_id"] + } + }, + { + "id": 11, + "references": [5], + "type": "index", + "data": { + "on": 5, + "name": "uniq_track_match", + "sql": null, + "unique": true, + "columns": ["track_id", "source_id", "source_type"] + } + } + ] +} diff --git a/drift_schemas/app_db/drift_schema_v9.json b/drift_schemas/app_db/drift_schema_v9.json new file mode 100644 index 00000000..73af2588 --- /dev/null +++ b/drift_schemas/app_db/drift_schema_v9.json @@ -0,0 +1 @@ +{"_meta":{"description":"This file contains a serialized version of schema entities for drift.","version":"1.2.0"},"options":{"store_date_time_values_as_text":false},"entities":[{"id":0,"references":[],"type":"table","data":{"name":"authentication_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"cookie","getter_name":"cookie","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"access_token","getter_name":"accessToken","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}},{"name":"expiration","getter_name":"expiration","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":1,"references":[],"type":"table","data":{"name":"blacklist_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"element_type","getter_name":"elementType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(BlacklistedType.values)","dart_type_name":"BlacklistedType"}},{"name":"element_id","getter_name":"elementId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":2,"references":[],"type":"table","data":{"name":"preferences_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"audio_quality","getter_name":"audioQuality","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceQualities.high.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceQualities.values)","dart_type_name":"SourceQualities"}},{"name":"album_color_sync","getter_name":"albumColorSync","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"album_color_sync\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"album_color_sync\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"amoled_dark_theme","getter_name":"amoledDarkTheme","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"amoled_dark_theme\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"amoled_dark_theme\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"check_update","getter_name":"checkUpdate","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"check_update\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"check_update\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"normalize_audio","getter_name":"normalizeAudio","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"normalize_audio\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"normalize_audio\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"show_system_tray_icon","getter_name":"showSystemTrayIcon","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"show_system_tray_icon\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"show_system_tray_icon\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"system_title_bar","getter_name":"systemTitleBar","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"system_title_bar\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"system_title_bar\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"skip_non_music","getter_name":"skipNonMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"skip_non_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"skip_non_music\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"close_behavior","getter_name":"closeBehavior","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(CloseBehavior.close.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(CloseBehavior.values)","dart_type_name":"CloseBehavior"}},{"name":"accent_color_scheme","getter_name":"accentColorScheme","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"Slate:0xff64748b\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeColorConverter()","dart_type_name":"SpotubeColor"}},{"name":"layout_mode","getter_name":"layoutMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(LayoutMode.adaptive.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(LayoutMode.values)","dart_type_name":"LayoutMode"}},{"name":"locale","getter_name":"locale","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('{\"languageCode\":\"system\",\"countryCode\":\"system\"}')","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const LocaleConverter()","dart_type_name":"Locale"}},{"name":"market","getter_name":"market","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(Market.US.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(Market.values)","dart_type_name":"Market"}},{"name":"search_mode","getter_name":"searchMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SearchMode.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SearchMode.values)","dart_type_name":"SearchMode"}},{"name":"download_location","getter_name":"downloadLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[]},{"name":"local_library_location","getter_name":"localLibraryLocation","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"piped_instance","getter_name":"pipedInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://pipedapi.kavin.rocks\")","default_client_dart":null,"dsl_features":[]},{"name":"invidious_instance","getter_name":"invidiousInstance","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"https://inv.nadeko.net\")","default_client_dart":null,"dsl_features":[]},{"name":"theme_mode","getter_name":"themeMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(ThemeMode.system.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(ThemeMode.values)","dart_type_name":"ThemeMode"}},{"name":"audio_source","getter_name":"audioSource","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(AudioSource.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(AudioSource.values)","dart_type_name":"AudioSource"}},{"name":"youtube_client_engine","getter_name":"youtubeClientEngine","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(YoutubeClientEngine.youtubeExplode.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(YoutubeClientEngine.values)","dart_type_name":"YoutubeClientEngine"}},{"name":"stream_music_codec","getter_name":"streamMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.weba.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"download_music_codec","getter_name":"downloadMusicCodec","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceCodecs.m4a.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceCodecs.values)","dart_type_name":"SourceCodecs"}},{"name":"discord_presence","getter_name":"discordPresence","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"discord_presence\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"discord_presence\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"endless_playback","getter_name":"endlessPlayback","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"endless_playback\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"endless_playback\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]},{"name":"enable_connect","getter_name":"enableConnect","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"enable_connect\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"enable_connect\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"connect_port","getter_name":"connectPort","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(-1)","default_client_dart":null,"dsl_features":[]},{"name":"cache_music","getter_name":"cacheMusic","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"cache_music\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"cache_music\" IN (0, 1))"},"default_dart":"const Constant(true)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":3,"references":[],"type":"table","data":{"name":"scrobbler_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"username","getter_name":"username","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"password_hash","getter_name":"passwordHash","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"EncryptedTextConverter()","dart_type_name":"DecryptedText"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":4,"references":[],"type":"table","data":{"name":"skip_segment_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"start","getter_name":"start","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"end","getter_name":"end","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":5,"references":[],"type":"table","data":{"name":"source_match_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_id","getter_name":"sourceId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"source_type","getter_name":"sourceType","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"Constant(SourceType.youtube.name)","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(SourceType.values)","dart_type_name":"SourceType"}},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":6,"references":[],"type":"table","data":{"name":"audio_player_state_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"playing","getter_name":"playing","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"playing\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"playing\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"loop_mode","getter_name":"loopMode","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(PlaylistMode.values)","dart_type_name":"PlaylistMode"}},{"name":"shuffled","getter_name":"shuffled","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"shuffled\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"shuffled\" IN (0, 1))"},"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"collections","getter_name":"collections","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"tracks","getter_name":"tracks","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant(\"[]\")","default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const SpotubeTrackObjectListConverter()","dart_type_name":"List"}},{"name":"current_index","getter_name":"currentIndex","moor_type":"int","nullable":false,"customConstraints":null,"default_dart":"const Constant(0)","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":7,"references":[],"type":"table","data":{"name":"history_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"created_at","getter_name":"createdAt","moor_type":"dateTime","nullable":false,"customConstraints":null,"default_dart":"currentDateAndTime","default_client_dart":null,"dsl_features":[]},{"name":"type","getter_name":"type","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const EnumNameConverter(HistoryEntryType.values)","dart_type_name":"HistoryEntryType"}},{"name":"item_id","getter_name":"itemId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const MapTypeConverter()","dart_type_name":"Map"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":8,"references":[],"type":"table","data":{"name":"lyrics_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"track_id","getter_name":"trackId","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"data","getter_name":"data","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"SubtitleTypeConverter()","dart_type_name":"SubtitleSimple"}}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":9,"references":[],"type":"table","data":{"name":"plugins_table","was_declared_in_moor":false,"columns":[{"name":"id","getter_name":"id","moor_type":"int","nullable":false,"customConstraints":null,"defaultConstraints":"PRIMARY KEY AUTOINCREMENT","dialectAwareDefaultConstraints":{"sqlite":"PRIMARY KEY AUTOINCREMENT"},"default_dart":null,"default_client_dart":null,"dsl_features":["auto-increment"]},{"name":"name","getter_name":"name","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[{"allowed-lengths":{"min":1,"max":50}}]},{"name":"description","getter_name":"description","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"version","getter_name":"version","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"author","getter_name":"author","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"entry_point","getter_name":"entryPoint","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"apis","getter_name":"apis","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"abilities","getter_name":"abilities","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[],"type_converter":{"dart_expr":"const StringListConverter()","dart_type_name":"List"}},{"name":"selected_for_metadata","getter_name":"selectedForMetadata","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"selected_for_metadata\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"selected_for_metadata\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"selected_for_audio_source","getter_name":"selectedForAudioSource","moor_type":"bool","nullable":false,"customConstraints":null,"defaultConstraints":"CHECK (\"selected_for_audio_source\" IN (0, 1))","dialectAwareDefaultConstraints":{"sqlite":"CHECK (\"selected_for_audio_source\" IN (0, 1))"},"default_dart":"const Constant(false)","default_client_dart":null,"dsl_features":[]},{"name":"repository","getter_name":"repository","moor_type":"string","nullable":true,"customConstraints":null,"default_dart":null,"default_client_dart":null,"dsl_features":[]},{"name":"plugin_api_version","getter_name":"pluginApiVersion","moor_type":"string","nullable":false,"customConstraints":null,"default_dart":"const Constant('2.0.0')","default_client_dart":null,"dsl_features":[]}],"is_virtual":false,"without_rowid":false,"constraints":[]}},{"id":10,"references":[1],"type":"index","data":{"on":1,"name":"unique_blacklist","sql":null,"unique":true,"columns":["element_type","element_id"]}},{"id":11,"references":[5],"type":"index","data":{"on":5,"name":"uniq_track_match","sql":null,"unique":true,"columns":["track_id","source_id","source_type"]}}]} \ No newline at end of file diff --git a/flutter_launcher_icons-nightly.yaml b/flutter_launcher_icons-nightly.yaml index c6892d4b..9e4e805c 100644 --- a/flutter_launcher_icons-nightly.yaml +++ b/flutter_launcher_icons-nightly.yaml @@ -1,6 +1,6 @@ flutter_launcher_icons: android: true ios: true - image_path: "assets/spotube-nightly-logo.png" - adaptive_icon_foreground: "assets/spotube-nightly-logo-foreground.jpg" + image_path: "assets/branding/spotube-nightly-logo.png" + adaptive_icon_foreground: "assets/branding/spotube-nightly-logo-foreground.png" adaptive_icon_background: "#242832" diff --git a/flutter_launcher_icons-stable.yaml b/flutter_launcher_icons-stable.yaml deleted file mode 100644 index 0d205cbd..00000000 --- a/flutter_launcher_icons-stable.yaml +++ /dev/null @@ -1,13 +0,0 @@ -flutter_launcher_icons: - ios: true - android: true - image_path: "assets/spotube-logo.png" - adaptive_icon_foreground: "assets/spotube-logo-foreground.jpg" - adaptive_icon_background: "#242832" - windows: - generate: true - image_path: "assets/spotube-logo.png" - icon_size: 48 # min:48, max:256, default: 48 - macos: - generate: true - image_path: "assets/spotube-logo-macos.png" diff --git a/flutter_launcher_icons.yaml b/flutter_launcher_icons.yaml new file mode 100644 index 00000000..e5b26882 --- /dev/null +++ b/flutter_launcher_icons.yaml @@ -0,0 +1,29 @@ +# flutter pub run flutter_launcher_icons +flutter_launcher_icons: + image_path: "assets/branding/spotube-logo.png" + + android: true + # image_path_android: "assets/branding/icon/icon.png" + min_sdk_android: 21 # android min sdk min:16, default 21 + adaptive_icon_background: "#242832" + adaptive_icon_foreground: "assets/branding/spotube-logo-foreground.png" + # adaptive_icon_monochrome: "assets/branding/icon/monochrome.png" + + ios: true + # image_path_ios: "assets/branding/icon/icon.png" + remove_alpha_channel_ios: true + # image_path_ios_dark_transparent: "assets/branding/icon/icon_dark.png" + # image_path_ios_tinted_grayscale: "assets/branding/icon/icon_tinted.png" + # desaturate_tinted_to_grayscale_ios: true + + web: + generate: false + + windows: + generate: true + image_path: "assets/branding/spotube-logo.png" + icon_size: 48 # min:48, max:256, default: 48 + + macos: + generate: true + image_path: "assets/branding/spotube-logo-macos.png" diff --git a/flutter_native_splash-nightly.yaml b/flutter_native_splash-nightly.yaml index 37da37d9..3b7daeec 100644 --- a/flutter_native_splash-nightly.yaml +++ b/flutter_native_splash-nightly.yaml @@ -1,9 +1,9 @@ flutter_native_splash: - background_image: assets/bengali-patterns-bg.jpg - image: assets/spotube-nightly-logo.png - branding: assets/branding.png + background_image: assets/images/bengali-patterns-bg.jpg + image: assets/branding/spotube-nightly-logo.png + branding: assets/branding/branding.png android_12: - image: assets/spotube-nightly-logo_android12.png - branding: assets/branding.png + image: assets/branding/spotube-nightly-logo_android12.png + branding: assets/branding/branding.png color: "#000000" icon_background_color: "#000000" diff --git a/ios/HomePlayerWidget/Assets.xcassets/AccentColor.colorset/Contents.json b/ios/HomePlayerWidget/Assets.xcassets/AccentColor.colorset/Contents.json new file mode 100644 index 00000000..eb878970 --- /dev/null +++ b/ios/HomePlayerWidget/Assets.xcassets/AccentColor.colorset/Contents.json @@ -0,0 +1,11 @@ +{ + "colors" : [ + { + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/ios/HomePlayerWidget/Assets.xcassets/AppIcon.appiconset/Contents.json b/ios/HomePlayerWidget/Assets.xcassets/AppIcon.appiconset/Contents.json new file mode 100644 index 00000000..23058801 --- /dev/null +++ b/ios/HomePlayerWidget/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -0,0 +1,35 @@ +{ + "images" : [ + { + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "tinted" + } + ], + "idiom" : "universal", + "platform" : "ios", + "size" : "1024x1024" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/ios/HomePlayerWidget/Assets.xcassets/Contents.json b/ios/HomePlayerWidget/Assets.xcassets/Contents.json new file mode 100644 index 00000000..73c00596 --- /dev/null +++ b/ios/HomePlayerWidget/Assets.xcassets/Contents.json @@ -0,0 +1,6 @@ +{ + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/ios/HomePlayerWidget/Assets.xcassets/WidgetBackground.colorset/Contents.json b/ios/HomePlayerWidget/Assets.xcassets/WidgetBackground.colorset/Contents.json new file mode 100644 index 00000000..eb878970 --- /dev/null +++ b/ios/HomePlayerWidget/Assets.xcassets/WidgetBackground.colorset/Contents.json @@ -0,0 +1,11 @@ +{ + "colors" : [ + { + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/ios/HomePlayerWidget/HomePlayerWidget.swift b/ios/HomePlayerWidget/HomePlayerWidget.swift new file mode 100644 index 00000000..8808aae1 --- /dev/null +++ b/ios/HomePlayerWidget/HomePlayerWidget.swift @@ -0,0 +1,86 @@ +// +// HomePlayerWidget.swift +// HomePlayerWidget +// +// Created by Kingkor Roy Tirtho on 15/12/24. +// + +import WidgetKit +import SwiftUI + +private let widgetGroupId = "group.spotube_home_player_widget" + +struct Provider: TimelineProvider { + func placeholder(in context: Context) -> SimpleEntry { + SimpleEntry(date: Date(), emoji: "😀") + } + + func getSnapshot(in context: Context, completion: @escaping (SimpleEntry) -> ()) { + let entry = SimpleEntry(date: Date(), emoji: "😀") + completion(entry) + } + + func getTimeline(in context: Context, completion: @escaping (Timeline) -> ()) { + var entries: [SimpleEntry] = [] + + // Generate a timeline consisting of five entries an hour apart, starting from the current date. + let currentDate = Date() + for hourOffset in 0 ..< 5 { + let entryDate = Calendar.current.date(byAdding: .hour, value: hourOffset, to: currentDate)! + let entry = SimpleEntry(date: entryDate, emoji: "😀") + entries.append(entry) + } + + let timeline = Timeline(entries: entries, policy: .atEnd) + completion(timeline) + } + +// func relevances() async -> WidgetRelevances { +// // Generate a list containing the contexts this widget is relevant in. +// } +} + +struct SimpleEntry: TimelineEntry { + let date: Date + let emoji: String +} + +struct HomePlayerWidgetEntryView : View { + var entry: Provider.Entry + + var body: some View { + VStack { + Text("Time:") + Text(entry.date, style: .time) + + Text("Emoji:") + Text(entry.emoji) + } + } +} + +struct HomePlayerWidget: Widget { + let kind: String = "HomePlayerWidget" + + var body: some WidgetConfiguration { + StaticConfiguration(kind: kind, provider: Provider()) { entry in + if #available(iOS 17.0, *) { + HomePlayerWidgetEntryView(entry: entry) + .containerBackground(.fill.tertiary, for: .widget) + } else { + HomePlayerWidgetEntryView(entry: entry) + .padding() + .background() + } + } + .configurationDisplayName("My Widget") + .description("This is an example widget.") + } +} + +#Preview(as: .systemSmall) { + HomePlayerWidget() +} timeline: { + SimpleEntry(date: .now, emoji: "😀") + SimpleEntry(date: .now, emoji: "🤩") +} diff --git a/ios/HomePlayerWidget/HomePlayerWidgetBundle.swift b/ios/HomePlayerWidget/HomePlayerWidgetBundle.swift new file mode 100644 index 00000000..68158b53 --- /dev/null +++ b/ios/HomePlayerWidget/HomePlayerWidgetBundle.swift @@ -0,0 +1,16 @@ +// +// HomePlayerWidgetBundle.swift +// HomePlayerWidget +// +// Created by Kingkor Roy Tirtho on 15/12/24. +// + +import WidgetKit +import SwiftUI + +@main +struct HomePlayerWidgetBundle: WidgetBundle { + var body: some Widget { + HomePlayerWidget() + } +} diff --git a/ios/HomePlayerWidget/Info.plist b/ios/HomePlayerWidget/Info.plist new file mode 100644 index 00000000..0f118fb7 --- /dev/null +++ b/ios/HomePlayerWidget/Info.plist @@ -0,0 +1,11 @@ + + + + + NSExtension + + NSExtensionPointIdentifier + com.apple.widgetkit-extension + + + diff --git a/ios/HomePlayerWidgetExtension.entitlements b/ios/HomePlayerWidgetExtension.entitlements new file mode 100644 index 00000000..58165678 --- /dev/null +++ b/ios/HomePlayerWidgetExtension.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.spotube_home_player_widget + + + diff --git a/ios/Podfile.lock b/ios/Podfile.lock index 104ff767..2ff415a0 100644 --- a/ios/Podfile.lock +++ b/ios/Podfile.lock @@ -1,13 +1,16 @@ PODS: - - app_links (0.0.2): + - app_links (6.4.1): - Flutter - audio_service (0.0.1): - Flutter + - FlutterMacOS - audio_session (0.0.1): - Flutter - bonsoir_darwin (0.0.1): - Flutter - FlutterMacOS + - connectivity_plus (0.0.1): + - Flutter - device_info_plus (0.0.1): - Flutter - DKImagePickerController/Core (4.3.4): @@ -46,6 +49,8 @@ PODS: - Flutter - file_selector_ios (0.0.1): - Flutter + - fk_user_agent (2.0.0): + - Flutter - Flutter (1.0.0) - flutter_broadcasts (0.0.1): - Flutter @@ -64,13 +69,17 @@ PODS: - Flutter - flutter_sharing_intent (0.0.1): - Flutter + - flutter_timezone (0.0.1): + - Flutter + - home_widget (0.0.1): + - Flutter - image_picker_ios (0.0.1): - Flutter - integration_test (0.0.1): - Flutter - - media_kit_libs_ios_audio (1.0.4): + - irondash_engine_context (0.0.1): - Flutter - - media_kit_native_event_loop (1.0.0): + - media_kit_libs_ios_audio (1.0.4): - Flutter - metadata_god (0.0.1): - Flutter @@ -93,36 +102,49 @@ PODS: - sqflite_darwin (0.0.4): - Flutter - FlutterMacOS - - sqlite3 (3.47.1): - - sqlite3/common (= 3.47.1) - - sqlite3/common (3.47.1) - - sqlite3/dbstatvtab (3.47.1): + - sqlite3 (3.50.4): + - sqlite3/common (= 3.50.4) + - sqlite3/common (3.50.4) + - sqlite3/dbstatvtab (3.50.4): - sqlite3/common - - sqlite3/fts5 (3.47.1): + - sqlite3/fts5 (3.50.4): - sqlite3/common - - sqlite3/perf-threadsafe (3.47.1): + - sqlite3/math (3.50.4): - sqlite3/common - - sqlite3/rtree (3.47.1): + - sqlite3/perf-threadsafe (3.50.4): + - sqlite3/common + - sqlite3/rtree (3.50.4): + - sqlite3/common + - sqlite3/session (3.50.4): - sqlite3/common - sqlite3_flutter_libs (0.0.1): - Flutter - - sqlite3 (~> 3.47.0) + - FlutterMacOS + - sqlite3 (~> 3.50.4) - sqlite3/dbstatvtab - sqlite3/fts5 + - sqlite3/math - sqlite3/perf-threadsafe - sqlite3/rtree + - sqlite3/session + - super_native_extensions (0.0.1): + - Flutter - SwiftyGif (5.4.4) + - system_theme (0.0.1): + - Flutter - url_launcher_ios (0.0.1): - Flutter DEPENDENCIES: - app_links (from `.symlinks/plugins/app_links/ios`) - - audio_service (from `.symlinks/plugins/audio_service/ios`) + - audio_service (from `.symlinks/plugins/audio_service/darwin`) - audio_session (from `.symlinks/plugins/audio_session/ios`) - bonsoir_darwin (from `.symlinks/plugins/bonsoir_darwin/darwin`) + - connectivity_plus (from `.symlinks/plugins/connectivity_plus/ios`) - device_info_plus (from `.symlinks/plugins/device_info_plus/ios`) - file_picker (from `.symlinks/plugins/file_picker/ios`) - file_selector_ios (from `.symlinks/plugins/file_selector_ios/ios`) + - fk_user_agent (from `.symlinks/plugins/fk_user_agent/ios`) - Flutter (from `Flutter`) - flutter_broadcasts (from `.symlinks/plugins/flutter_broadcasts/ios`) - flutter_discord_rpc (from `.symlinks/plugins/flutter_discord_rpc/ios`) @@ -130,10 +152,12 @@ DEPENDENCIES: - flutter_native_splash (from `.symlinks/plugins/flutter_native_splash/ios`) - flutter_secure_storage (from `.symlinks/plugins/flutter_secure_storage/ios`) - flutter_sharing_intent (from `.symlinks/plugins/flutter_sharing_intent/ios`) + - flutter_timezone (from `.symlinks/plugins/flutter_timezone/ios`) + - home_widget (from `.symlinks/plugins/home_widget/ios`) - image_picker_ios (from `.symlinks/plugins/image_picker_ios/ios`) - integration_test (from `.symlinks/plugins/integration_test/ios`) + - irondash_engine_context (from `.symlinks/plugins/irondash_engine_context/ios`) - media_kit_libs_ios_audio (from `.symlinks/plugins/media_kit_libs_ios_audio/ios`) - - media_kit_native_event_loop (from `.symlinks/plugins/media_kit_native_event_loop/ios`) - metadata_god (from `.symlinks/plugins/metadata_god/ios`) - open_file_ios (from `.symlinks/plugins/open_file_ios/ios`) - package_info_plus (from `.symlinks/plugins/package_info_plus/ios`) @@ -141,7 +165,9 @@ DEPENDENCIES: - permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`) - shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`) - sqflite_darwin (from `.symlinks/plugins/sqflite_darwin/darwin`) - - sqlite3_flutter_libs (from `.symlinks/plugins/sqlite3_flutter_libs/ios`) + - sqlite3_flutter_libs (from `.symlinks/plugins/sqlite3_flutter_libs/darwin`) + - super_native_extensions (from `.symlinks/plugins/super_native_extensions/ios`) + - system_theme (from `.symlinks/plugins/system_theme/ios`) - url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) SPEC REPOS: @@ -157,17 +183,21 @@ EXTERNAL SOURCES: app_links: :path: ".symlinks/plugins/app_links/ios" audio_service: - :path: ".symlinks/plugins/audio_service/ios" + :path: ".symlinks/plugins/audio_service/darwin" audio_session: :path: ".symlinks/plugins/audio_session/ios" bonsoir_darwin: :path: ".symlinks/plugins/bonsoir_darwin/darwin" + connectivity_plus: + :path: ".symlinks/plugins/connectivity_plus/ios" device_info_plus: :path: ".symlinks/plugins/device_info_plus/ios" file_picker: :path: ".symlinks/plugins/file_picker/ios" file_selector_ios: :path: ".symlinks/plugins/file_selector_ios/ios" + fk_user_agent: + :path: ".symlinks/plugins/fk_user_agent/ios" Flutter: :path: Flutter flutter_broadcasts: @@ -182,14 +212,18 @@ EXTERNAL SOURCES: :path: ".symlinks/plugins/flutter_secure_storage/ios" flutter_sharing_intent: :path: ".symlinks/plugins/flutter_sharing_intent/ios" + flutter_timezone: + :path: ".symlinks/plugins/flutter_timezone/ios" + home_widget: + :path: ".symlinks/plugins/home_widget/ios" image_picker_ios: :path: ".symlinks/plugins/image_picker_ios/ios" integration_test: :path: ".symlinks/plugins/integration_test/ios" + irondash_engine_context: + :path: ".symlinks/plugins/irondash_engine_context/ios" media_kit_libs_ios_audio: :path: ".symlinks/plugins/media_kit_libs_ios_audio/ios" - media_kit_native_event_loop: - :path: ".symlinks/plugins/media_kit_native_event_loop/ios" metadata_god: :path: ".symlinks/plugins/metadata_god/ios" open_file_ios: @@ -205,44 +239,54 @@ EXTERNAL SOURCES: sqflite_darwin: :path: ".symlinks/plugins/sqflite_darwin/darwin" sqlite3_flutter_libs: - :path: ".symlinks/plugins/sqlite3_flutter_libs/ios" + :path: ".symlinks/plugins/sqlite3_flutter_libs/darwin" + super_native_extensions: + :path: ".symlinks/plugins/super_native_extensions/ios" + system_theme: + :path: ".symlinks/plugins/system_theme/ios" url_launcher_ios: :path: ".symlinks/plugins/url_launcher_ios/ios" SPEC CHECKSUMS: - app_links: e7a6750a915a9e161c58d91bc610e8cd1d4d0ad0 - audio_service: f509d65da41b9521a61f1c404dd58651f265a567 - audio_session: 088d2483ebd1dc43f51d253d4a1c517d9a2e7207 - bonsoir_darwin: e3b8526c42ca46a885142df84229131dfabea842 - device_info_plus: bf2e3232933866d73fe290f2942f2156cdd10342 + app_links: 3dbc685f76b1693c66a6d9dd1e9ab6f73d97dc0a + audio_service: aa99a6ba2ae7565996015322b0bb024e1d25c6fd + audio_session: 9bb7f6c970f21241b19f5a3658097ae459681ba0 + bonsoir_darwin: 29c7ccf356646118844721f36e1de4b61f6cbd0e + connectivity_plus: cb623214f4e1f6ef8fe7403d580fdad517d2f7dd + device_info_plus: 21fcca2080fbcd348be798aa36c3e5ed849eefbe DKImagePickerController: b512c28220a2b8ac7419f21c491fc8534b7601ac DKPhotoGallery: fdfad5125a9fdda9cc57df834d49df790dbb4179 - file_picker: 09aa5ec1ab24135ccd7a1621c46c84134bfd6655 - file_selector_ios: f0670c1064a8c8450e38145d8043160105d0b97c - Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 - flutter_broadcasts: 3ece15b27d8ccbe2132c3df303e7c3401feab882 - flutter_discord_rpc: e1c342f29ceb9dd76cdc01db59a70c93bb4d9ec5 - flutter_inappwebview_ios: 6f63631e2c62a7c350263b13fa5427aedefe81d4 - flutter_native_splash: e8a1e01082d97a8099d973f919f57904c925008a - flutter_secure_storage: d33dac7ae2ea08509be337e775f6b59f1ff45f12 - flutter_sharing_intent: e35380d0e1501d7111dbb7e46d5ac6339da6da98 - image_picker_ios: c560581cceedb403a6ff17f2f816d7fea1421fc1 - integration_test: 252f60fa39af5e17c3aa9899d35d908a0721b573 - media_kit_libs_ios_audio: 8f39d96a9c630685dfb844c289bd1d114c486fb3 - media_kit_native_event_loop: 99111eded5acbdc9c2738021ea6550dd36ca8837 - metadata_god: 4bbd8523cdb5d42c5e59d2fabad01ff8f4bc53f9 - open_file_ios: 461db5853723763573e140de3193656f91990d9e + file_picker: 9b3292d7c8bc68c8a7bf8eb78f730e49c8efc517 + file_selector_ios: f92e583d43608aebc2e4a18daac30b8902845502 + fk_user_agent: 137145b086229251761678fe034da53753f4ce59 + Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467 + flutter_broadcasts: 7bb7cc1024900a7f85e98b6faab795290b7c2339 + flutter_discord_rpc: 0572e8227ea730c5afe5876a37c08c728ce95f3a + flutter_inappwebview_ios: b89ba3482b96fb25e00c967aae065701b66e9b99 + flutter_native_splash: c32d145d68aeda5502d5f543ee38c192065986cf + flutter_secure_storage: 1ed9476fba7e7a782b22888f956cce43e2c62f13 + flutter_sharing_intent: afdc98985814d2c01d8c0956a177d6b6dfbdc373 + flutter_timezone: 7c838e17ffd4645d261e87037e5bebf6d38fe544 + home_widget: f169fc41fd807b4d46ab6615dc44d62adbf9f64f + image_picker_ios: 7fe1ff8e34c1790d6fff70a32484959f563a928a + integration_test: 4a889634ef21a45d28d50d622cf412dc6d9f586e + irondash_engine_context: 8e58ca8e0212ee9d1c7dc6a42121849986c88486 + media_kit_libs_ios_audio: 905e6323b72e65c63ab9262b2e473f52c024a3a8 + metadata_god: 018b59c2f3617569928550dcbd17481591557c1d + open_file_ios: 5ff7526df64e4394b4fe207636b67a95e83078bb OrderedSet: e539b66b644ff081c73a262d24ad552a69be3a94 - package_info_plus: 58f0028419748fad15bf008b270aaa8e54380b1c - path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46 - permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2 + package_info_plus: 580e9a5f1b6ca5594e7c9ed5f92d1dfb2a66b5e1 + path_provider_foundation: 080d55be775b7414fd5a5ef3ac137b97b097e564 + permission_handler_apple: 4ed2196e43d0651e8ff7ca3483a069d469701f2d SDWebImage: a81bbb3ba4ea5f810f4069c68727cb118467a04a - shared_preferences_foundation: fcdcbc04712aee1108ac7fda236f363274528f78 - sqflite_darwin: 5a7236e3b501866c1c9befc6771dfd73ffb8702d - sqlite3: 1e522f0938463e44b7faf50393b40bdc1e1e456d - sqlite3_flutter_libs: b55ef23cfafea5318ae5081e0bf3fbbce8417c94 + shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7 + sqflite_darwin: 20b2a3a3b70e43edae938624ce550a3cbf66a3d0 + sqlite3: 73513155ec6979715d3904ef53a8d68892d4032b + sqlite3_flutter_libs: 83f8e9f5b6554077f1d93119fe20ebaa5f3a9ef1 + super_native_extensions: b763c02dc3a8fd078389f410bf15149179020cb4 SwiftyGif: 93a1cc87bf3a51916001cf8f3d63835fb64c819f - url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe + system_theme: a94f91f49eeb97cfa768c7d5a9b2f6aa51b00494 + url_launcher_ios: 694010445543906933d732453a59da0a173ae33d PODFILE CHECKSUM: 0659b64ac6e9e96b61d8550decffa8bff51a957e diff --git a/ios/Runner.xcodeproj/project.pbxproj b/ios/Runner.xcodeproj/project.pbxproj index 34793f68..88a40d6f 100644 --- a/ios/Runner.xcodeproj/project.pbxproj +++ b/ios/Runner.xcodeproj/project.pbxproj @@ -36,8 +36,21 @@ B536BDD62B4060B3009B3CE4 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; }; B536BDD72B4060B3009B3CE4 /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; }; C36A05AD330BBFAED75A62D5 /* Pods_dev.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4238A4985255EC9F93067739 /* Pods_dev.framework */; }; + E612EC3B2D0F07A90022720C /* WidgetKit.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = E612EC3A2D0F07A90022720C /* WidgetKit.framework */; }; + E612EC3D2D0F07A90022720C /* SwiftUI.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = E612EC3C2D0F07A90022720C /* SwiftUI.framework */; }; + E612EC482D0F07AD0022720C /* HomePlayerWidgetExtension.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = E612EC392D0F07A90022720C /* HomePlayerWidgetExtension.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; /* End PBXBuildFile section */ +/* Begin PBXContainerItemProxy section */ + E612EC462D0F07AD0022720C /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 97C146E61CF9000F007C117D /* Project object */; + proxyType = 1; + remoteGlobalIDString = E612EC382D0F07A80022720C; + remoteInfo = HomePlayerWidgetExtension; + }; +/* End PBXContainerItemProxy section */ + /* Begin PBXCopyFilesBuildPhase section */ 9705A1C41CF9048500538489 /* Embed Frameworks */ = { isa = PBXCopyFilesBuildPhase; @@ -79,6 +92,17 @@ name = "Embed Frameworks"; runOnlyForDeploymentPostprocessing = 0; }; + E612EC492D0F07AD0022720C /* Embed Foundation Extensions */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 13; + files = ( + E612EC482D0F07AD0022720C /* HomePlayerWidgetExtension.appex in Embed Foundation Extensions */, + ); + name = "Embed Foundation Extensions"; + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXCopyFilesBuildPhase section */ /* Begin PBXFileReference section */ @@ -148,6 +172,14 @@ D32BAE0F55672DD7669755B8 /* Pods-Runner.debug-stable.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug-stable.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug-stable.xcconfig"; sourceTree = ""; }; D9A69004587D01A7C68666CF /* Pods-dev.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-dev.release.xcconfig"; path = "Target Support Files/Pods-dev/Pods-dev.release.xcconfig"; sourceTree = ""; }; E0EAB4380EE7C7EA7A350B6F /* Pods-stable.release-nightly.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-stable.release-nightly.xcconfig"; path = "Target Support Files/Pods-stable/Pods-stable.release-nightly.xcconfig"; sourceTree = ""; }; + E612EC392D0F07A90022720C /* HomePlayerWidgetExtension.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = HomePlayerWidgetExtension.appex; sourceTree = BUILT_PRODUCTS_DIR; }; + E612EC3A2D0F07A90022720C /* WidgetKit.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = WidgetKit.framework; path = System/Library/Frameworks/WidgetKit.framework; sourceTree = SDKROOT; }; + E612EC3C2D0F07A90022720C /* SwiftUI.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = SwiftUI.framework; path = System/Library/Frameworks/SwiftUI.framework; sourceTree = SDKROOT; }; + E6F17DB92D0F34E500BC2FA2 /* HomePlayerWidgetExtension.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = HomePlayerWidgetExtension.entitlements; sourceTree = ""; }; + E6F17DBA2D0F352C00BC2FA2 /* Runner.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = Runner.entitlements; sourceTree = ""; }; + E6F17DBB2D0F356700BC2FA2 /* stable.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = stable.entitlements; sourceTree = ""; }; + E6F17DBC2D0F357500BC2FA2 /* dev.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = dev.entitlements; sourceTree = ""; }; + E6F17DBD2D0F357F00BC2FA2 /* nightly.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = nightly.entitlements; sourceTree = ""; }; E81F11471FD7D807286E33D6 /* Pods-dev.debug-dev.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-dev.debug-dev.xcconfig"; path = "Target Support Files/Pods-dev/Pods-dev.debug-dev.xcconfig"; sourceTree = ""; }; EB7783C1029CEC13F4B05D36 /* Pods-nightly.debug-nightly.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-nightly.debug-nightly.xcconfig"; path = "Target Support Files/Pods-nightly/Pods-nightly.debug-nightly.xcconfig"; sourceTree = ""; }; EBBED0A8DE0D0E230CD03613 /* Pods-dev.release-stable.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-dev.release-stable.xcconfig"; path = "Target Support Files/Pods-dev/Pods-dev.release-stable.xcconfig"; sourceTree = ""; }; @@ -155,6 +187,20 @@ F7E9EBDD27997A73A4D38EE1 /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; }; /* End PBXFileReference section */ +/* Begin PBXFileSystemSynchronizedBuildFileExceptionSet section */ + E612EC562D0F07AD0022720C /* PBXFileSystemSynchronizedBuildFileExceptionSet */ = { + isa = PBXFileSystemSynchronizedBuildFileExceptionSet; + membershipExceptions = ( + Info.plist, + ); + target = E612EC382D0F07A80022720C /* HomePlayerWidgetExtension */; + }; +/* End PBXFileSystemSynchronizedBuildFileExceptionSet section */ + +/* Begin PBXFileSystemSynchronizedRootGroup section */ + E612EC3E2D0F07A90022720C /* HomePlayerWidget */ = {isa = PBXFileSystemSynchronizedRootGroup; exceptions = (E612EC562D0F07AD0022720C /* PBXFileSystemSynchronizedBuildFileExceptionSet */, ); explicitFileTypes = {}; explicitFolders = (); path = HomePlayerWidget; sourceTree = ""; }; +/* End PBXFileSystemSynchronizedRootGroup section */ + /* Begin PBXFrameworksBuildPhase section */ 97C146EB1CF9000F007C117D /* Frameworks */ = { isa = PBXFrameworksBuildPhase; @@ -189,6 +235,15 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + E612EC362D0F07A80022720C /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + E612EC3D2D0F07A90022720C /* SwiftUI.framework in Frameworks */, + E612EC3B2D0F07A90022720C /* WidgetKit.framework in Frameworks */, + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXFrameworksBuildPhase section */ /* Begin PBXGroup section */ @@ -199,6 +254,8 @@ 4238A4985255EC9F93067739 /* Pods_dev.framework */, BAC36FC304DBD4E8A8C00694 /* Pods_nightly.framework */, B5F91A319C771EEC978B238A /* Pods_stable.framework */, + E612EC3A2D0F07A90022720C /* WidgetKit.framework */, + E612EC3C2D0F07A90022720C /* SwiftUI.framework */, ); name = Frameworks; sourceTree = ""; @@ -272,8 +329,13 @@ 97C146E51CF9000F007C117D = { isa = PBXGroup; children = ( + E6F17DBD2D0F357F00BC2FA2 /* nightly.entitlements */, + E6F17DBC2D0F357500BC2FA2 /* dev.entitlements */, + E6F17DBB2D0F356700BC2FA2 /* stable.entitlements */, + E6F17DB92D0F34E500BC2FA2 /* HomePlayerWidgetExtension.entitlements */, 9740EEB11CF90186004384FC /* Flutter */, 97C146F01CF9000F007C117D /* Runner */, + E612EC3E2D0F07A90022720C /* HomePlayerWidget */, 97C146EF1CF9000F007C117D /* Products */, 67CBFE209DF24C94A9837AD5 /* Pods */, 0E0B839C4E103F896209E822 /* Frameworks */, @@ -290,6 +352,7 @@ B536BDA02B405DB1009B3CE4 /* stable.app */, B536BDBF2B405FDE009B3CE4 /* dev.app */, B536BDE42B4060B3009B3CE4 /* nightly.app */, + E612EC392D0F07A90022720C /* HomePlayerWidgetExtension.appex */, ); name = Products; sourceTree = ""; @@ -297,6 +360,7 @@ 97C146F01CF9000F007C117D /* Runner */ = { isa = PBXGroup; children = ( + E6F17DBA2D0F352C00BC2FA2 /* Runner.entitlements */, 97C146FA1CF9000F007C117D /* Main.storyboard */, 97C146FD1CF9000F007C117D /* Assets.xcassets */, 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */, @@ -325,10 +389,13 @@ 3B06AD1E1E4923F5004D2608 /* Thin Binary */, 6E9FEF583EA597C8B76255B2 /* [CP] Embed Pods Frameworks */, 46F6EB27C31C41D86428A28B /* [CP] Copy Pods Resources */, + E612EC492D0F07AD0022720C /* Embed Foundation Extensions */, + E63F9CBC2D10709D00CD9E72 /* ShellScript */, ); buildRules = ( ); dependencies = ( + E612EC472D0F07AD0022720C /* PBXTargetDependency */, ); name = Runner; productName = Runner; @@ -404,12 +471,35 @@ productReference = B536BDE42B4060B3009B3CE4 /* nightly.app */; productType = "com.apple.product-type.application"; }; + E612EC382D0F07A80022720C /* HomePlayerWidgetExtension */ = { + isa = PBXNativeTarget; + buildConfigurationList = E612EC572D0F07AD0022720C /* Build configuration list for PBXNativeTarget "HomePlayerWidgetExtension" */; + buildPhases = ( + E612EC352D0F07A80022720C /* Sources */, + E612EC362D0F07A80022720C /* Frameworks */, + E612EC372D0F07A80022720C /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + ); + fileSystemSynchronizedGroups = ( + E612EC3E2D0F07A90022720C /* HomePlayerWidget */, + ); + name = HomePlayerWidgetExtension; + packageProductDependencies = ( + ); + productName = HomePlayerWidgetExtension; + productReference = E612EC392D0F07A90022720C /* HomePlayerWidgetExtension.appex */; + productType = "com.apple.product-type.app-extension"; + }; /* End PBXNativeTarget section */ /* Begin PBXProject section */ 97C146E61CF9000F007C117D /* Project object */ = { isa = PBXProject; attributes = { + LastSwiftUpdateCheck = 1620; LastUpgradeCheck = 1510; ORGANIZATIONNAME = ""; TargetAttributes = { @@ -417,6 +507,9 @@ CreatedOnToolsVersion = 7.3.1; LastSwiftMigration = 1100; }; + E612EC382D0F07A80022720C = { + CreatedOnToolsVersion = 16.2; + }; }; }; buildConfigurationList = 97C146E91CF9000F007C117D /* Build configuration list for PBXProject "Runner" */; @@ -436,6 +529,7 @@ B536BD8C2B405DB1009B3CE4 /* stable */, B536BDAB2B405FDE009B3CE4 /* dev */, B536BDCD2B4060B3009B3CE4 /* nightly */, + E612EC382D0F07A80022720C /* HomePlayerWidgetExtension */, ); }; /* End PBXProject section */ @@ -485,6 +579,13 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + E612EC372D0F07A80022720C /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXResourcesBuildPhase section */ /* Begin PBXShellScriptBuildPhase section */ @@ -685,7 +786,7 @@ ); runOnlyForDeploymentPostprocessing = 0; shellPath = /bin/sh; - shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; + shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build\n"; }; A6D446F111DE4C4A202BE7F7 /* [CP] Embed Pods Frameworks */ = { isa = PBXShellScriptBuildPhase; @@ -814,6 +915,23 @@ shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-nightly/Pods-nightly-frameworks.sh\"\n"; showEnvVarsInLog = 0; }; + E63F9CBC2D10709D00CD9E72 /* ShellScript */ = { + isa = PBXShellScriptBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + ); + inputPaths = ( + ); + outputFileListPaths = ( + ); + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "# Type a script or drag a sgeneratedPath=\"$SRCROOT/Flutter/Generated.xcconfig\"\n\n# Read and trim versionNumber and buildNumber\nversionNumber=$(grep FLUTTER_BUILD_NAME \"$generatedPath\" | cut -d '=' -f2 | xargs)\nbuildNumber=$(grep FLUTTER_BUILD_NUMBER \"$generatedPath\" | cut -d '=' -f2 | xargs)\n\ninfoPlistPath=\"$SRCROOT/HomePlayerWidget/Info.plist\"\n\n# Check and add CFBundleVersion if it does not exist\n/usr/libexec/PlistBuddy -c \"Print :CFBundleVersion\" \"$infoPlistPath\" 2>/dev/null\nif [ $? != 0 ]; then\n /usr/libexec/PlistBuddy -c \"Add :CFBundleVersion string $buildNumber\" \"$infoPlistPath\"\nelse\n /usr/libexec/PlistBuddy -c \"Set :CFBundleVersion $buildNumber\" \"$infoPlistPath\"\nfi\n\n# Check and add CFBundleShortVersionString if it does not exist\n/usr/libexec/PlistBuddy -c \"Print :CFBundleShortVersionString\" \"$infoPlistPath\" 2>/dev/null\nif [ $? != 0 ]; then\n /usr/libexec/PlistBuddy -c \"Add :CFBundleShortVersionString string $versionNumber\" \"$infoPlistPath\"\nelse\n /usr/libexec/PlistBuddy -c \"Set :CFBundleShortVersionString $versionNumber\" \"$infoPlistPath\"\nfi\n\ncript file from your workspace to insert its path.\n"; + }; F0C8BA10A27CA77E18F842E7 /* [CP] Check Pods Manifest.lock */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; @@ -875,8 +993,23 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + E612EC352D0F07A80022720C /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXSourcesBuildPhase section */ +/* Begin PBXTargetDependency section */ + E612EC472D0F07AD0022720C /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = E612EC382D0F07A80022720C /* HomePlayerWidgetExtension */; + targetProxy = E612EC462D0F07AD0022720C /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + /* Begin PBXVariantGroup section */ 97C146FA1CF9000F007C117D /* Main.storyboard */ = { isa = PBXVariantGroup; @@ -953,6 +1086,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1082,6 +1216,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1105,6 +1240,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1127,6 +1263,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1150,6 +1287,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1172,6 +1310,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1249,6 +1388,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1272,6 +1412,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1347,6 +1488,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1369,6 +1511,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1441,6 +1584,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1463,6 +1607,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1485,6 +1630,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1508,6 +1654,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1531,6 +1678,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1553,6 +1701,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1575,6 +1724,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1597,6 +1747,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1674,6 +1825,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1697,6 +1849,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1720,6 +1873,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1795,6 +1949,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1817,6 +1972,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1839,6 +1995,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1911,6 +2068,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1933,6 +2091,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1955,6 +2114,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1975,8 +2135,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -1998,8 +2159,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2021,8 +2183,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2044,8 +2207,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2066,8 +2230,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2088,8 +2253,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2110,8 +2276,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2132,8 +2299,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2154,8 +2322,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2233,6 +2402,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2256,6 +2426,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2279,6 +2450,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2300,8 +2472,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 9740EEB21CF90195004384FC /* Debug.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2377,6 +2550,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2399,6 +2573,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2421,6 +2596,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2441,8 +2617,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2515,6 +2692,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = Runner/Runner.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2537,6 +2715,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = stable.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2559,6 +2738,7 @@ buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = dev.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2579,8 +2759,9 @@ isa = XCBuildConfiguration; baseConfigurationReference = 7AFA3C8E1D35360C0083082E /* Release.xcconfig */; buildSettings = { - ASSETCATALOG_COMPILER_APPICON_NAME = "AppIcon-nightly"; + ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; CLANG_ENABLE_MODULES = YES; + CODE_SIGN_ENTITLEMENTS = nightly.entitlements; CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; DEVELOPMENT_TEAM = 88NVGSJ5N3; ENABLE_BITCODE = NO; @@ -2597,6 +2778,498 @@ }; name = "Profile-nightly"; }; + E612EC4A2D0F07AD0022720C /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Debug; + }; + E612EC4B2D0F07AD0022720C /* Debug-nightly */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Debug-nightly"; + }; + E612EC4C2D0F07AD0022720C /* Debug-dev */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Debug-dev"; + }; + E612EC4D2D0F07AD0022720C /* Debug-stable */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Debug-stable"; + }; + E612EC4E2D0F07AD0022720C /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Release; + }; + E612EC4F2D0F07AD0022720C /* Release-nightly */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Release-nightly"; + }; + E612EC502D0F07AD0022720C /* Release-dev */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Release-dev"; + }; + E612EC512D0F07AD0022720C /* Release-stable */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Release-stable"; + }; + E612EC522D0F07AD0022720C /* Profile */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Profile; + }; + E612EC532D0F07AD0022720C /* Profile-nightly */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Profile-nightly"; + }; + E612EC542D0F07AD0022720C /* Profile-dev */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Profile-dev"; + }; + E612EC552D0F07AD0022720C /* Profile-stable */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = AppIcon; + ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AppIcon; + ASSETCATALOG_COMPILER_WIDGET_BACKGROUND_COLOR_NAME = AppIcon; + CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CLANG_WARN_DOCUMENTATION_COMMENTS = YES; + CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; + CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; + CODE_SIGN_ENTITLEMENTS = HomePlayerWidgetExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = 88NVGSJ5N3; + ENABLE_USER_SCRIPT_SANDBOXING = YES; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = HomePlayerWidget/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = HomePlayerWidget; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 18.2; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = oss.krtirtho.spotube.HomePlayerWidget; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = "Profile-stable"; + }; /* End XCBuildConfiguration section */ /* Begin XCConfigurationList section */ @@ -2695,6 +3368,25 @@ defaultConfigurationIsVisible = 0; defaultConfigurationName = Release; }; + E612EC572D0F07AD0022720C /* Build configuration list for PBXNativeTarget "HomePlayerWidgetExtension" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + E612EC4A2D0F07AD0022720C /* Debug */, + E612EC4B2D0F07AD0022720C /* Debug-nightly */, + E612EC4C2D0F07AD0022720C /* Debug-dev */, + E612EC4D2D0F07AD0022720C /* Debug-stable */, + E612EC4E2D0F07AD0022720C /* Release */, + E612EC4F2D0F07AD0022720C /* Release-nightly */, + E612EC502D0F07AD0022720C /* Release-dev */, + E612EC512D0F07AD0022720C /* Release-stable */, + E612EC522D0F07AD0022720C /* Profile */, + E612EC532D0F07AD0022720C /* Profile-nightly */, + E612EC542D0F07AD0022720C /* Profile-dev */, + E612EC552D0F07AD0022720C /* Profile-stable */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; /* End XCConfigurationList section */ }; rootObject = 97C146E61CF9000F007C117D /* Project object */; diff --git a/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme b/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme index 5e31d3d3..c53e2b31 100644 --- a/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme +++ b/ios/Runner.xcodeproj/xcshareddata/xcschemes/Runner.xcscheme @@ -48,6 +48,7 @@ ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" debugServiceExtension = "internal" + enableGPUValidationMode = "1" allowLocationSimulation = "YES"> diff --git a/ios/Runner/AppDelegate.swift b/ios/Runner/AppDelegate.swift index b6363034..f512ac86 100644 --- a/ios/Runner/AppDelegate.swift +++ b/ios/Runner/AppDelegate.swift @@ -1,12 +1,17 @@ import UIKit import Flutter -@main +@UIApplicationMain @objc class AppDelegate: FlutterAppDelegate { override func application( _ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? ) -> Bool { + // Add this to get Documents directory path + if let documentsPath = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).first?.path { + UserDefaults.standard.set(documentsPath, forKey: "download_path") + } + GeneratedPluginRegistrant.register(with: self) return super.application(application, didFinishLaunchingWithOptions: launchOptions) } diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-1024x1024@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-1024x1024@1x.png index dbc4596b..2185c35d 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-1024x1024@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-1024x1024@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@1x.png index 4836771d..172bc383 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@2x.png index 90954ce9..876617d9 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@3x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@3x.png index 9c0ebd5f..fec86d87 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@3x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-20x20@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@1x.png index 94cd79be..fbb0f45a 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@2x.png index ff70cab7..854e1e45 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@3x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@3x.png index 6cdda1b6..420d2a17 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@3x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-29x29@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@1x.png index 90954ce9..876617d9 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@2x.png index 5184f84f..6b7a608d 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@3x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@3x.png index 57e21a75..d871fa3f 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@3x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-40x40@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@1x.png index 93e157b6..e77b38df 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@2x.png index d175beb2..358cb28f 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-50x50@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@1x.png index 6d634c87..87f4290b 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@2x.png index 22da4950..53c10a09 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-57x57@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@2x.png index 57e21a75..d871fa3f 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@3x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@3x.png index 3cfd01c2..4bb39789 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@3x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-60x60@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@1x.png index a826bb73..8164cc67 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@2x.png index 3a8a7832..e58ef25f 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-72x72@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@1x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@1x.png index f233322b..f5e1ae5a 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@1x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@2x.png index 2f5b082a..99bd36c9 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-76x76@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-83.5x83.5@2x.png b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-83.5x83.5@2x.png index e4ecc19a..8d199658 100644 Binary files a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-83.5x83.5@2x.png and b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/AppIcon-nightly-83.5x83.5@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/Contents.json b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/Contents.json index e8947587..1ce0f517 100644 --- a/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/Contents.json +++ b/ios/Runner/Assets.xcassets/AppIcon-nightly.appiconset/Contents.json @@ -1 +1 @@ -{"images":[{"size":"20x20","idiom":"iphone","filename":"AppIcon-nightly-20x20@2x.png","scale":"2x"},{"size":"20x20","idiom":"iphone","filename":"AppIcon-nightly-20x20@3x.png","scale":"3x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-nightly-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-nightly-29x29@2x.png","scale":"2x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-nightly-29x29@3x.png","scale":"3x"},{"size":"40x40","idiom":"iphone","filename":"AppIcon-nightly-40x40@2x.png","scale":"2x"},{"size":"40x40","idiom":"iphone","filename":"AppIcon-nightly-40x40@3x.png","scale":"3x"},{"size":"50x50","idiom":"ipad","filename":"AppIcon-nightly-50x50@1x.png","scale":"1x"},{"size":"50x50","idiom":"ipad","filename":"AppIcon-nightly-50x50@2x.png","scale":"2x"},{"size":"57x57","idiom":"iphone","filename":"AppIcon-nightly-57x57@1x.png","scale":"1x"},{"size":"57x57","idiom":"iphone","filename":"AppIcon-nightly-57x57@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"AppIcon-nightly-60x60@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"AppIcon-nightly-60x60@3x.png","scale":"3x"},{"size":"20x20","idiom":"ipad","filename":"AppIcon-nightly-20x20@1x.png","scale":"1x"},{"size":"20x20","idiom":"ipad","filename":"AppIcon-nightly-20x20@2x.png","scale":"2x"},{"size":"29x29","idiom":"ipad","filename":"AppIcon-nightly-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"ipad","filename":"AppIcon-nightly-29x29@2x.png","scale":"2x"},{"size":"40x40","idiom":"ipad","filename":"AppIcon-nightly-40x40@1x.png","scale":"1x"},{"size":"40x40","idiom":"ipad","filename":"AppIcon-nightly-40x40@2x.png","scale":"2x"},{"size":"72x72","idiom":"ipad","filename":"AppIcon-nightly-72x72@1x.png","scale":"1x"},{"size":"72x72","idiom":"ipad","filename":"AppIcon-nightly-72x72@2x.png","scale":"2x"},{"size":"76x76","idiom":"ipad","filename":"AppIcon-nightly-76x76@1x.png","scale":"1x"},{"size":"76x76","idiom":"ipad","filename":"AppIcon-nightly-76x76@2x.png","scale":"2x"},{"size":"83.5x83.5","idiom":"ipad","filename":"AppIcon-nightly-83.5x83.5@2x.png","scale":"2x"},{"size":"1024x1024","idiom":"ios-marketing","filename":"AppIcon-nightly-1024x1024@1x.png","scale":"1x"}],"info":{"version":1,"author":"xcode"}} \ No newline at end of file +{"images":[{"size":"20x20","idiom":"iphone","filename":"AppIcon-nightly-20x20@2x.png","scale":"2x"},{"size":"20x20","idiom":"iphone","filename":"AppIcon-nightly-20x20@3x.png","scale":"3x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-nightly-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-nightly-29x29@2x.png","scale":"2x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-nightly-29x29@3x.png","scale":"3x"},{"size":"40x40","idiom":"iphone","filename":"AppIcon-nightly-40x40@2x.png","scale":"2x"},{"size":"40x40","idiom":"iphone","filename":"AppIcon-nightly-40x40@3x.png","scale":"3x"},{"size":"57x57","idiom":"iphone","filename":"AppIcon-nightly-57x57@1x.png","scale":"1x"},{"size":"57x57","idiom":"iphone","filename":"AppIcon-nightly-57x57@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"AppIcon-nightly-60x60@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"AppIcon-nightly-60x60@3x.png","scale":"3x"},{"size":"20x20","idiom":"ipad","filename":"AppIcon-nightly-20x20@1x.png","scale":"1x"},{"size":"20x20","idiom":"ipad","filename":"AppIcon-nightly-20x20@2x.png","scale":"2x"},{"size":"29x29","idiom":"ipad","filename":"AppIcon-nightly-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"ipad","filename":"AppIcon-nightly-29x29@2x.png","scale":"2x"},{"size":"40x40","idiom":"ipad","filename":"AppIcon-nightly-40x40@1x.png","scale":"1x"},{"size":"40x40","idiom":"ipad","filename":"AppIcon-nightly-40x40@2x.png","scale":"2x"},{"size":"50x50","idiom":"ipad","filename":"AppIcon-nightly-50x50@1x.png","scale":"1x"},{"size":"50x50","idiom":"ipad","filename":"AppIcon-nightly-50x50@2x.png","scale":"2x"},{"size":"72x72","idiom":"ipad","filename":"AppIcon-nightly-72x72@1x.png","scale":"1x"},{"size":"72x72","idiom":"ipad","filename":"AppIcon-nightly-72x72@2x.png","scale":"2x"},{"size":"76x76","idiom":"ipad","filename":"AppIcon-nightly-76x76@1x.png","scale":"1x"},{"size":"76x76","idiom":"ipad","filename":"AppIcon-nightly-76x76@2x.png","scale":"2x"},{"size":"83.5x83.5","idiom":"ipad","filename":"AppIcon-nightly-83.5x83.5@2x.png","scale":"2x"},{"size":"1024x1024","idiom":"ios-marketing","filename":"AppIcon-nightly-1024x1024@1x.png","scale":"1x"}],"info":{"version":1,"author":"xcode"}} \ No newline at end of file diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json index 05843b52..d0d98aa1 100644 --- a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json +++ b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Contents.json @@ -1 +1 @@ -{"images":[{"size":"20x20","idiom":"iphone","filename":"AppIcon-20x20@2x.png","scale":"2x"},{"size":"20x20","idiom":"iphone","filename":"AppIcon-20x20@3x.png","scale":"3x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-29x29@2x.png","scale":"2x"},{"size":"29x29","idiom":"iphone","filename":"AppIcon-29x29@3x.png","scale":"3x"},{"size":"40x40","idiom":"iphone","filename":"AppIcon-40x40@2x.png","scale":"2x"},{"size":"40x40","idiom":"iphone","filename":"AppIcon-40x40@3x.png","scale":"3x"},{"size":"50x50","idiom":"ipad","filename":"AppIcon-50x50@1x.png","scale":"1x"},{"size":"50x50","idiom":"ipad","filename":"AppIcon-50x50@2x.png","scale":"2x"},{"size":"57x57","idiom":"iphone","filename":"AppIcon-57x57@1x.png","scale":"1x"},{"size":"57x57","idiom":"iphone","filename":"AppIcon-57x57@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"AppIcon-60x60@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"AppIcon-60x60@3x.png","scale":"3x"},{"size":"20x20","idiom":"ipad","filename":"AppIcon-20x20@1x.png","scale":"1x"},{"size":"20x20","idiom":"ipad","filename":"AppIcon-20x20@2x.png","scale":"2x"},{"size":"29x29","idiom":"ipad","filename":"AppIcon-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"ipad","filename":"AppIcon-29x29@2x.png","scale":"2x"},{"size":"40x40","idiom":"ipad","filename":"AppIcon-40x40@1x.png","scale":"1x"},{"size":"40x40","idiom":"ipad","filename":"AppIcon-40x40@2x.png","scale":"2x"},{"size":"72x72","idiom":"ipad","filename":"AppIcon-72x72@1x.png","scale":"1x"},{"size":"72x72","idiom":"ipad","filename":"AppIcon-72x72@2x.png","scale":"2x"},{"size":"76x76","idiom":"ipad","filename":"AppIcon-76x76@1x.png","scale":"1x"},{"size":"76x76","idiom":"ipad","filename":"AppIcon-76x76@2x.png","scale":"2x"},{"size":"83.5x83.5","idiom":"ipad","filename":"AppIcon-83.5x83.5@2x.png","scale":"2x"},{"size":"1024x1024","idiom":"ios-marketing","filename":"AppIcon-1024x1024@1x.png","scale":"1x"}],"info":{"version":1,"author":"xcode"}} \ No newline at end of file +{"images":[{"size":"20x20","idiom":"iphone","filename":"Icon-App-20x20@2x.png","scale":"2x"},{"size":"20x20","idiom":"iphone","filename":"Icon-App-20x20@3x.png","scale":"3x"},{"size":"29x29","idiom":"iphone","filename":"Icon-App-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"iphone","filename":"Icon-App-29x29@2x.png","scale":"2x"},{"size":"29x29","idiom":"iphone","filename":"Icon-App-29x29@3x.png","scale":"3x"},{"size":"40x40","idiom":"iphone","filename":"Icon-App-40x40@2x.png","scale":"2x"},{"size":"40x40","idiom":"iphone","filename":"Icon-App-40x40@3x.png","scale":"3x"},{"size":"57x57","idiom":"iphone","filename":"Icon-App-57x57@1x.png","scale":"1x"},{"size":"57x57","idiom":"iphone","filename":"Icon-App-57x57@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"Icon-App-60x60@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"Icon-App-60x60@3x.png","scale":"3x"},{"size":"20x20","idiom":"ipad","filename":"Icon-App-20x20@1x.png","scale":"1x"},{"size":"20x20","idiom":"ipad","filename":"Icon-App-20x20@2x.png","scale":"2x"},{"size":"29x29","idiom":"ipad","filename":"Icon-App-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"ipad","filename":"Icon-App-29x29@2x.png","scale":"2x"},{"size":"40x40","idiom":"ipad","filename":"Icon-App-40x40@1x.png","scale":"1x"},{"size":"40x40","idiom":"ipad","filename":"Icon-App-40x40@2x.png","scale":"2x"},{"size":"50x50","idiom":"ipad","filename":"Icon-App-50x50@1x.png","scale":"1x"},{"size":"50x50","idiom":"ipad","filename":"Icon-App-50x50@2x.png","scale":"2x"},{"size":"72x72","idiom":"ipad","filename":"Icon-App-72x72@1x.png","scale":"1x"},{"size":"72x72","idiom":"ipad","filename":"Icon-App-72x72@2x.png","scale":"2x"},{"size":"76x76","idiom":"ipad","filename":"Icon-App-76x76@1x.png","scale":"1x"},{"size":"76x76","idiom":"ipad","filename":"Icon-App-76x76@2x.png","scale":"2x"},{"size":"83.5x83.5","idiom":"ipad","filename":"Icon-App-83.5x83.5@2x.png","scale":"2x"},{"size":"1024x1024","idiom":"ios-marketing","filename":"Icon-App-1024x1024@1x.png","scale":"1x"}],"info":{"version":1,"author":"xcode"}} \ No newline at end of file diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png new file mode 100644 index 00000000..59407b44 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-1024x1024@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png new file mode 100644 index 00000000..5eacaa5f Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png new file mode 100644 index 00000000..bbb9f839 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png new file mode 100644 index 00000000..28d0d8a8 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-20x20@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png new file mode 100644 index 00000000..b1df0e72 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png new file mode 100644 index 00000000..24b76e25 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png new file mode 100644 index 00000000..5c0b6d57 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-29x29@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png new file mode 100644 index 00000000..bbb9f839 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png new file mode 100644 index 00000000..bdc5656b Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png new file mode 100644 index 00000000..c03c89fe Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-40x40@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-50x50@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-50x50@1x.png new file mode 100644 index 00000000..40b88968 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-50x50@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-50x50@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-50x50@2x.png new file mode 100644 index 00000000..2050f427 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-50x50@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-57x57@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-57x57@1x.png new file mode 100644 index 00000000..d1ccab30 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-57x57@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-57x57@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-57x57@2x.png new file mode 100644 index 00000000..47c629f3 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-57x57@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png new file mode 100644 index 00000000..c03c89fe Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png new file mode 100644 index 00000000..22d28c12 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-60x60@3x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-72x72@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-72x72@1x.png new file mode 100644 index 00000000..c9e5cfad Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-72x72@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-72x72@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-72x72@2x.png new file mode 100644 index 00000000..3450fb00 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-72x72@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png new file mode 100644 index 00000000..3dd3eda3 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@1x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png new file mode 100644 index 00000000..2e69c843 Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-76x76@2x.png differ diff --git a/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png new file mode 100644 index 00000000..f769eb6e Binary files /dev/null and b/ios/Runner/Assets.xcassets/AppIcon.appiconset/Icon-App-83.5x83.5@2x.png differ diff --git a/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage.png b/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage.png index 5a197da3..3ff2a2da 100644 Binary files a/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage.png and b/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage.png differ diff --git a/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@2x.png b/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@2x.png index efc7e6c8..8e2bb197 100644 Binary files a/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@2x.png and b/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@2x.png differ diff --git a/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@3x.png b/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@3x.png index 593877e0..d301093a 100644 Binary files a/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@3x.png and b/ios/Runner/Assets.xcassets/BrandingImage.imageset/BrandingImage@3x.png differ diff --git a/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage.png b/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage.png index 80579983..3ff2a2da 100644 Binary files a/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage.png and b/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage.png differ diff --git a/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@2x.png b/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@2x.png index 0bcf138d..8e2bb197 100644 Binary files a/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@2x.png and b/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@2x.png differ diff --git a/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@3x.png b/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@3x.png index c7d01776..d301093a 100644 Binary files a/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@3x.png and b/ios/Runner/Assets.xcassets/BrandingImageNightly.imageset/BrandingImage@3x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchBackground.imageset/background.png b/ios/Runner/Assets.xcassets/LaunchBackground.imageset/background.png index 203fc77a..4bebb9de 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchBackground.imageset/background.png and b/ios/Runner/Assets.xcassets/LaunchBackground.imageset/background.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchBackgroundNightly.imageset/background.png b/ios/Runner/Assets.xcassets/LaunchBackgroundNightly.imageset/background.png index 203fc77a..4bebb9de 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchBackgroundNightly.imageset/background.png and b/ios/Runner/Assets.xcassets/LaunchBackgroundNightly.imageset/background.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png index c5e4aca0..6e04efdc 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png and b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png index 41222c6c..51a669aa 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png and b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@2x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png index 5ab19d32..cc79cb85 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png and b/ios/Runner/Assets.xcassets/LaunchImage.imageset/LaunchImage@3x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage.png b/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage.png index 86d3fe74..d8f7cc0e 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage.png and b/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@2x.png b/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@2x.png index dbb0ea02..17a2c373 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@2x.png and b/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@2x.png differ diff --git a/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@3x.png b/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@3x.png index 12eb5531..db53f016 100644 Binary files a/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@3x.png and b/ios/Runner/Assets.xcassets/LaunchImageNightly.imageset/LaunchImage@3x.png differ diff --git a/ios/Runner/Base.lproj/LaunchScreen.storyboard b/ios/Runner/Base.lproj/LaunchScreen.storyboard index 208e1c53..ec8a1de3 100644 --- a/ios/Runner/Base.lproj/LaunchScreen.storyboard +++ b/ios/Runner/Base.lproj/LaunchScreen.storyboard @@ -22,7 +22,7 @@ - + diff --git a/ios/Runner/Base.lproj/LaunchScreenNightly.storyboard b/ios/Runner/Base.lproj/LaunchScreenNightly.storyboard index 6869214f..645a417f 100644 --- a/ios/Runner/Base.lproj/LaunchScreenNightly.storyboard +++ b/ios/Runner/Base.lproj/LaunchScreenNightly.storyboard @@ -22,7 +22,7 @@ - + diff --git a/ios/Runner/Info.plist b/ios/Runner/Info.plist index ffd511a4..8d6c09a2 100644 --- a/ios/Runner/Info.plist +++ b/ios/Runner/Info.plist @@ -1,76 +1,82 @@ - - CADisableMinimumFrameDurationOnPhone - - CFBundleDevelopmentRegion - $(DEVELOPMENT_LANGUAGE) - CFBundleDisplayName - Spotube - CFBundleExecutable - $(EXECUTABLE_NAME) - CFBundleIdentifier - $(PRODUCT_BUNDLE_IDENTIFIER) - CFBundleInfoDictionaryVersion - 6.0 - CFBundleName - spotube - CFBundlePackageType - APPL - CFBundleShortVersionString - $(FLUTTER_BUILD_NAME) - CFBundleSignature - ???? - CFBundleVersion - $(FLUTTER_BUILD_NUMBER) - LSRequiresIPhoneOS - - NSAppTransportSecurity - - NSAllowsArbitraryLoads - - NSAllowsArbitraryLoadsForMedia - - - NSCameraUsageDescription - This app require access to the device camera - NSMicrophoneUsageDescription - This app does not require access to the device microphone - NSPhotoLibraryUsageDescription - This app require access to the photo library - UIApplicationSupportsIndirectInputEvents - - UIBackgroundModes - - audio - - UILaunchStoryboardName - LaunchScreen - UIMainStoryboardFile - Main - UIStatusBarHidden - - UISupportedInterfaceOrientations - - UIInterfaceOrientationPortrait - UIInterfaceOrientationLandscapeLeft - UIInterfaceOrientationLandscapeRight - - UISupportedInterfaceOrientations~ipad - - UIInterfaceOrientationPortrait - UIInterfaceOrientationPortraitUpsideDown - UIInterfaceOrientationLandscapeLeft - UIInterfaceOrientationLandscapeRight - - UIViewControllerBasedStatusBarAppearance - - NSLocalNetworkUsageDescription - To allow other devices on the network control playback of Spotube securely. - NSBonjourServices - - _spotube._tcp - - - \ No newline at end of file + + CADisableMinimumFrameDurationOnPhone + + CFBundleDevelopmentRegion + $(DEVELOPMENT_LANGUAGE) + CFBundleDisplayName + Spotube + CFBundleExecutable + $(EXECUTABLE_NAME) + CFBundleIdentifier + $(PRODUCT_BUNDLE_IDENTIFIER) + CFBundleInfoDictionaryVersion + 6.0 + CFBundleName + spotube + CFBundlePackageType + APPL + CFBundleShortVersionString + $(FLUTTER_BUILD_NAME) + CFBundleSignature + ???? + CFBundleVersion + $(FLUTTER_BUILD_NUMBER) + LSRequiresIPhoneOS + + NSAppTransportSecurity + + NSAllowsArbitraryLoads + + NSAllowsArbitraryLoadsForMedia + + + NSCameraUsageDescription + This app require access to the device camera + NSMicrophoneUsageDescription + This app does not require access to the device microphone + NSPhotoLibraryUsageDescription + This app require access to the photo library + UIApplicationSupportsIndirectInputEvents + + UIBackgroundModes + + audio + + UILaunchStoryboardName + LaunchScreen + UIMainStoryboardFile + Main + UIStatusBarHidden + + UISupportedInterfaceOrientations + + UIInterfaceOrientationPortrait + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UISupportedInterfaceOrientations~ipad + + UIInterfaceOrientationPortrait + UIInterfaceOrientationPortraitUpsideDown + UIInterfaceOrientationLandscapeLeft + UIInterfaceOrientationLandscapeRight + + UIViewControllerBasedStatusBarAppearance + + NSLocalNetworkUsageDescription + To allow other devices on the network control playback of Spotube securely. + NSBonjourServices + + _spotube._tcp + + UIFileSharingEnabled + + LSSupportsOpeningDocumentsInPlace + + UISupportsDocumentBrowser + + + diff --git a/ios/Runner/Runner.entitlements b/ios/Runner/Runner.entitlements new file mode 100644 index 00000000..58165678 --- /dev/null +++ b/ios/Runner/Runner.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.spotube_home_player_widget + + + diff --git a/ios/dev.entitlements b/ios/dev.entitlements new file mode 100644 index 00000000..58165678 --- /dev/null +++ b/ios/dev.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.spotube_home_player_widget + + + diff --git a/ios/nightly.entitlements b/ios/nightly.entitlements new file mode 100644 index 00000000..58165678 --- /dev/null +++ b/ios/nightly.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.spotube_home_player_widget + + + diff --git a/ios/stable.entitlements b/ios/stable.entitlements new file mode 100644 index 00000000..58165678 --- /dev/null +++ b/ios/stable.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.spotube_home_player_widget + + + diff --git a/l10n.yaml b/l10n.yaml index b49b5df4..d5911fe1 100644 --- a/l10n.yaml +++ b/l10n.yaml @@ -1,4 +1,4 @@ arb-dir: lib/l10n template-arb-file: app_en.arb -output-localization-file: app_localizations.dart +output-dir: lib/l10n/generated untranslated-messages-file: untranslated_messages.json diff --git a/lib/collections/assets.gen.dart b/lib/collections/assets.gen.dart index 6825fbd5..7ab0ad03 100644 --- a/lib/collections/assets.gen.dart +++ b/lib/collections/assets.gen.dart @@ -1,3 +1,5 @@ +// dart format width=80 + /// GENERATED CODE - DO NOT MODIFY BY HAND /// ***************************************************** /// FlutterGen @@ -5,122 +7,129 @@ // coverage:ignore-file // ignore_for_file: type=lint -// ignore_for_file: directives_ordering,unnecessary_import,implicit_dynamic_list_literal,deprecated_member_use +// ignore_for_file: deprecated_member_use,directives_ordering,implicit_dynamic_list_literal,unnecessary_import import 'package:flutter/widgets.dart'; -class $AssetsLogosGen { - const $AssetsLogosGen(); +class $AssetsBrandingGen { + const $AssetsBrandingGen(); - /// File path: assets/logos/songlink-transparent.png - AssetGenImage get songlinkTransparent => - const AssetGenImage('assets/logos/songlink-transparent.png'); + /// File path: assets/branding/spotube-logo-light.png + AssetGenImage get spotubeLogoLight => + const AssetGenImage('assets/branding/spotube-logo-light.png'); - /// File path: assets/logos/songlink.png - AssetGenImage get songlink => - const AssetGenImage('assets/logos/songlink.png'); + /// File path: assets/branding/spotube-logo.ico + String get spotubeLogoIco => 'assets/branding/spotube-logo.ico'; + + /// File path: assets/branding/spotube-logo.png + AssetGenImage get spotubeLogoPng => + const AssetGenImage('assets/branding/spotube-logo.png'); /// List of all assets - List get values => [songlinkTransparent, songlink]; + List get values => + [spotubeLogoLight, spotubeLogoIco, spotubeLogoPng]; } -class $AssetsTutorialGen { - const $AssetsTutorialGen(); +class $AssetsImagesGen { + const $AssetsImagesGen(); - /// File path: assets/tutorial/step-1.png - AssetGenImage get step1 => const AssetGenImage('assets/tutorial/step-1.png'); + /// File path: assets/images/album-placeholder.png + AssetGenImage get albumPlaceholder => + const AssetGenImage('assets/images/album-placeholder.png'); - /// File path: assets/tutorial/step-2.png - AssetGenImage get step2 => const AssetGenImage('assets/tutorial/step-2.png'); + /// File path: assets/images/bengali-patterns-bg.jpg + AssetGenImage get bengaliPatternsBg => + const AssetGenImage('assets/images/bengali-patterns-bg.jpg'); - /// File path: assets/tutorial/step-3.png - AssetGenImage get step3 => const AssetGenImage('assets/tutorial/step-3.png'); + /// File path: assets/images/liked-tracks.jpg + AssetGenImage get likedTracks => + const AssetGenImage('assets/images/liked-tracks.jpg'); + + /// Directory path: assets/images/logos + $AssetsImagesLogosGen get logos => const $AssetsImagesLogosGen(); + + /// File path: assets/images/placeholder.png + AssetGenImage get placeholder => + const AssetGenImage('assets/images/placeholder.png'); + + /// File path: assets/images/user-placeholder.png + AssetGenImage get userPlaceholder => + const AssetGenImage('assets/images/user-placeholder.png'); /// List of all assets - List get values => [step1, step2, step3]; + List get values => [ + albumPlaceholder, + bengaliPatternsBg, + likedTracks, + placeholder, + userPlaceholder + ]; +} + +class $AssetsPluginsGen { + const $AssetsPluginsGen(); + + /// Directory path: assets/plugins/spotube-plugin-musicbrainz-listenbrainz + $AssetsPluginsSpotubePluginMusicbrainzListenbrainzGen + get spotubePluginMusicbrainzListenbrainz => + const $AssetsPluginsSpotubePluginMusicbrainzListenbrainzGen(); + + /// Directory path: assets/plugins/spotube-plugin-youtube-audio + $AssetsPluginsSpotubePluginYoutubeAudioGen get spotubePluginYoutubeAudio => + const $AssetsPluginsSpotubePluginYoutubeAudioGen(); +} + +class $AssetsImagesLogosGen { + const $AssetsImagesLogosGen(); + + /// File path: assets/images/logos/dab-music.png + AssetGenImage get dabMusic => + const AssetGenImage('assets/images/logos/dab-music.png'); + + /// File path: assets/images/logos/invidious.jpg + AssetGenImage get invidious => + const AssetGenImage('assets/images/logos/invidious.jpg'); + + /// File path: assets/images/logos/jiosaavn.png + AssetGenImage get jiosaavn => + const AssetGenImage('assets/images/logos/jiosaavn.png'); + + /// List of all assets + List get values => [dabMusic, invidious, jiosaavn]; +} + +class $AssetsPluginsSpotubePluginMusicbrainzListenbrainzGen { + const $AssetsPluginsSpotubePluginMusicbrainzListenbrainzGen(); + + /// File path: assets/plugins/spotube-plugin-musicbrainz-listenbrainz/plugin.smplug + String get plugin => + 'assets/plugins/spotube-plugin-musicbrainz-listenbrainz/plugin.smplug'; + + /// List of all assets + List get values => [plugin]; +} + +class $AssetsPluginsSpotubePluginYoutubeAudioGen { + const $AssetsPluginsSpotubePluginYoutubeAudioGen(); + + /// File path: assets/plugins/spotube-plugin-youtube-audio/plugin.smplug + String get plugin => + 'assets/plugins/spotube-plugin-youtube-audio/plugin.smplug'; + + /// List of all assets + List get values => [plugin]; } class Assets { - Assets._(); + const Assets._(); static const String license = 'LICENSE'; - static const AssetGenImage albumPlaceholder = - AssetGenImage('assets/album-placeholder.png'); - static const AssetGenImage bengaliPatternsBg = - AssetGenImage('assets/bengali-patterns-bg.jpg'); - static const AssetGenImage branding = AssetGenImage('assets/branding.png'); - static const AssetGenImage emptyBox = AssetGenImage('assets/empty_box.png'); - static const AssetGenImage invidious = AssetGenImage('assets/invidious.jpg'); - static const AssetGenImage jiosaavn = AssetGenImage('assets/jiosaavn.png'); - static const AssetGenImage likedTracks = - AssetGenImage('assets/liked-tracks.jpg'); - static const $AssetsLogosGen logos = $AssetsLogosGen(); - static const AssetGenImage placeholder = - AssetGenImage('assets/placeholder.png'); - static const AssetGenImage spotubeHeroBanner = - AssetGenImage('assets/spotube-hero-banner.png'); - static const AssetGenImage spotubeLogoForeground = - AssetGenImage('assets/spotube-logo-foreground.jpg'); - static const AssetGenImage spotubeLogoBmp = - AssetGenImage('assets/spotube-logo.bmp'); - static const String spotubeLogoIco = 'assets/spotube-logo.ico'; - static const AssetGenImage spotubeLogoPng = - AssetGenImage('assets/spotube-logo.png'); - static const String spotubeLogoSvg = 'assets/spotube-logo.svg'; - static const AssetGenImage spotubeLogoAndroid12 = - AssetGenImage('assets/spotube-logo_android12.png'); - static const AssetGenImage spotubeNightlyLogoForeground = - AssetGenImage('assets/spotube-nightly-logo-foreground.jpg'); - static const AssetGenImage spotubeNightlyLogoPng = - AssetGenImage('assets/spotube-nightly-logo.png'); - static const String spotubeNightlyLogoSvg = 'assets/spotube-nightly-logo.svg'; - static const AssetGenImage spotubeNightlyLogoAndroid12 = - AssetGenImage('assets/spotube-nightly-logo_android12.png'); - static const AssetGenImage spotubeScreenshot = - AssetGenImage('assets/spotube-screenshot.png'); - static const AssetGenImage spotubeTallCapsule = - AssetGenImage('assets/spotube-tall-capsule.png'); - static const AssetGenImage spotubeWideCapsuleLarge = - AssetGenImage('assets/spotube-wide-capsule-large.png'); - static const AssetGenImage spotubeWideCapsuleSmall = - AssetGenImage('assets/spotube-wide-capsule-small.png'); - static const AssetGenImage spotubeBanner = - AssetGenImage('assets/spotube_banner.png'); - static const AssetGenImage success = AssetGenImage('assets/success.png'); - static const $AssetsTutorialGen tutorial = $AssetsTutorialGen(); - static const AssetGenImage userPlaceholder = - AssetGenImage('assets/user-placeholder.png'); + static const $AssetsBrandingGen branding = $AssetsBrandingGen(); + static const $AssetsImagesGen images = $AssetsImagesGen(); + static const $AssetsPluginsGen plugins = $AssetsPluginsGen(); /// List of all assets - static List get values => [ - license, - albumPlaceholder, - bengaliPatternsBg, - branding, - emptyBox, - invidious, - jiosaavn, - likedTracks, - placeholder, - spotubeHeroBanner, - spotubeLogoForeground, - spotubeLogoBmp, - spotubeLogoIco, - spotubeLogoPng, - spotubeLogoSvg, - spotubeLogoAndroid12, - spotubeNightlyLogoForeground, - spotubeNightlyLogoPng, - spotubeNightlyLogoSvg, - spotubeNightlyLogoAndroid12, - spotubeScreenshot, - spotubeTallCapsule, - spotubeWideCapsuleLarge, - spotubeWideCapsuleSmall, - spotubeBanner, - success, - userPlaceholder - ]; + static List get values => [license]; } class AssetGenImage { @@ -128,12 +137,14 @@ class AssetGenImage { this._assetName, { this.size, this.flavors = const {}, + this.animation, }); final String _assetName; final Size? size; final Set flavors; + final AssetGenImageAnimation? animation; Image image({ Key? key, @@ -156,7 +167,7 @@ class AssetGenImage { bool gaplessPlayback = true, bool isAntiAlias = false, String? package, - FilterQuality filterQuality = FilterQuality.low, + FilterQuality filterQuality = FilterQuality.medium, int? cacheWidth, int? cacheHeight, }) { @@ -203,3 +214,15 @@ class AssetGenImage { String get keyName => _assetName; } + +class AssetGenImageAnimation { + const AssetGenImageAnimation({ + required this.isAnimation, + required this.duration, + required this.frames, + }); + + final bool isAnimation; + final Duration duration; + final int frames; +} diff --git a/lib/collections/env.dart b/lib/collections/env.dart index eb60851f..52ef2bbf 100644 --- a/lib/collections/env.dart +++ b/lib/collections/env.dart @@ -10,9 +10,6 @@ enum ReleaseChannel { @Envied(obfuscate: true, requireEnvFile: true, path: ".env") abstract class Env { - @EnviedField(varName: 'SPOTIFY_SECRETS') - static final String rawSpotifySecrets = _Env.rawSpotifySecrets; - @EnviedField(varName: 'LASTFM_API_KEY') static final String lastFmApiKey = _Env.lastFmApiKey; @@ -24,14 +21,6 @@ abstract class Env { static bool get hideDonations => _hideDonations == 1; - static final spotifySecrets = rawSpotifySecrets.split(',').map((e) { - final secrets = e.trim().split(":").map((e) => e.trim()); - return { - "clientId": secrets.first, - "clientSecret": secrets.last, - }; - }).toList(); - @EnviedField(varName: 'ENABLE_UPDATE_CHECK', defaultValue: "1") static final String _enableUpdateChecker = _Env._enableUpdateChecker; diff --git a/lib/collections/fake.dart b/lib/collections/fake.dart index 31f97e0c..7d201ae2 100644 --- a/lib/collections/fake.dart +++ b/lib/collections/fake.dart @@ -1,230 +1,112 @@ -import 'package:spotify/spotify.dart'; import 'package:spotube/models/database/database.dart'; -import 'package:spotube/models/spotify/home_feed.dart'; -import 'package:spotube/models/spotify_friends.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/history/summary.dart'; abstract class FakeData { - static final Image image = Image() - ..height = 1 - ..width = 1 - ..url = "https://dummyimage.com/100x100/cfcfcf/cfcfcf.jpg"; + static final SpotubeImageObject image = SpotubeImageObject( + height: 100, + width: 100, + url: "https://dummyimage.com/100x100/cfcfcf/cfcfcf.jpg", + ); - static final Followers followers = Followers() - ..href = "text" - ..total = 1; - - static final Artist artist = Artist() - ..id = "1" - ..name = "Wow artist Good!" - ..images = [image] - ..popularity = 1 - ..type = "type" - ..uri = "uri" - ..externalUrls = externalUrls - ..genres = ["genre"] - ..href = "text" - ..followers = followers; - - static final externalIds = ExternalIds() - ..isrc = "text" - ..ean = "text" - ..upc = "text"; - - static final externalUrls = ExternalUrls()..spotify = "text"; - - static final Album album = Album() - ..id = "1" - ..genres = ["genre"] - ..label = "label" - ..popularity = 1 - ..albumType = AlbumType.album - ..artists = [artist] - ..availableMarkets = [Market.BD] - ..externalUrls = externalUrls - ..href = "text" - ..images = [image] - ..name = "Another good album" - ..releaseDate = "2021-01-01" - ..releaseDatePrecision = DatePrecision.day - ..tracks = [track] - ..type = "type" - ..uri = "uri" - ..externalIds = externalIds - ..copyrights = [ - Copyright() - ..type = CopyrightType.C - ..text = "text", - ]; - - static final ArtistSimple artistSimple = ArtistSimple() - ..id = "1" - ..name = "What an artist" - ..type = "type" - ..uri = "uri" - ..externalUrls = externalUrls; - - static final AlbumSimple albumSimple = AlbumSimple() - ..id = "1" - ..albumType = AlbumType.album - ..artists = [artistSimple] - ..availableMarkets = [Market.BD] - ..externalUrls = externalUrls - ..href = "text" - ..images = [image] - ..name = "A good album" - ..releaseDate = "2021-01-01" - ..releaseDatePrecision = DatePrecision.day - ..type = "type" - ..uri = "uri"; - - static final Track track = Track() - ..id = "1" - ..artists = [artist, artist, artist] - ..album = albumSimple - ..availableMarkets = [Market.BD] - ..discNumber = 1 - ..durationMs = 50000 - ..explicit = false - ..externalUrls = externalUrls - ..href = "text" - ..name = "A Track Name" - ..popularity = 1 - ..previewUrl = "url" - ..trackNumber = 1 - ..type = "type" - ..uri = "uri" - ..isPlayable = true - ..explicit = false - ..linkedFrom = trackLink; - - static final TrackLink trackLink = TrackLink() - ..id = "1" - ..type = "type" - ..uri = "uri" - ..externalUrls = {"spotify": "text"} - ..href = "text"; - - static final Paging paging = Paging() - ..href = "text" - ..itemsNative = [track.toJson()] - ..limit = 1 - ..next = "text" - ..offset = 1 - ..previous = "text" - ..total = 1; - - static final User user = User() - ..id = "1" - ..displayName = "Your Name" - ..birthdate = "2021-01-01" - ..country = Market.BD - ..email = "test@email.com" - ..followers = followers - ..href = "text" - ..images = [image] - ..type = "type" - ..uri = "uri"; - - static final TracksLink tracksLink = TracksLink() - ..href = "text" - ..total = 1; - - static final Playlist playlist = Playlist() - ..id = "1" - ..collaborative = false - ..description = "A very good playlist description" - ..externalUrls = externalUrls - ..followers = followers - ..href = "text" - ..images = [image] - ..name = "A good playlist" - ..owner = user - ..public = true - ..snapshotId = "text" - ..tracks = paging - ..tracksLink = tracksLink - ..type = "type" - ..uri = "uri"; - - static final PlaylistSimple playlistSimple = PlaylistSimple() - ..id = "1" - ..collaborative = false - ..externalUrls = externalUrls - ..href = "text" - ..images = [image] - ..name = "A good playlist" - ..owner = user - ..public = true - ..snapshotId = "text" - ..tracksLink = tracksLink - ..type = "type" - ..description = "A very good playlist description" - ..uri = "uri"; - - static final Category category = Category() - ..href = "text" - ..icons = [image] - ..id = "1" - ..name = "category"; - - static final friends = SpotifyFriends( - friends: [ - for (var i = 0; i < 3; i++) - SpotifyFriendActivity( - user: const SpotifyFriend( - name: "name", - imageUrl: "imageUrl", - uri: "uri", - ), - track: SpotifyActivityTrack( - name: "name", - artist: const SpotifyActivityArtist( - name: "name", - uri: "uri", - ), - album: const SpotifyActivityAlbum( - name: "name", - uri: "uri", - ), - context: SpotifyActivityContext( - name: "name", - index: i, - uri: "uri", - ), - imageUrl: "imageUrl", - uri: "uri", - ), - ), + static final SpotubeFullArtistObject artist = SpotubeFullArtistObject( + id: "1", + name: "What an artist", + externalUri: "https://example.com", + followers: 10000, + genres: ["genre"], + images: [ + SpotubeImageObject( + height: 100, + width: 100, + url: "https://dummyimage.com/100x100/cfcfcf/cfcfcf.jpg", + ), ], ); - static final feedSection = SpotifyHomeFeedSection( - typename: "HomeGenericSectionData", - uri: "spotify:section:lol", - title: "Dummy", - items: [ - for (int i = 0; i < 10; i++) - SpotifyHomeFeedSectionItem( - typename: "PlaylistResponseWrapper", - playlist: SpotifySectionPlaylist( - name: "Playlist $i", - description: "Really super important description $i", - format: "daily-mix", - images: [ - const SpotifySectionItemImage( - height: 1, - width: 1, - url: "https://dummyimage.com/100x100/cfcfcf/cfcfcf.jpg", - ), - ], - owner: "Spotify", - uri: "spotify:playlist:id", - ), - ) + static final SpotubeFullAlbumObject album = SpotubeFullAlbumObject( + id: "1", + name: "A good album", + externalUri: "https://example.com", + artists: [artistSimple], + releaseDate: "2021-01-01", + albumType: SpotubeAlbumType.album, + images: [image], + totalTracks: 10, + genres: ["genre"], + recordLabel: "Record Label", + ); + + static final SpotubeSimpleArtistObject artistSimple = + SpotubeSimpleArtistObject( + id: "1", + name: "What an artist", + externalUri: "https://example.com", + images: null, + ); + + static final SpotubeSimpleAlbumObject albumSimple = SpotubeSimpleAlbumObject( + albumType: SpotubeAlbumType.album, + artists: [], + externalUri: "https://example.com", + id: "1", + name: "A good album", + releaseDate: "2021-01-01", + images: [ + SpotubeImageObject( + height: 1, + width: 1, + url: "https://dummyimage.com/100x100/cfcfcf/cfcfcf.jpg", + ) ], ); + static final SpotubeFullTrackObject track = SpotubeTrackObject.full( + id: "1", + name: "A good track", + externalUri: "https://example.com", + album: albumSimple, + durationMs: 3 * 60 * 1000, // 3 minutes + isrc: "USUM72112345", + explicit: false, + ) as SpotubeFullTrackObject; + + static final SpotubeUserObject user = SpotubeUserObject( + id: "1", + name: "User Name", + externalUri: "https://example.com", + images: [image], + ); + + static final SpotubeFullPlaylistObject playlist = SpotubeFullPlaylistObject( + id: "1", + name: "A good playlist", + description: "A very good playlist description", + externalUri: "https://example.com", + collaborative: false, + public: true, + owner: user, + images: [image], + collaborators: [user]); + + static final SpotubeSimplePlaylistObject playlistSimple = + SpotubeSimplePlaylistObject( + id: "1", + name: "A good playlist", + description: "A very good playlist description", + externalUri: "https://example.com", + owner: user, + images: [image], + ); + + static final SpotubeBrowseSectionObject browseSection = + SpotubeBrowseSectionObject( + id: "section-id", + title: "Browse Section", + browseMore: true, + externalUri: "https://example.com/browse/section", + items: [playlistSimple, playlistSimple, playlistSimple]); + static const historySummary = PlaybackHistorySummary( albums: 1, artists: 1, diff --git a/lib/collections/fonts.gen.dart b/lib/collections/fonts.gen.dart new file mode 100644 index 00000000..d2c68231 --- /dev/null +++ b/lib/collections/fonts.gen.dart @@ -0,0 +1,25 @@ +// dart format width=80 +/// GENERATED CODE - DO NOT MODIFY BY HAND +/// ***************************************************** +/// FlutterGen +/// ***************************************************** + +// coverage:ignore-file +// ignore_for_file: type=lint +// ignore_for_file: deprecated_member_use,directives_ordering,implicit_dynamic_list_literal,unnecessary_import + +class FontFamily { + FontFamily._(); + + /// Font family: BootstrapIcons + static const String bootstrapIcons = 'BootstrapIcons'; + + /// Font family: Cookie + static const String cookie = 'Cookie'; + + /// Font family: RadixIcons + static const String radixIcons = 'RadixIcons'; + + /// Font family: Ubuntu Mono + static const String ubuntuMono = 'Ubuntu Mono'; +} diff --git a/lib/collections/gradients.dart b/lib/collections/gradients.dart index e861dde7..a7936ee2 100644 --- a/lib/collections/gradients.dart +++ b/lib/collections/gradients.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; const gradients = [ LinearGradient(colors: [ diff --git a/lib/collections/http-override.dart b/lib/collections/http-override.dart new file mode 100644 index 00000000..3bf4f30e --- /dev/null +++ b/lib/collections/http-override.dart @@ -0,0 +1,17 @@ +import 'dart:io'; + +const allowList = [ + "spotify.com", +]; + +class BadCertificateAllowlistOverrides extends HttpOverrides { + @override + HttpClient createHttpClient(SecurityContext? context) { + return super.createHttpClient(context) + ..badCertificateCallback = (X509Certificate cert, String host, int port) { + return allowList.any((allowedHost) { + return host.endsWith(allowedHost); + }); + }; + } +} diff --git a/lib/collections/intents.dart b/lib/collections/intents.dart index 4f446831..42c580ca 100644 --- a/lib/collections/intents.dart +++ b/lib/collections/intents.dart @@ -3,13 +3,9 @@ import 'dart:io'; import 'package:flutter/cupertino.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:go_router/go_router.dart'; import 'package:spotube/collections/routes.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/modules/player/player_controls.dart'; -import 'package:spotube/pages/home/home.dart'; -import 'package:spotube/pages/library/library.dart'; -import 'package:spotube/pages/lyrics/lyrics.dart'; -import 'package:spotube/pages/search/search.dart'; import 'package:spotube/provider/audio_player/querying_track_info.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/utils/platform.dart'; @@ -36,7 +32,7 @@ class PlayPauseAction extends Action { } class NavigationIntent extends Intent { - final GoRouter router; + final AppRouter router; final String path; const NavigationIntent(this.router, this.path); } @@ -44,7 +40,7 @@ class NavigationIntent extends Intent { class NavigationAction extends Action { @override invoke(intent) { - intent.router.go(intent.path); + intent.router.navigateNamed(intent.path); return null; } } @@ -52,32 +48,49 @@ class NavigationAction extends Action { enum HomeTabs { browse, search, - library, + lyrics, + userPlaylists, + userArtists, + userAlbums, + userLocalLibrary, + userDownloads, } class HomeTabIntent extends Intent { - final WidgetRef ref; + final AppRouter router; final HomeTabs tab; - const HomeTabIntent(this.ref, {required this.tab}); + const HomeTabIntent(this.router, {required this.tab}); } class HomeTabAction extends Action { @override invoke(intent) { - final router = intent.ref.read(routerProvider); + final router = intent.router; switch (intent.tab) { case HomeTabs.browse: - router.goNamed(HomePage.name); + router.navigate(const HomeRoute()); break; case HomeTabs.search: - router.goNamed(SearchPage.name); - break; - case HomeTabs.library: - router.goNamed(LibraryPage.name); + router.navigate(const SearchRoute()); break; case HomeTabs.lyrics: - router.goNamed(LyricsPage.name); + router.navigate(const LyricsRoute()); + break; + case HomeTabs.userPlaylists: + router.navigate(const UserPlaylistsRoute()); + break; + case HomeTabs.userArtists: + router.navigate(const UserArtistsRoute()); + break; + case HomeTabs.userAlbums: + router.navigate(const UserAlbumsRoute()); + break; + case HomeTabs.userLocalLibrary: + router.navigate(const UserLocalLibraryRoute()); + break; + case HomeTabs.userDownloads: + router.navigate(const UserDownloadsRoute()); break; } return null; diff --git a/lib/collections/language_codes.dart b/lib/collections/language_codes.dart index 44da6ee6..b5d3f7c8 100644 --- a/lib/collections/language_codes.dart +++ b/lib/collections/language_codes.dart @@ -133,10 +133,14 @@ abstract class LanguageLocals { // name: "Chichewa", // nativeName: "chiCheŵa", // ), - "zh": const ISOLanguageName( + "zh_CN": const ISOLanguageName( name: "Simplified Chinese", nativeName: "简体中文", ), + "zh_TW": const ISOLanguageName( + name: "Traditional Chinese", + nativeName: "繁體中文(台灣)", + ), // "cv": const ISOLanguageName( // name: "Chuvash", // nativeName: "чӑваш чӗлхи", @@ -625,10 +629,10 @@ abstract class LanguageLocals { // name: "Swedish", // nativeName: "svenska", // ), - // "ta": const ISOLanguageName( - // name: "Tamil", - // nativeName: "தமிழ்", - // ), + "ta": const ISOLanguageName( + name: "Tamil", + nativeName: "தமிழ்", + ), // "te": const ISOLanguageName( // name: "Telugu", // nativeName: "తెలుగు", @@ -653,10 +657,10 @@ abstract class LanguageLocals { // name: "Turkmen", // nativeName: "Türkmen, Түркмен", // ), - // "tl": const ISOLanguageName( - // name: "Tagalog", - // nativeName: "Wikang Tagalog, ᜏᜒᜃᜅ᜔ ᜆᜄᜎᜓᜄ᜔", - // ), + "tl": const ISOLanguageName( + name: "Tagalog", + nativeName: "Wikang Tagalog", + ), // "tn": const ISOLanguageName( // name: "Tswana", // nativeName: "Setswana", @@ -747,9 +751,13 @@ abstract class LanguageLocals { // ) }; - static ISOLanguageName getDisplayLanguage(key) { + static ISOLanguageName getDisplayLanguage(String key, String? countryCode) { if (isoLangs.containsKey(key)) { return isoLangs[key]!; + } else if (countryCode != null && + countryCode.isNotEmpty && + isoLangs.containsKey("${key}_$countryCode")) { + return isoLangs["${key}_$countryCode"]!; } else { throw Exception("Language key incorrect"); } diff --git a/lib/collections/spotify_markets.dart b/lib/collections/markets.dart similarity index 98% rename from lib/collections/spotify_markets.dart rename to lib/collections/markets.dart index 514b3f0b..8398c662 100644 --- a/lib/collections/spotify_markets.dart +++ b/lib/collections/markets.dart @@ -1,8 +1,8 @@ // Country Codes contributed by momobobe -import 'package:spotify/spotify.dart'; +import 'package:spotube/models/metadata/market.dart'; -final spotifyMarkets = [ +final marketsMap = [ (Market.AL, "Albania (AL)"), (Market.DZ, "Algeria (DZ)"), (Market.AD, "Andorra (AD)"), diff --git a/lib/collections/routes.dart b/lib/collections/routes.dart index a0380e29..4dcd9657 100644 --- a/lib/collections/routes.dart +++ b/lib/collections/routes.dart @@ -1,329 +1,227 @@ -import 'package:flutter/foundation.dart' hide Category; -import 'package:flutter/widgets.dart'; -import 'package:go_router/go_router.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/foundation.dart'; +import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart' hide Search; -import 'package:spotube/models/spotify/recommendation_seeds.dart'; -import 'package:spotube/pages/album/album.dart'; -import 'package:spotube/pages/connect/connect.dart'; -import 'package:spotube/pages/connect/control/control.dart'; -import 'package:spotube/pages/getting_started/getting_started.dart'; -import 'package:spotube/pages/home/feed/feed_section.dart'; -import 'package:spotube/pages/home/genres/genre_playlists.dart'; -import 'package:spotube/pages/home/genres/genres.dart'; -import 'package:spotube/pages/home/home.dart'; -import 'package:spotube/pages/lastfm_login/lastfm_login.dart'; -import 'package:spotube/pages/library/local_folder.dart'; -import 'package:spotube/pages/library/playlist_generate/playlist_generate.dart'; -import 'package:spotube/pages/library/playlist_generate/playlist_generate_result.dart'; -import 'package:spotube/pages/lyrics/mini_lyrics.dart'; -import 'package:spotube/pages/playlist/liked_playlist.dart'; -import 'package:spotube/pages/playlist/playlist.dart'; -import 'package:spotube/pages/profile/profile.dart'; -import 'package:spotube/pages/search/search.dart'; -import 'package:spotube/pages/settings/blacklist.dart'; -import 'package:spotube/pages/settings/about.dart'; -import 'package:spotube/pages/settings/logs.dart'; -import 'package:spotube/pages/stats/albums/albums.dart'; -import 'package:spotube/pages/stats/artists/artists.dart'; -import 'package:spotube/pages/stats/fees/fees.dart'; -import 'package:spotube/pages/stats/minutes/minutes.dart'; -import 'package:spotube/pages/stats/playlists/playlists.dart'; -import 'package:spotube/pages/stats/stats.dart'; -import 'package:spotube/pages/stats/streams/streams.dart'; -import 'package:spotube/pages/track/track.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; import 'package:spotube/services/kv_store/kv_store.dart'; -import 'package:spotube/components/spotube_page_route.dart'; -import 'package:spotube/pages/artist/artist.dart'; -import 'package:spotube/pages/library/library.dart'; -import 'package:spotube/pages/lyrics/lyrics.dart'; -import 'package:spotube/pages/root/root_app.dart'; -import 'package:spotube/pages/settings/settings.dart'; -import 'package:spotube/pages/mobile_login/mobile_login.dart'; final rootNavigatorKey = GlobalKey(); -final shellRouteNavigatorKey = GlobalKey(); -final routerProvider = Provider((ref) { - return GoRouter( - navigatorKey: rootNavigatorKey, - routes: [ - ShellRoute( - navigatorKey: shellRouteNavigatorKey, - builder: (context, state, child) => RootApp(child: child), - routes: [ - GoRoute( - path: "/", - name: HomePage.name, - redirect: (context, state) async { - final auth = await ref.read(authenticationProvider.future); - if (auth == null && !KVStoreService.doneGettingStarted) { - return "/getting-started"; - } +@AutoRouterConfig(replaceInRouteName: 'Screen|Page,Route') +class AppRouter extends RootStackRouter { + final WidgetRef ref; - return null; - }, - pageBuilder: (context, state) => - const SpotubePage(child: HomePage()), - routes: [ - GoRoute( - path: "genres", - name: GenrePage.name, - pageBuilder: (context, state) => - const SpotubePage(child: GenrePage()), - ), - GoRoute( - path: "genre/:categoryId", - name: GenrePlaylistsPage.name, - pageBuilder: (context, state) => SpotubePage( - child: GenrePlaylistsPage( - category: state.extra as Category, - ), - ), - ), - GoRoute( - path: "feeds/:feedId", - name: HomeFeedSectionPage.name, - pageBuilder: (context, state) => SpotubePage( - child: HomeFeedSectionPage( - sectionUri: state.pathParameters["feedId"] as String, - ), - ), - ) - ], - ), - GoRoute( - path: "/search", - name: SearchPage.name, - pageBuilder: (context, state) => - const SpotubePage(child: SearchPage()), - ), - GoRoute( - path: "/library", - name: LibraryPage.name, - pageBuilder: (context, state) => - const SpotubePage(child: LibraryPage()), - routes: [ - GoRoute( - path: "generate", - name: PlaylistGeneratorPage.name, - pageBuilder: (context, state) => - const SpotubePage(child: PlaylistGeneratorPage()), - routes: [ - GoRoute( - path: "result", - name: PlaylistGenerateResultPage.name, - pageBuilder: (context, state) => SpotubePage( - child: PlaylistGenerateResultPage( - state: state.extra as GeneratePlaylistProviderInput, - ), - ), - ) - ], - ), - GoRoute( - path: "local", - name: LocalLibraryPage.name, - pageBuilder: (context, state) { - assert(state.extra is String); - return SpotubePage( - child: LocalLibraryPage( - state.extra as String, - isDownloads: - state.uri.queryParameters["downloads"] != null, - isCache: state.uri.queryParameters["cache"] != null, - ), - ); + AppRouter(this.ref) : super(navigatorKey: rootNavigatorKey); + + @override + List get routes => [ + AutoRoute( + page: RootAppRoute.page, + path: "/", + initial: true, + children: [ + AutoRoute( + path: "home", + page: HomeRoute.page, + initial: true, + guards: [ + AutoRouteGuardCallback( + (resolver, router) async { + final authenticated = await ref + .read(metadataPluginAuthenticatedProvider.future); + + if (!authenticated && !KVStoreService.doneGettingStarted) { + resolver.redirect(const GettingStartedRoute()); + } else { + resolver.next(true); + } }, ), - ]), - GoRoute( - path: "/lyrics", - name: LyricsPage.name, - pageBuilder: (context, state) => - const SpotubePage(child: LyricsPage()), - ), - GoRoute( - path: "/settings", - name: SettingsPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: SettingsPage(), + ], ), - routes: [ - GoRoute( - path: "blacklist", - name: BlackListPage.name, - pageBuilder: (context, state) => SpotubeSlidePage( - child: const BlackListPage(), - ), - ), - if (!kIsWeb) - GoRoute( - path: "logs", - name: LogsPage.name, - pageBuilder: (context, state) => SpotubeSlidePage( - child: const LogsPage(), - ), - ), - GoRoute( - path: "about", - name: AboutSpotube.name, - pageBuilder: (context, state) => SpotubeSlidePage( - child: const AboutSpotube(), - ), - ), - ], - ), - GoRoute( - path: "/album/:id", - name: AlbumPage.name, - pageBuilder: (context, state) { - assert(state.extra is AlbumSimple); - return SpotubePage( - child: AlbumPage(album: state.extra as AlbumSimple), - ); - }, - ), - GoRoute( - path: "/artist/:id", - name: ArtistPage.name, - pageBuilder: (context, state) { - assert(state.pathParameters["id"] != null); - return SpotubePage( - child: ArtistPage(state.pathParameters["id"]!)); - }, - ), - GoRoute( - path: "/playlist/:id", - name: PlaylistPage.name, - pageBuilder: (context, state) { - assert(state.extra is PlaylistSimple); - return SpotubePage( - child: state.pathParameters["id"] == "user-liked-tracks" - ? LikedPlaylistPage(playlist: state.extra as PlaylistSimple) - : PlaylistPage(playlist: state.extra as PlaylistSimple), - ); - }, - ), - GoRoute( - path: "/track/:id", - name: TrackPage.name, - pageBuilder: (context, state) { - final id = state.pathParameters["id"]!; - return SpotubePage( - child: TrackPage(trackId: id), - ); - }, - ), - GoRoute( - path: "/connect", - name: ConnectPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: ConnectPage(), + AutoRoute( + path: "home/sections/:sectionId", + page: HomeBrowseSectionItemsRoute.page, ), - routes: [ - GoRoute( - path: "control", - name: ConnectControlPage.name, - pageBuilder: (context, state) { - return const SpotubePage( - child: ConnectControlPage(), - ); - }, - ) - ], - ), - GoRoute( - path: "/profile", - name: ProfilePage.name, - pageBuilder: (context, state) => - const SpotubePage(child: ProfilePage()), - ), - GoRoute( - path: "/stats", - name: StatsPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsPage(), + AutoRoute( + path: "search", + page: SearchRoute.page, ), - routes: [ - GoRoute( - path: "minutes", - name: StatsMinutesPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsMinutesPage(), + AutoRoute( + path: "library", + page: LibraryRoute.page, + children: [ + AutoRoute( + path: "playlists", + page: UserPlaylistsRoute.page, ), - ), - GoRoute( - path: "streams", - name: StatsStreamsPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsStreamsPage(), + AutoRoute( + path: "artists", + page: UserArtistsRoute.page, ), - ), - GoRoute( - path: "fees", - name: StatsStreamFeesPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsStreamFeesPage(), + AutoRoute( + path: "albums", + page: UserAlbumsRoute.page, ), - ), - GoRoute( - path: "artists", - name: StatsArtistsPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsArtistsPage(), + AutoRoute( + path: "local", + page: UserLocalLibraryRoute.page, ), - ), - GoRoute( - path: "albums", - name: StatsAlbumsPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsAlbumsPage(), + AutoRoute( + path: "downloads", + page: UserDownloadsRoute.page, ), + ], + ), + AutoRoute( + path: "local/folder", + page: LocalLibraryRoute.page, + // parentNavigatorKey: shellRouteNavigatorKey, + ), + AutoRoute( + path: "lyrics", + page: LyricsRoute.page, + ), + AutoRoute( + path: "settings", + page: SettingsRoute.page, + ), + AutoRoute( + path: "settings/metadata-provider", + page: SettingsMetadataProviderRoute.page, + ), + AutoRoute( + path: "settings/metadata-provider/metadata-form", + page: SettingsMetadataProviderFormRoute.page, + ), + AutoRoute( + path: "settings/blacklist", + page: BlackListRoute.page, + ), + if (!kIsWeb) + AutoRoute( + path: "settings/logs", + page: LogsRoute.page, ), - GoRoute( - path: "playlists", - name: StatsPlaylistsPage.name, - pageBuilder: (context, state) => const SpotubePage( - child: StatsPlaylistsPage(), + AutoRoute( + path: "settings/about", + page: AboutSpotubeRoute.page, + ), + AutoRoute( + path: "settings/scrobbling", + page: SettingsScrobblingRoute.page, + ), + AutoRoute( + path: "album/:id", + page: AlbumRoute.page, + ), + AutoRoute( + path: "artist/:id", + page: ArtistRoute.page, + ), + AutoRoute( + path: "liked-tracks", + page: LikedPlaylistRoute.page, + ), + AutoRoute( + path: "playlist/:id", + page: PlaylistRoute.page, + guards: [ + AutoRouteGuard.redirect( + (resolver) { + final PlaylistRouteArgs(:id, :playlist) = + resolver.route.args as PlaylistRouteArgs; + if (id == "user-liked-tracks") { + return LikedPlaylistRoute(playlist: playlist); + } + + return null; + }, ), - ), - ], - ) - ], - ), - GoRoute( - path: "/mini-player", - name: MiniLyricsPage.name, - parentNavigatorKey: rootNavigatorKey, - pageBuilder: (context, state) => SpotubePage( - child: MiniLyricsPage(prevSize: state.extra as Size), + ], + ), + AutoRoute( + path: "track/:id", + page: TrackRoute.page, + ), + AutoRoute( + path: "connect", + page: ConnectRoute.page, + ), + AutoRoute( + path: "connect/control", + page: ConnectControlRoute.page, + ), + AutoRoute( + path: "profile", + page: ProfileRoute.page, + ), + AutoRoute( + path: "stats", + page: StatsRoute.page, + ), + AutoRoute( + path: "stats/minutes", + page: StatsMinutesRoute.page, + ), + AutoRoute( + path: "stats/streams", + page: StatsStreamsRoute.page, + ), + AutoRoute( + path: "stats/fees", + page: StatsStreamFeesRoute.page, + ), + AutoRoute( + path: "stats/artists", + page: StatsArtistsRoute.page, + ), + AutoRoute( + path: "stats/albums", + page: StatsAlbumsRoute.page, + ), + AutoRoute( + path: "stats/playlists", + page: StatsPlaylistsRoute.page, + ), + ], ), - ), - GoRoute( - path: "/getting-started", - name: GettingStarting.name, - parentNavigatorKey: rootNavigatorKey, - pageBuilder: (context, state) => const SpotubePage( - child: GettingStarting(), + CustomRoute( + transitionsBuilder: TransitionsBuilders.slideBottom, + durationInMilliseconds: 200, + reverseDurationInMilliseconds: 200, + path: "/player/queue", + page: PlayerQueueRoute.page, ), - ), - GoRoute( - path: "/login", - name: WebViewLogin.name, - parentNavigatorKey: rootNavigatorKey, - pageBuilder: (context, state) => const SpotubePage( - child: WebViewLogin(), + CustomRoute( + transitionsBuilder: TransitionsBuilders.slideBottom, + durationInMilliseconds: 200, + reverseDurationInMilliseconds: 200, + path: "/player/sources", + page: PlayerTrackSourcesRoute.page, ), - ), - GoRoute( - path: "/lastfm-login", - name: LastFMLoginPage.name, - parentNavigatorKey: rootNavigatorKey, - pageBuilder: (context, state) => - const SpotubePage(child: LastFMLoginPage()), - ), - ], - ); -}); + CustomRoute( + transitionsBuilder: TransitionsBuilders.slideBottom, + durationInMilliseconds: 200, + reverseDurationInMilliseconds: 200, + path: "/player/lyrics", + page: PlayerLyricsRoute.page, + ), + AutoRoute( + path: "/mini-player", + page: MiniLyricsRoute.page, + // parentNavigatorKey: rootNavigatorKey, + ), + AutoRoute( + path: "/getting-started", + page: GettingStartedRoute.page, + // parentNavigatorKey: rootNavigatorKey, + ), + AutoRoute( + path: "/lastfm-login", + page: LastFMLoginRoute.page, + // parentNavigatorKey: rootNavigatorKey, + ), + ]; +} diff --git a/lib/collections/routes.gr.dart b/lib/collections/routes.gr.dart new file mode 100644 index 00000000..f5ff24bf --- /dev/null +++ b/lib/collections/routes.gr.dart @@ -0,0 +1,960 @@ +// dart format width=80 +// GENERATED CODE - DO NOT MODIFY BY HAND + +// ************************************************************************** +// AutoRouterGenerator +// ************************************************************************** + +// ignore_for_file: type=lint +// coverage:ignore-file + +// ignore_for_file: no_leading_underscores_for_library_prefixes +import 'package:auto_route/auto_route.dart' as _i41; +import 'package:flutter/material.dart' as _i42; +import 'package:shadcn_flutter/shadcn_flutter.dart' as _i44; +import 'package:spotube/models/metadata/metadata.dart' as _i43; +import 'package:spotube/pages/album/album.dart' as _i2; +import 'package:spotube/pages/artist/artist.dart' as _i3; +import 'package:spotube/pages/connect/connect.dart' as _i6; +import 'package:spotube/pages/connect/control/control.dart' as _i5; +import 'package:spotube/pages/getting_started/getting_started.dart' as _i7; +import 'package:spotube/pages/home/home.dart' as _i9; +import 'package:spotube/pages/home/sections/section_items.dart' as _i8; +import 'package:spotube/pages/lastfm_login/lastfm_login.dart' as _i10; +import 'package:spotube/pages/library/library.dart' as _i11; +import 'package:spotube/pages/library/user_albums.dart' as _i36; +import 'package:spotube/pages/library/user_artists.dart' as _i37; +import 'package:spotube/pages/library/user_downloads.dart' as _i38; +import 'package:spotube/pages/library/user_local_tracks/local_folder.dart' + as _i13; +import 'package:spotube/pages/library/user_local_tracks/user_local_tracks.dart' + as _i39; +import 'package:spotube/pages/library/user_playlists.dart' as _i40; +import 'package:spotube/pages/lyrics/lyrics.dart' as _i15; +import 'package:spotube/pages/lyrics/mini_lyrics.dart' as _i16; +import 'package:spotube/pages/player/lyrics.dart' as _i17; +import 'package:spotube/pages/player/queue.dart' as _i18; +import 'package:spotube/pages/player/sources.dart' as _i19; +import 'package:spotube/pages/playlist/liked_playlist.dart' as _i12; +import 'package:spotube/pages/playlist/playlist.dart' as _i20; +import 'package:spotube/pages/profile/profile.dart' as _i21; +import 'package:spotube/pages/root/root_app.dart' as _i22; +import 'package:spotube/pages/search/search.dart' as _i23; +import 'package:spotube/pages/settings/about.dart' as _i1; +import 'package:spotube/pages/settings/blacklist.dart' as _i4; +import 'package:spotube/pages/settings/logs.dart' as _i14; +import 'package:spotube/pages/settings/metadata/metadata_form.dart' as _i24; +import 'package:spotube/pages/settings/metadata_plugins.dart' as _i25; +import 'package:spotube/pages/settings/scrobbling/scrobbling.dart' as _i27; +import 'package:spotube/pages/settings/settings.dart' as _i26; +import 'package:spotube/pages/stats/albums/albums.dart' as _i28; +import 'package:spotube/pages/stats/artists/artists.dart' as _i29; +import 'package:spotube/pages/stats/fees/fees.dart' as _i33; +import 'package:spotube/pages/stats/minutes/minutes.dart' as _i30; +import 'package:spotube/pages/stats/playlists/playlists.dart' as _i32; +import 'package:spotube/pages/stats/stats.dart' as _i31; +import 'package:spotube/pages/stats/streams/streams.dart' as _i34; +import 'package:spotube/pages/track/track.dart' as _i35; + +/// generated route for +/// [_i1.AboutSpotubePage] +class AboutSpotubeRoute extends _i41.PageRouteInfo { + const AboutSpotubeRoute({List<_i41.PageRouteInfo>? children}) + : super(AboutSpotubeRoute.name, initialChildren: children); + + static const String name = 'AboutSpotubeRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i1.AboutSpotubePage(); + }, + ); +} + +/// generated route for +/// [_i2.AlbumPage] +class AlbumRoute extends _i41.PageRouteInfo { + AlbumRoute({ + _i42.Key? key, + required String id, + required _i43.SpotubeSimpleAlbumObject album, + List<_i41.PageRouteInfo>? children, + }) : super( + AlbumRoute.name, + args: AlbumRouteArgs(key: key, id: id, album: album), + rawPathParams: {'id': id}, + initialChildren: children, + ); + + static const String name = 'AlbumRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i2.AlbumPage(key: args.key, id: args.id, album: args.album); + }, + ); +} + +class AlbumRouteArgs { + const AlbumRouteArgs({this.key, required this.id, required this.album}); + + final _i42.Key? key; + + final String id; + + final _i43.SpotubeSimpleAlbumObject album; + + @override + String toString() { + return 'AlbumRouteArgs{key: $key, id: $id, album: $album}'; + } +} + +/// generated route for +/// [_i3.ArtistPage] +class ArtistRoute extends _i41.PageRouteInfo { + ArtistRoute({ + required String artistId, + _i42.Key? key, + List<_i41.PageRouteInfo>? children, + }) : super( + ArtistRoute.name, + args: ArtistRouteArgs(artistId: artistId, key: key), + rawPathParams: {'id': artistId}, + initialChildren: children, + ); + + static const String name = 'ArtistRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + final args = data.argsAs( + orElse: () => ArtistRouteArgs(artistId: pathParams.getString('id')), + ); + return _i3.ArtistPage(args.artistId, key: args.key); + }, + ); +} + +class ArtistRouteArgs { + const ArtistRouteArgs({required this.artistId, this.key}); + + final String artistId; + + final _i42.Key? key; + + @override + String toString() { + return 'ArtistRouteArgs{artistId: $artistId, key: $key}'; + } +} + +/// generated route for +/// [_i4.BlackListPage] +class BlackListRoute extends _i41.PageRouteInfo { + const BlackListRoute({List<_i41.PageRouteInfo>? children}) + : super(BlackListRoute.name, initialChildren: children); + + static const String name = 'BlackListRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i4.BlackListPage(); + }, + ); +} + +/// generated route for +/// [_i5.ConnectControlPage] +class ConnectControlRoute extends _i41.PageRouteInfo { + const ConnectControlRoute({List<_i41.PageRouteInfo>? children}) + : super(ConnectControlRoute.name, initialChildren: children); + + static const String name = 'ConnectControlRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i5.ConnectControlPage(); + }, + ); +} + +/// generated route for +/// [_i6.ConnectPage] +class ConnectRoute extends _i41.PageRouteInfo { + const ConnectRoute({List<_i41.PageRouteInfo>? children}) + : super(ConnectRoute.name, initialChildren: children); + + static const String name = 'ConnectRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i6.ConnectPage(); + }, + ); +} + +/// generated route for +/// [_i7.GettingStartedPage] +class GettingStartedRoute extends _i41.PageRouteInfo { + const GettingStartedRoute({List<_i41.PageRouteInfo>? children}) + : super(GettingStartedRoute.name, initialChildren: children); + + static const String name = 'GettingStartedRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i7.GettingStartedPage(); + }, + ); +} + +/// generated route for +/// [_i8.HomeBrowseSectionItemsPage] +class HomeBrowseSectionItemsRoute + extends _i41.PageRouteInfo { + HomeBrowseSectionItemsRoute({ + _i44.Key? key, + required String sectionId, + required _i43.SpotubeBrowseSectionObject section, + List<_i41.PageRouteInfo>? children, + }) : super( + HomeBrowseSectionItemsRoute.name, + args: HomeBrowseSectionItemsRouteArgs( + key: key, + sectionId: sectionId, + section: section, + ), + rawPathParams: {'sectionId': sectionId}, + initialChildren: children, + ); + + static const String name = 'HomeBrowseSectionItemsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i8.HomeBrowseSectionItemsPage( + key: args.key, + sectionId: args.sectionId, + section: args.section, + ); + }, + ); +} + +class HomeBrowseSectionItemsRouteArgs { + const HomeBrowseSectionItemsRouteArgs({ + this.key, + required this.sectionId, + required this.section, + }); + + final _i44.Key? key; + + final String sectionId; + + final _i43.SpotubeBrowseSectionObject section; + + @override + String toString() { + return 'HomeBrowseSectionItemsRouteArgs{key: $key, sectionId: $sectionId, section: $section}'; + } +} + +/// generated route for +/// [_i9.HomePage] +class HomeRoute extends _i41.PageRouteInfo { + const HomeRoute({List<_i41.PageRouteInfo>? children}) + : super(HomeRoute.name, initialChildren: children); + + static const String name = 'HomeRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i9.HomePage(); + }, + ); +} + +/// generated route for +/// [_i10.LastFMLoginPage] +class LastFMLoginRoute extends _i41.PageRouteInfo { + const LastFMLoginRoute({List<_i41.PageRouteInfo>? children}) + : super(LastFMLoginRoute.name, initialChildren: children); + + static const String name = 'LastFMLoginRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i10.LastFMLoginPage(); + }, + ); +} + +/// generated route for +/// [_i11.LibraryPage] +class LibraryRoute extends _i41.PageRouteInfo { + const LibraryRoute({List<_i41.PageRouteInfo>? children}) + : super(LibraryRoute.name, initialChildren: children); + + static const String name = 'LibraryRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i11.LibraryPage(); + }, + ); +} + +/// generated route for +/// [_i12.LikedPlaylistPage] +class LikedPlaylistRoute extends _i41.PageRouteInfo { + LikedPlaylistRoute({ + _i42.Key? key, + required _i43.SpotubeSimplePlaylistObject playlist, + List<_i41.PageRouteInfo>? children, + }) : super( + LikedPlaylistRoute.name, + args: LikedPlaylistRouteArgs(key: key, playlist: playlist), + initialChildren: children, + ); + + static const String name = 'LikedPlaylistRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i12.LikedPlaylistPage(key: args.key, playlist: args.playlist); + }, + ); +} + +class LikedPlaylistRouteArgs { + const LikedPlaylistRouteArgs({this.key, required this.playlist}); + + final _i42.Key? key; + + final _i43.SpotubeSimplePlaylistObject playlist; + + @override + String toString() { + return 'LikedPlaylistRouteArgs{key: $key, playlist: $playlist}'; + } +} + +/// generated route for +/// [_i13.LocalLibraryPage] +class LocalLibraryRoute extends _i41.PageRouteInfo { + LocalLibraryRoute({ + required String location, + _i42.Key? key, + bool isDownloads = false, + bool isCache = false, + List<_i41.PageRouteInfo>? children, + }) : super( + LocalLibraryRoute.name, + args: LocalLibraryRouteArgs( + location: location, + key: key, + isDownloads: isDownloads, + isCache: isCache, + ), + initialChildren: children, + ); + + static const String name = 'LocalLibraryRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i13.LocalLibraryPage( + args.location, + key: args.key, + isDownloads: args.isDownloads, + isCache: args.isCache, + ); + }, + ); +} + +class LocalLibraryRouteArgs { + const LocalLibraryRouteArgs({ + required this.location, + this.key, + this.isDownloads = false, + this.isCache = false, + }); + + final String location; + + final _i42.Key? key; + + final bool isDownloads; + + final bool isCache; + + @override + String toString() { + return 'LocalLibraryRouteArgs{location: $location, key: $key, isDownloads: $isDownloads, isCache: $isCache}'; + } +} + +/// generated route for +/// [_i14.LogsPage] +class LogsRoute extends _i41.PageRouteInfo { + const LogsRoute({List<_i41.PageRouteInfo>? children}) + : super(LogsRoute.name, initialChildren: children); + + static const String name = 'LogsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i14.LogsPage(); + }, + ); +} + +/// generated route for +/// [_i15.LyricsPage] +class LyricsRoute extends _i41.PageRouteInfo { + const LyricsRoute({List<_i41.PageRouteInfo>? children}) + : super(LyricsRoute.name, initialChildren: children); + + static const String name = 'LyricsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i15.LyricsPage(); + }, + ); +} + +/// generated route for +/// [_i16.MiniLyricsPage] +class MiniLyricsRoute extends _i41.PageRouteInfo { + MiniLyricsRoute({ + _i44.Key? key, + required _i44.Size prevSize, + List<_i41.PageRouteInfo>? children, + }) : super( + MiniLyricsRoute.name, + args: MiniLyricsRouteArgs(key: key, prevSize: prevSize), + initialChildren: children, + ); + + static const String name = 'MiniLyricsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i16.MiniLyricsPage(key: args.key, prevSize: args.prevSize); + }, + ); +} + +class MiniLyricsRouteArgs { + const MiniLyricsRouteArgs({this.key, required this.prevSize}); + + final _i44.Key? key; + + final _i44.Size prevSize; + + @override + String toString() { + return 'MiniLyricsRouteArgs{key: $key, prevSize: $prevSize}'; + } +} + +/// generated route for +/// [_i17.PlayerLyricsPage] +class PlayerLyricsRoute extends _i41.PageRouteInfo { + const PlayerLyricsRoute({List<_i41.PageRouteInfo>? children}) + : super(PlayerLyricsRoute.name, initialChildren: children); + + static const String name = 'PlayerLyricsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i17.PlayerLyricsPage(); + }, + ); +} + +/// generated route for +/// [_i18.PlayerQueuePage] +class PlayerQueueRoute extends _i41.PageRouteInfo { + const PlayerQueueRoute({List<_i41.PageRouteInfo>? children}) + : super(PlayerQueueRoute.name, initialChildren: children); + + static const String name = 'PlayerQueueRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i18.PlayerQueuePage(); + }, + ); +} + +/// generated route for +/// [_i19.PlayerTrackSourcesPage] +class PlayerTrackSourcesRoute extends _i41.PageRouteInfo { + const PlayerTrackSourcesRoute({List<_i41.PageRouteInfo>? children}) + : super(PlayerTrackSourcesRoute.name, initialChildren: children); + + static const String name = 'PlayerTrackSourcesRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i19.PlayerTrackSourcesPage(); + }, + ); +} + +/// generated route for +/// [_i20.PlaylistPage] +class PlaylistRoute extends _i41.PageRouteInfo { + PlaylistRoute({ + _i42.Key? key, + required String id, + required _i43.SpotubeSimplePlaylistObject playlist, + List<_i41.PageRouteInfo>? children, + }) : super( + PlaylistRoute.name, + args: PlaylistRouteArgs(key: key, id: id, playlist: playlist), + rawPathParams: {'id': id}, + initialChildren: children, + ); + + static const String name = 'PlaylistRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i20.PlaylistPage( + key: args.key, + id: args.id, + playlist: args.playlist, + ); + }, + ); +} + +class PlaylistRouteArgs { + const PlaylistRouteArgs({this.key, required this.id, required this.playlist}); + + final _i42.Key? key; + + final String id; + + final _i43.SpotubeSimplePlaylistObject playlist; + + @override + String toString() { + return 'PlaylistRouteArgs{key: $key, id: $id, playlist: $playlist}'; + } +} + +/// generated route for +/// [_i21.ProfilePage] +class ProfileRoute extends _i41.PageRouteInfo { + const ProfileRoute({List<_i41.PageRouteInfo>? children}) + : super(ProfileRoute.name, initialChildren: children); + + static const String name = 'ProfileRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i21.ProfilePage(); + }, + ); +} + +/// generated route for +/// [_i22.RootAppPage] +class RootAppRoute extends _i41.PageRouteInfo { + const RootAppRoute({List<_i41.PageRouteInfo>? children}) + : super(RootAppRoute.name, initialChildren: children); + + static const String name = 'RootAppRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i22.RootAppPage(); + }, + ); +} + +/// generated route for +/// [_i23.SearchPage] +class SearchRoute extends _i41.PageRouteInfo { + const SearchRoute({List<_i41.PageRouteInfo>? children}) + : super(SearchRoute.name, initialChildren: children); + + static const String name = 'SearchRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i23.SearchPage(); + }, + ); +} + +/// generated route for +/// [_i24.SettingsMetadataProviderFormPage] +class SettingsMetadataProviderFormRoute + extends _i41.PageRouteInfo { + SettingsMetadataProviderFormRoute({ + _i44.Key? key, + required String title, + required List<_i43.MetadataFormFieldObject> fields, + List<_i41.PageRouteInfo>? children, + }) : super( + SettingsMetadataProviderFormRoute.name, + args: SettingsMetadataProviderFormRouteArgs( + key: key, + title: title, + fields: fields, + ), + initialChildren: children, + ); + + static const String name = 'SettingsMetadataProviderFormRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final args = data.argsAs(); + return _i24.SettingsMetadataProviderFormPage( + key: args.key, + title: args.title, + fields: args.fields, + ); + }, + ); +} + +class SettingsMetadataProviderFormRouteArgs { + const SettingsMetadataProviderFormRouteArgs({ + this.key, + required this.title, + required this.fields, + }); + + final _i44.Key? key; + + final String title; + + final List<_i43.MetadataFormFieldObject> fields; + + @override + String toString() { + return 'SettingsMetadataProviderFormRouteArgs{key: $key, title: $title, fields: $fields}'; + } +} + +/// generated route for +/// [_i25.SettingsMetadataProviderPage] +class SettingsMetadataProviderRoute extends _i41.PageRouteInfo { + const SettingsMetadataProviderRoute({List<_i41.PageRouteInfo>? children}) + : super(SettingsMetadataProviderRoute.name, initialChildren: children); + + static const String name = 'SettingsMetadataProviderRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i25.SettingsMetadataProviderPage(); + }, + ); +} + +/// generated route for +/// [_i26.SettingsPage] +class SettingsRoute extends _i41.PageRouteInfo { + const SettingsRoute({List<_i41.PageRouteInfo>? children}) + : super(SettingsRoute.name, initialChildren: children); + + static const String name = 'SettingsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i26.SettingsPage(); + }, + ); +} + +/// generated route for +/// [_i27.SettingsScrobblingPage] +class SettingsScrobblingRoute extends _i41.PageRouteInfo { + const SettingsScrobblingRoute({List<_i41.PageRouteInfo>? children}) + : super(SettingsScrobblingRoute.name, initialChildren: children); + + static const String name = 'SettingsScrobblingRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i27.SettingsScrobblingPage(); + }, + ); +} + +/// generated route for +/// [_i28.StatsAlbumsPage] +class StatsAlbumsRoute extends _i41.PageRouteInfo { + const StatsAlbumsRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsAlbumsRoute.name, initialChildren: children); + + static const String name = 'StatsAlbumsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i28.StatsAlbumsPage(); + }, + ); +} + +/// generated route for +/// [_i29.StatsArtistsPage] +class StatsArtistsRoute extends _i41.PageRouteInfo { + const StatsArtistsRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsArtistsRoute.name, initialChildren: children); + + static const String name = 'StatsArtistsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i29.StatsArtistsPage(); + }, + ); +} + +/// generated route for +/// [_i30.StatsMinutesPage] +class StatsMinutesRoute extends _i41.PageRouteInfo { + const StatsMinutesRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsMinutesRoute.name, initialChildren: children); + + static const String name = 'StatsMinutesRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i30.StatsMinutesPage(); + }, + ); +} + +/// generated route for +/// [_i31.StatsPage] +class StatsRoute extends _i41.PageRouteInfo { + const StatsRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsRoute.name, initialChildren: children); + + static const String name = 'StatsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i31.StatsPage(); + }, + ); +} + +/// generated route for +/// [_i32.StatsPlaylistsPage] +class StatsPlaylistsRoute extends _i41.PageRouteInfo { + const StatsPlaylistsRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsPlaylistsRoute.name, initialChildren: children); + + static const String name = 'StatsPlaylistsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i32.StatsPlaylistsPage(); + }, + ); +} + +/// generated route for +/// [_i33.StatsStreamFeesPage] +class StatsStreamFeesRoute extends _i41.PageRouteInfo { + const StatsStreamFeesRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsStreamFeesRoute.name, initialChildren: children); + + static const String name = 'StatsStreamFeesRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i33.StatsStreamFeesPage(); + }, + ); +} + +/// generated route for +/// [_i34.StatsStreamsPage] +class StatsStreamsRoute extends _i41.PageRouteInfo { + const StatsStreamsRoute({List<_i41.PageRouteInfo>? children}) + : super(StatsStreamsRoute.name, initialChildren: children); + + static const String name = 'StatsStreamsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i34.StatsStreamsPage(); + }, + ); +} + +/// generated route for +/// [_i35.TrackPage] +class TrackRoute extends _i41.PageRouteInfo { + TrackRoute({ + _i44.Key? key, + required String trackId, + List<_i41.PageRouteInfo>? children, + }) : super( + TrackRoute.name, + args: TrackRouteArgs(key: key, trackId: trackId), + rawPathParams: {'id': trackId}, + initialChildren: children, + ); + + static const String name = 'TrackRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + final pathParams = data.inheritedPathParams; + final args = data.argsAs( + orElse: () => TrackRouteArgs(trackId: pathParams.getString('id')), + ); + return _i35.TrackPage(key: args.key, trackId: args.trackId); + }, + ); +} + +class TrackRouteArgs { + const TrackRouteArgs({this.key, required this.trackId}); + + final _i44.Key? key; + + final String trackId; + + @override + String toString() { + return 'TrackRouteArgs{key: $key, trackId: $trackId}'; + } +} + +/// generated route for +/// [_i36.UserAlbumsPage] +class UserAlbumsRoute extends _i41.PageRouteInfo { + const UserAlbumsRoute({List<_i41.PageRouteInfo>? children}) + : super(UserAlbumsRoute.name, initialChildren: children); + + static const String name = 'UserAlbumsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i36.UserAlbumsPage(); + }, + ); +} + +/// generated route for +/// [_i37.UserArtistsPage] +class UserArtistsRoute extends _i41.PageRouteInfo { + const UserArtistsRoute({List<_i41.PageRouteInfo>? children}) + : super(UserArtistsRoute.name, initialChildren: children); + + static const String name = 'UserArtistsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i37.UserArtistsPage(); + }, + ); +} + +/// generated route for +/// [_i38.UserDownloadsPage] +class UserDownloadsRoute extends _i41.PageRouteInfo { + const UserDownloadsRoute({List<_i41.PageRouteInfo>? children}) + : super(UserDownloadsRoute.name, initialChildren: children); + + static const String name = 'UserDownloadsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i38.UserDownloadsPage(); + }, + ); +} + +/// generated route for +/// [_i39.UserLocalLibraryPage] +class UserLocalLibraryRoute extends _i41.PageRouteInfo { + const UserLocalLibraryRoute({List<_i41.PageRouteInfo>? children}) + : super(UserLocalLibraryRoute.name, initialChildren: children); + + static const String name = 'UserLocalLibraryRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i39.UserLocalLibraryPage(); + }, + ); +} + +/// generated route for +/// [_i40.UserPlaylistsPage] +class UserPlaylistsRoute extends _i41.PageRouteInfo { + const UserPlaylistsRoute({List<_i41.PageRouteInfo>? children}) + : super(UserPlaylistsRoute.name, initialChildren: children); + + static const String name = 'UserPlaylistsRoute'; + + static _i41.PageInfo page = _i41.PageInfo( + name, + builder: (data) { + return const _i40.UserPlaylistsPage(); + }, + ); +} diff --git a/lib/collections/side_bar_tiles.dart b/lib/collections/side_bar_tiles.dart index 4f23c049..c647c9fb 100644 --- a/lib/collections/side_bar_tiles.dart +++ b/lib/collections/side_bar_tiles.dart @@ -1,81 +1,113 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; -import 'package:flutter_gen/gen_l10n/app_localizations.dart'; -import 'package:spotube/pages/home/home.dart'; -import 'package:spotube/pages/library/library.dart'; -import 'package:spotube/pages/lyrics/lyrics.dart'; -import 'package:spotube/pages/search/search.dart'; -import 'package:spotube/pages/stats/stats.dart'; +import 'package:spotube/l10n/l10n.dart'; class SideBarTiles { final IconData icon; final String title; final String id; - final String name; + final String pathPrefix; + final PageRouteInfo route; SideBarTiles({ required this.icon, required this.title, required this.id, - required this.name, + required this.route, + required this.pathPrefix, }); } List getSidebarTileList(AppLocalizations l10n) => [ SideBarTiles( - id: "browse", - name: HomePage.name, + id: "home", + pathPrefix: "/home", + route: const HomeRoute(), icon: SpotubeIcons.home, title: l10n.browse, ), SideBarTiles( id: "search", - name: SearchPage.name, + pathPrefix: "/search", + route: const SearchRoute(), icon: SpotubeIcons.search, title: l10n.search, ), - SideBarTiles( - id: "library", - name: LibraryPage.name, - icon: SpotubeIcons.library, - title: l10n.library, - ), SideBarTiles( id: "lyrics", - name: LyricsPage.name, + pathPrefix: "/lyrics", + route: const LyricsRoute(), icon: SpotubeIcons.music, title: l10n.lyrics, ), SideBarTiles( id: "stats", - name: StatsPage.name, + pathPrefix: "/stats", + route: const StatsRoute(), icon: SpotubeIcons.chart, title: l10n.stats, ), ]; +List getSidebarLibraryTileList(AppLocalizations l10n) => [ + SideBarTiles( + id: "playlists", + pathPrefix: "/library/playlists", + title: l10n.playlists, + route: const UserPlaylistsRoute(), + icon: SpotubeIcons.playlist, + ), + SideBarTiles( + id: "artists", + pathPrefix: "/library/artists", + title: l10n.artists, + route: const UserArtistsRoute(), + icon: SpotubeIcons.artist, + ), + SideBarTiles( + id: "albums", + pathPrefix: "/library/albums", + title: l10n.albums, + route: const UserAlbumsRoute(), + icon: SpotubeIcons.album, + ), + SideBarTiles( + id: "local_library", + pathPrefix: "/library/local", + title: l10n.local_library, + route: const UserLocalLibraryRoute(), + icon: SpotubeIcons.device, + ), + ]; + List getNavbarTileList(AppLocalizations l10n) => [ SideBarTiles( - id: "browse", - name: HomePage.name, + id: "home", + pathPrefix: "/home", + route: const HomeRoute(), icon: SpotubeIcons.home, title: l10n.browse, ), SideBarTiles( id: "search", - name: SearchPage.name, + pathPrefix: "/search", + route: const SearchRoute(), icon: SpotubeIcons.search, title: l10n.search, ), SideBarTiles( id: "library", - name: LibraryPage.name, + pathPrefix: "/library", + route: const UserPlaylistsRoute(), icon: SpotubeIcons.library, title: l10n.library, ), SideBarTiles( id: "stats", - name: StatsPage.name, + pathPrefix: "/stats", + route: const StatsRoute(), icon: SpotubeIcons.chart, title: l10n.stats, ), diff --git a/lib/collections/spotube_icons.dart b/lib/collections/spotube_icons.dart index 5c4df85f..99d9ff74 100644 --- a/lib/collections/spotube_icons.dart +++ b/lib/collections/spotube_icons.dart @@ -1,5 +1,5 @@ import 'package:fluentui_system_icons/fluentui_system_icons.dart'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_feather_icons/flutter_feather_icons.dart'; import 'package:simple_icons/simple_icons.dart'; @@ -37,6 +37,7 @@ abstract class SpotubeIcons { static const share = FeatherIcons.share2; static const playlistAdd = Icons.playlist_add_rounded; static const playlistRemove = Icons.playlist_remove_rounded; + static const playlist = Icons.playlist_play_rounded; static const trash = FeatherIcons.trash2; static const clock = FeatherIcons.clock; static const lyrics = Icons.lyrics_rounded; @@ -79,6 +80,7 @@ abstract class SpotubeIcons { static const hoverOff = Icons.back_hand_outlined; static const dragHandle = Icons.drag_indicator; static const lightning = Icons.flash_on_rounded; + static const lightningOutlined = FeatherIcons.zap; static const colorSync = FeatherIcons.activity; static const language = FeatherIcons.globe; static const error = FeatherIcons.alertTriangle; @@ -104,7 +106,6 @@ abstract class SpotubeIcons { static const file = FeatherIcons.file; static const stream = Icons.stream_rounded; static const lastFm = SimpleIcons.lastdotfm; - static const spotify = SimpleIcons.spotify; static const eye = FeatherIcons.eye; static const noEye = FeatherIcons.eyeOff; static const normalize = FeatherIcons.barChart2; @@ -127,4 +128,16 @@ abstract class SpotubeIcons { static const cache = FeatherIcons.hardDrive; static const export = Icons.file_open_outlined; static const delete = FeatherIcons.trash2; + static const open = FeatherIcons.externalLink; + static const radioChecked = Icons.radio_button_on_rounded; + static const radioUnchecked = Icons.radio_button_off_rounded; + static const grid = FeatherIcons.grid; + static const list = FeatherIcons.list; + static const device = FeatherIcons.smartphone; + static const engine = FeatherIcons.server; + static const extensions = Icons.extension_rounded; + static const message = FeatherIcons.send; + static const upload = FeatherIcons.uploadCloud; + static const plugin = Icons.extension_outlined; + static const warning = FeatherIcons.alertTriangle; } diff --git a/lib/components/adaptive/adaptive_list_tile.dart b/lib/components/adaptive/adaptive_list_tile.dart index 33df44c1..c6d00bd4 100644 --- a/lib/components/adaptive/adaptive_list_tile.dart +++ b/lib/components/adaptive/adaptive_list_tile.dart @@ -1,5 +1,6 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/constrains.dart'; class AdaptiveListTile extends HookWidget { @@ -24,41 +25,39 @@ class AdaptiveListTile extends HookWidget { Widget build(BuildContext context) { final mediaQuery = MediaQuery.of(context); - return ListTile( + return ButtonTile( title: title, subtitle: subtitle, trailing: breakOn ?? mediaQuery.smAndDown ? null : trailing?.call(context, null), leading: leading, - onTap: breakOn ?? mediaQuery.smAndDown - ? () { - onTap?.call(); - showDialog( - context: context, - barrierDismissible: true, - builder: (context) { - return StatefulBuilder(builder: (context, update) { - return AlertDialog( - title: title != null - ? Row( - crossAxisAlignment: CrossAxisAlignment.center, - children: [ - if (leading != null) ...[ - leading!, - const SizedBox(width: 5) - ], - Flexible(child: title!), - ], - ) - : Container(), - content: trailing?.call(context, update), - ); - }); - }, + enabled: breakOn ?? mediaQuery.smAndDown, + onPressed: () { + onTap?.call(); + showDialog( + context: context, + barrierDismissible: true, + builder: (context) { + return StatefulBuilder(builder: (context, update) { + return AlertDialog( + title: title != null + ? Row( + crossAxisAlignment: CrossAxisAlignment.center, + spacing: 5, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + if (leading != null) leading!, + Flexible(child: title!), + ], + ) + : const SizedBox.shrink(), + content: Center(child: trailing?.call(context, update)), ); - } - : null, + }); + }, + ); + }, ); } } diff --git a/lib/components/adaptive/adaptive_pop_sheet_list.dart b/lib/components/adaptive/adaptive_pop_sheet_list.dart index 97dc6132..6eba1148 100644 --- a/lib/components/adaptive/adaptive_pop_sheet_list.dart +++ b/lib/components/adaptive/adaptive_pop_sheet_list.dart @@ -1,235 +1,190 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/extensions/constrains.dart'; -_emptyCB() {} - -class PopSheetEntry extends ListTile { +class AdaptiveMenuButton extends MenuButton { final T? value; - const PopSheetEntry({ - this.value, + const AdaptiveMenuButton({ super.key, - super.leading, - super.title, - super.subtitle, + this.value, + required super.child, + super.subMenu, + super.onPressed, super.trailing, - super.isThreeLine = false, - super.dense, - super.visualDensity, - super.shape, - super.style, - super.selectedColor, - super.iconColor, - super.textColor, - super.titleTextStyle, - super.subtitleTextStyle, - super.leadingAndTrailingTextStyle, - super.contentPadding, + super.leading, super.enabled = true, - super.onTap = _emptyCB, - super.onLongPress, - super.onFocusChange, - super.mouseCursor, - super.selected = false, - super.focusColor, - super.hoverColor, - super.splashColor, super.focusNode, - super.autofocus = false, - super.tileColor, - super.selectedTileColor, - super.enableFeedback, - super.horizontalTitleGap, - super.minVerticalPadding, - super.minLeadingWidth, - super.titleAlignment, - }); + super.autoClose = true, + super.popoverController, + }) : assert( + value != null || onPressed != null, + 'Either value or onPressed must be provided', + ); } /// An adaptive widget that shows a [PopupMenuButton] when screen size is above /// or equal to 640px -/// In smaller screen, a [IconButton] with a [showModalBottomSheet] is shown +/// In smaller screen, a [IconButton] with a [openDrawer] is shown class AdaptivePopSheetList extends StatelessWidget { - final List> children; + final List> Function(BuildContext context) items; final Widget? icon; final Widget? child; final bool useRootNavigator; final List? headings; - final String? tooltip; + final String tooltip; final ValueChanged? onSelected; - final BorderRadius borderRadius; final Offset offset; + final AbstractButtonStyle variance; + const AdaptivePopSheetList({ super.key, - required this.children, + required this.items, this.icon, this.child, this.useRootNavigator = true, this.headings, this.onSelected, - this.borderRadius = const BorderRadius.all(Radius.circular(999)), - this.tooltip, + required this.tooltip, this.offset = Offset.zero, + this.variance = ButtonVariance.ghost, }) : assert( !(icon != null && child != null), 'Either icon or child must be provided', ); - Future showPopupMenu(BuildContext context, RelativeRect position) { + Future showDropdownMenu(BuildContext context, Offset position) async { final mediaQuery = MediaQuery.of(context); + List childrenModified(BuildContext context) => + items(context).map((s) { + if (s.onPressed == null) { + return MenuButton( + key: s.key, + autoClose: s.autoClose, + enabled: s.enabled, + leading: s.leading, + focusNode: s.focusNode, + onPressed: (context) { + if (s.value != null) { + onSelected?.call(s.value as T); + } + }, + popoverController: s.popoverController, + subMenu: s.subMenu, + trailing: s.trailing, + child: s.child, + ); + } + return s; + }).toList(); - return showMenu( - context: context, - useRootNavigator: useRootNavigator, - constraints: BoxConstraints( - maxHeight: mediaQuery.size.height * 0.6, - ), - position: position, - items: children - .map( - (item) => PopupMenuItem( - padding: EdgeInsets.zero, - enabled: false, - child: _AdaptivePopSheetListItem( - item: item, - onSelected: onSelected, - ), + if (mediaQuery.mdAndUp) { + await showDropdown( + context: context, + rootOverlay: useRootNavigator, + // heightConstraint: PopoverConstraint.anchorFixedSize, + // constraints: BoxConstraints( + // maxHeight: mediaQuery.size.height * 0.6, + // ), + position: position, + builder: (context) { + return WidgetStatesProvider.boundary( + child: DropdownMenu( + children: childrenModified(context), ), - ) - .toList(), + ); + }, + ).future; + return; + } + + await openDrawer( + context: context, + draggable: true, + showDragHandle: true, + position: OverlayPosition.bottom, + borderRadius: context.theme.borderRadiusMd, + transformBackdrop: false, + builder: (context) { + final children = childrenModified(context); + return ListView.builder( + itemCount: children.length, + shrinkWrap: true, + itemBuilder: (context, index) { + final data = children[index]; + + return Button( + enabled: data.enabled, + style: ButtonVariance.ghost.copyWith( + padding: (context, state, value) => const EdgeInsets.all(16), + ), + onPressed: () { + data.onPressed?.call(context); + if (data.autoClose) { + closeDrawer(context); + } + }, + leading: data.leading, + trailing: data.trailing, + alignment: Alignment.centerLeft, + child: data.child, + ); + }, + ); + }, ); } @override Widget build(BuildContext context) { final mediaQuery = MediaQuery.of(context); - final theme = Theme.of(context); if (mediaQuery.mdAndUp) { - return PopupMenuButton( - icon: icon, - tooltip: tooltip, - offset: offset, - child: child == null ? null : IgnorePointer(child: child), - itemBuilder: (context) => children - .map( - (item) => PopupMenuItem( - padding: EdgeInsets.zero, - enabled: false, - child: _AdaptivePopSheetListItem( - item: item, - onSelected: onSelected, - ), + return Tooltip( + tooltip: TooltipContainer( + child: Text(tooltip), + ).call, + child: IconButton( + variance: variance, + icon: icon ?? const Icon(SpotubeIcons.moreVertical), + onPressed: () { + final renderBox = context.findRenderObject() as RenderBox; + final position = RelativeRect.fromRect( + Rect.fromPoints( + renderBox.localToGlobal(Offset.zero, + ancestor: context.findRenderObject()), + renderBox.localToGlobal(renderBox.size.bottomRight(Offset.zero), + ancestor: context.findRenderObject()), ), - ) - .toList(), - ); - } - - void showSheet() { - showModalBottomSheet( - context: context, - useRootNavigator: useRootNavigator, - isScrollControlled: true, - showDragHandle: true, - constraints: BoxConstraints( - maxHeight: mediaQuery.size.height * 0.6, + Offset.zero & mediaQuery.size, + ); + final offset = Offset(position.left, position.top); + showDropdownMenu(context, offset); + }, ), - builder: (context) { - return Padding( - padding: const EdgeInsets.all(8.0).copyWith(top: 0), - child: DefaultTextStyle( - style: theme.textTheme.titleMedium!, - child: SingleChildScrollView( - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - if (headings != null) ...[ - ...headings!, - const SizedBox(height: 8), - Divider( - color: theme.colorScheme.primary, - thickness: 0.3, - endIndent: 16, - indent: 16, - ), - ], - ...children.map( - (item) => _AdaptivePopSheetListItem( - item: item, - onSelected: onSelected, - ), - ) - ], - ), - ), - ), - ); - }, ); } if (child != null) { return Tooltip( - message: tooltip ?? '', - child: InkWell( - onTap: showSheet, - borderRadius: borderRadius, + tooltip: TooltipContainer(child: Text(tooltip)).call, + child: Button( + onPressed: () => showDropdownMenu(context, Offset.zero), + style: variance, child: IgnorePointer(child: child), ), ); } - return IconButton( - icon: icon ?? const Icon(SpotubeIcons.moreVertical), - tooltip: tooltip, - style: theme.iconButtonTheme.style?.copyWith( - shape: WidgetStatePropertyAll( - RoundedRectangleBorder( - borderRadius: borderRadius, - ), - ), - ), - onPressed: showSheet, - ); - } -} - -class _AdaptivePopSheetListItem extends StatelessWidget { - final PopSheetEntry item; - final ValueChanged? onSelected; - const _AdaptivePopSheetListItem({ - super.key, - required this.item, - this.onSelected, - }); - - @override - Widget build(BuildContext context) { - final theme = Theme.of(context); - - return InkWell( - borderRadius: (theme.listTileTheme.shape as RoundedRectangleBorder?) - ?.borderRadius as BorderRadius? ?? - const BorderRadius.all(Radius.circular(10)), - onTap: !item.enabled - ? null - : () { - item.onTap?.call(); - if (item.value != null) { - Navigator.pop(context); - onSelected?.call(item.value as T); - } - }, - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 8), - child: IconTheme.merge( - data: const IconThemeData(opacity: 1), - child: IgnorePointer(child: item), - ), + return Tooltip( + tooltip: TooltipContainer(child: Text(tooltip)).call, + child: IconButton( + variance: variance, + icon: icon ?? const Icon(SpotubeIcons.moreVertical), + onPressed: () => showDropdownMenu(context, Offset.zero), ), ); } diff --git a/lib/components/adaptive/adaptive_popup_menu_button.dart b/lib/components/adaptive/adaptive_popup_menu_button.dart deleted file mode 100644 index 02fced52..00000000 --- a/lib/components/adaptive/adaptive_popup_menu_button.dart +++ /dev/null @@ -1,106 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; - -import 'package:popover/popover.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/extensions/constrains.dart'; - -class Action extends StatelessWidget { - final Widget text; - final Widget icon; - final void Function() onPressed; - final bool isExpanded; - final Color? backgroundColor; - const Action({ - super.key, - required this.icon, - required this.text, - required this.onPressed, - this.isExpanded = true, - this.backgroundColor, - }); - - @override - Widget build(BuildContext context) { - if (isExpanded != true) { - return IconButton( - icon: icon, - onPressed: onPressed, - style: IconButton.styleFrom( - backgroundColor: backgroundColor, - ), - tooltip: text is Text - ? (text as Text).data - : text.toStringShallow().split(",").last.replaceAll( - "\"", - "", - ), - ); - } - - return ListTile( - tileColor: backgroundColor, - onTap: onPressed, - leading: icon, - title: text, - ); - } -} - -class AdaptiveActions extends HookWidget { - final List actions; - final bool? breakOn; - const AdaptiveActions({ - required this.actions, - this.breakOn, - super.key, - }); - - @override - Widget build(BuildContext context) { - final mediaQuery = MediaQuery.of(context); - - if (breakOn ?? mediaQuery.lgAndUp) { - return IconButton( - icon: const Icon(SpotubeIcons.moreHorizontal), - onPressed: () { - showPopover( - context: context, - direction: PopoverDirection.left, - bodyBuilder: (context) { - return Column( - crossAxisAlignment: CrossAxisAlignment.start, - mainAxisSize: MainAxisSize.min, - children: actions - .map( - (action) => SizedBox( - width: 200, - child: Row( - children: [ - Expanded(child: action), - ], - ), - ), - ) - .toList(), - ); - }, - backgroundColor: Theme.of(context).cardColor, - ); - }, - ); - } - - return Row( - children: actions.map((action) { - return Action( - icon: action.icon, - onPressed: action.onPressed, - text: action.text, - backgroundColor: action.backgroundColor, - isExpanded: false, - ); - }).toList(), - ); - } -} diff --git a/lib/components/adaptive/adaptive_select_tile.dart b/lib/components/adaptive/adaptive_select_tile.dart index 3f6d2700..afa982af 100644 --- a/lib/components/adaptive/adaptive_select_tile.dart +++ b/lib/components/adaptive/adaptive_select_tile.dart @@ -1,5 +1,6 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' show ListTile, ListTileControlAffinity; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/extensions/constrains.dart'; @@ -7,11 +8,12 @@ class AdaptiveSelectTile extends HookWidget { final Widget title; final Widget? subtitle; final Widget? secondary; + final List? trailing; final ListTileControlAffinity? controlAffinity; final T value; final ValueChanged? onChanged; - final List> options; + final List> options; /// Show the smaller value when the breakpoint is reached /// @@ -22,6 +24,9 @@ class AdaptiveSelectTile extends HookWidget { final bool? breakLayout; + final BoxConstraints? popupConstraints; + final PopoverConstraint? popupWidthConstraint; + const AdaptiveSelectTile({ required this.title, required this.value, @@ -30,64 +35,50 @@ class AdaptiveSelectTile extends HookWidget { this.controlAffinity = ListTileControlAffinity.trailing, this.subtitle, this.secondary, + this.trailing, this.breakLayout, this.showValueWhenUnfolded = true, super.key, + this.popupConstraints, + this.popupWidthConstraint, }); @override Widget build(BuildContext context) { final theme = Theme.of(context); - final mediaQuery = MediaQuery.of(context); - final rawControl = DecoratedBox( - decoration: BoxDecoration( - color: theme.colorScheme.secondaryContainer, - borderRadius: BorderRadius.circular(10), - ), - child: DropdownButton( - items: options, - value: value, - onChanged: onChanged, - menuMaxHeight: mediaQuery.size.height * 0.6, - underline: const SizedBox.shrink(), - padding: const EdgeInsets.symmetric(horizontal: 10), - borderRadius: BorderRadius.circular(10), - icon: const Icon(SpotubeIcons.angleDown), - dropdownColor: theme.colorScheme.secondaryContainer, - ), - ); - final controlPlaceholder = useMemoized( - () => options - .firstWhere( - (element) => element.value == value, - orElse: () => DropdownMenuItem( - value: null, - child: Container(), - ), - ) - .child, - [value, options]); + final mediaQuery = MediaQuery.sizeOf(context); - final control = breakLayout ?? mediaQuery.mdAndUp - ? rawControl - : showValueWhenUnfolded - ? Container( - padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4), - decoration: BoxDecoration( - border: Border.all( - color: theme.colorScheme.primary, - width: 2, - ), - borderRadius: BorderRadius.circular(10), - ), - child: DefaultTextStyle( - style: TextStyle( - color: theme.colorScheme.primary, - ), - child: controlPlaceholder, - ), - ) - : const SizedBox.shrink(); + Widget? control = Select( + itemBuilder: (context, item) { + return options.firstWhere((element) => element.value == item).child; + }, + value: value, + onChanged: onChanged, + popupConstraints: popupConstraints ?? const BoxConstraints(maxWidth: 200), + popupWidthConstraint: popupWidthConstraint ?? PopoverConstraint.flexible, + autoClosePopover: true, + popup: (context) { + return SelectPopup( + autoClose: true, + items: SelectItemBuilder( + childCount: options.length, + builder: (context, index) { + return options[index]; + }, + ), + ); + }, + ); + + if (mediaQuery.smAndDown) { + if (showValueWhenUnfolded) { + control = OutlineBadge( + child: options.firstWhere((element) => element.value == value).child, + ); + } else { + control = null; + } + } return ListTile( title: title, @@ -95,29 +86,48 @@ class AdaptiveSelectTile extends HookWidget { leading: controlAffinity != ListTileControlAffinity.leading ? secondary : control, - trailing: controlAffinity == ListTileControlAffinity.leading - ? secondary - : control, + trailing: Row( + mainAxisSize: MainAxisSize.min, + mainAxisAlignment: MainAxisAlignment.end, + spacing: 5, + children: [ + ...?trailing, + if (controlAffinity == ListTileControlAffinity.leading && + secondary != null) + secondary! + else if (controlAffinity == ListTileControlAffinity.trailing && + control != null) + control, + ], + ), onTap: breakLayout ?? mediaQuery.mdAndUp ? null : () { showDialog( context: context, builder: (context) { - return SimpleDialog( - title: title, - children: [ - for (final option in options) - RadioListTile( - title: option.child, - value: option.value as T, - groupValue: value, - onChanged: (v) { - Navigator.pop(context); - onChanged?.call(v); - }, - ), - ], + return AlertDialog( + content: Flexible( + child: ListView.builder( + shrinkWrap: true, + itemCount: options.length, + itemBuilder: (context, index) { + final item = options[index]; + + return ListTile( + iconColor: theme.colorScheme.primary, + leading: item.value == value + ? const Icon(SpotubeIcons.radioChecked) + : const Icon(SpotubeIcons.radioUnchecked), + title: item.child, + onTap: () { + onChanged?.call(item.value); + Navigator.of(context).pop(); + }, + ); + }, + ), + ), ); }, ); diff --git a/lib/components/animated_gradient.dart b/lib/components/animated_gradient.dart deleted file mode 100644 index aaba2ff9..00000000 --- a/lib/components/animated_gradient.dart +++ /dev/null @@ -1,126 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; - -class AnimateGradient extends HookWidget { - const AnimateGradient({ - super.key, - required this.primaryColors, - required this.secondaryColors, - this.child, - this.primaryBegin, - this.primaryEnd, - this.secondaryBegin, - this.secondaryEnd, - AnimationController? controller, - this.duration = const Duration(seconds: 4), - this.animateAlignments = true, - this.reverse = true, - }) : assert(primaryColors.length >= 2), - assert(primaryColors.length == secondaryColors.length), - _controller = controller; - - /// [controller]: pass this to have a fine control over the [Animation] - final AnimationController? _controller; - - /// [duration]: Time to switch between [Gradient]. - /// By default its value is [Duration(seconds:4)] - final Duration duration; - - /// [primaryColors]: These will be the starting colors of the [Animation]. - final List primaryColors; - - /// [secondaryColors]: These Colors are those in which the [primaryColors] will transition into. - final List secondaryColors; - - /// [primaryBegin]: This is begin [Alignment] for [primaryColors]. - /// By default its value is [Alignment.topLeft] - final Alignment? primaryBegin; - - /// [primaryBegin]: This is end [Alignment] for [primaryColors]. - /// By default its value is [Alignment.topRight] - final Alignment? primaryEnd; - - /// [secondaryBegin]: This is begin [Alignment] for [secondaryColors]. - /// By default its value is [Alignment.bottomLeft] - final Alignment? secondaryBegin; - - /// [secondaryEnd]: This is end [Alignment] for [secondaryColors]. - /// By default its value is [Alignment.bottomRight] - final Alignment? secondaryEnd; - - /// [animateAlignments]: set to false if you don't want to animate the alignments. - /// This can provide you way cooler animations - final bool animateAlignments; - - /// [reverse]: set it to false if you don't want to reverse the animation. - /// using that it will go into one direction only - final bool reverse; - - final Widget? child; - - @override - Widget build(BuildContext context) { - // ignore: no_leading_underscores_for_local_identifiers - final __controller = useAnimationController( - duration: duration, - )..repeat(reverse: reverse); - - final controller = _controller ?? __controller; - - final animation = useMemoized( - () => CurvedAnimation( - parent: controller, - curve: Curves.easeInOut, - ), - [controller]); - - final colorTween = useMemoized( - () => primaryColors.map((color) { - return ColorTween( - begin: color, - end: color, - ); - }).toList(), - [primaryColors]); - final colors = useMemoized( - () => colorTween.map((color) { - return color.evaluate(animation)!; - }).toList(), - [colorTween, animation]); - - final begin = useMemoized( - () => AlignmentTween( - begin: primaryBegin ?? Alignment.topLeft, - end: primaryEnd ?? Alignment.topRight, - ), - [primaryBegin, primaryEnd]); - - final end = useMemoized( - () => AlignmentTween( - begin: secondaryBegin ?? Alignment.bottomLeft, - end: secondaryEnd ?? Alignment.bottomRight, - ), - [secondaryBegin, secondaryEnd]); - - return AnimatedBuilder( - animation: animation, - child: useMemoized(() => child, [child]), - builder: (BuildContext context, Widget? child) { - return Container( - decoration: BoxDecoration( - gradient: LinearGradient( - begin: animateAlignments - ? begin.evaluate(animation) - : (primaryBegin as Alignment), - end: animateAlignments - ? end.evaluate(animation) - : primaryEnd as Alignment, - colors: colors, - ), - ), - child: child, - ); - }, - ); - } -} diff --git a/lib/components/bordered_text.dart b/lib/components/bordered_text.dart deleted file mode 100644 index f25f2208..00000000 --- a/lib/components/bordered_text.dart +++ /dev/null @@ -1,88 +0,0 @@ -library bordered_text; - -import 'package:flutter/widgets.dart'; - -/// Adds stroke to text widget -/// We can apply a very thin and subtle stroke to a [Text] -/// ```dart -/// BorderedText( -/// strokeWidth: 1.0, -/// text: Text( -/// 'Bordered Text', -/// style: TextStyle( -/// decoration: TextDecoration.none, -/// decorationStyle: TextDecorationStyle.wavy, -/// decorationColor: Colors.red, -/// ), -/// ), -/// ) -/// ``` -class BorderedText extends StatelessWidget { - const BorderedText({ - super.key, - required this.child, - this.strokeCap = StrokeCap.round, - this.strokeJoin = StrokeJoin.round, - this.strokeWidth = 6.0, - this.strokeColor = const Color.fromRGBO(53, 0, 71, 1), - }); - - /// the stroke cap style - final StrokeCap strokeCap; - - /// the stroke joint style - final StrokeJoin strokeJoin; - - /// the stroke width - final double strokeWidth; - - /// the stroke color - final Color strokeColor; - - /// the [Text] widget to apply stroke on - final Text child; - - @override - Widget build(BuildContext context) { - TextStyle style; - if (child.style != null) { - style = child.style!.copyWith( - foreground: Paint() - ..style = PaintingStyle.stroke - ..strokeCap = strokeCap - ..strokeJoin = strokeJoin - ..strokeWidth = strokeWidth - ..color = strokeColor, - color: null, - ); - } else { - style = TextStyle( - foreground: Paint() - ..style = PaintingStyle.stroke - ..strokeCap = strokeCap - ..strokeJoin = strokeJoin - ..strokeWidth = strokeWidth - ..color = strokeColor, - ); - } - return Stack( - alignment: Alignment.center, - textDirection: child.textDirection, - children: [ - Text( - child.data!, - style: style, - maxLines: child.maxLines, - overflow: child.overflow, - semanticsLabel: child.semanticsLabel, - softWrap: child.softWrap, - strutStyle: child.strutStyle, - textAlign: child.textAlign, - textDirection: child.textDirection, - textScaler: child.textScaler, - ), - child, - ], - ); - } -} diff --git a/lib/components/button/back_button.dart b/lib/components/button/back_button.dart new file mode 100644 index 00000000..dc899616 --- /dev/null +++ b/lib/components/button/back_button.dart @@ -0,0 +1,21 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/spotube_icons.dart'; + +class BackButton extends StatelessWidget { + final Color? color; + final IconData icon; + const BackButton({ + super.key, + this.color, + this.icon = SpotubeIcons.angleLeft, + }); + + @override + Widget build(BuildContext context) { + return IconButton.ghost( + size: const ButtonSize(1.2), + icon: Icon(icon, color: color), + onPressed: () => Navigator.of(context).pop(), + ); + } +} diff --git a/lib/components/compact_search.dart b/lib/components/compact_search.dart deleted file mode 100644 index d37cb673..00000000 --- a/lib/components/compact_search.dart +++ /dev/null @@ -1,52 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; - -import 'package:popover/popover.dart'; -import 'package:spotube/collections/spotube_icons.dart'; - -class CompactSearch extends HookWidget { - final ValueChanged? onChanged; - final String placeholder; - final IconData icon; - final Color? iconColor; - - const CompactSearch({ - super.key, - this.onChanged, - this.placeholder = "Search...", - this.icon = SpotubeIcons.search, - this.iconColor, - }); - - @override - Widget build(BuildContext context) { - return IconButton( - onPressed: () { - showPopover( - context: context, - backgroundColor: Theme.of(context).cardColor, - transitionDuration: const Duration(milliseconds: 100), - barrierColor: Colors.transparent, - arrowDxOffset: -6, - bodyBuilder: (context) { - return Container( - padding: const EdgeInsets.all(8.0), - width: 300, - child: TextField( - autofocus: true, - onChanged: onChanged, - decoration: InputDecoration( - hintText: placeholder, - prefixIcon: Icon(icon), - ), - ), - ); - }, - height: 60, - ); - }, - tooltip: placeholder, - icon: Icon(icon, color: iconColor), - ); - } -} diff --git a/lib/components/dialogs/confirm_download_dialog.dart b/lib/components/dialogs/confirm_download_dialog.dart index 897c64cb..a2df0e9c 100644 --- a/lib/components/dialogs/confirm_download_dialog.dart +++ b/lib/components/dialogs/confirm_download_dialog.dart @@ -1,5 +1,4 @@ -import 'package:flutter/material.dart'; - +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; @@ -9,13 +8,15 @@ class ConfirmDownloadDialog extends StatelessWidget { @override Widget build(BuildContext context) { - return AlertDialog( - title: Padding( - padding: const EdgeInsets.all(15), - child: Row( + final screenSize = MediaQuery.sizeOf(context); + + return ConstrainedBox( + constraints: BoxConstraints(maxWidth: Breakpoints.sm), + child: AlertDialog( + title: Row( + spacing: 10, children: [ Text(context.l10n.are_you_sure), - const SizedBox(width: 10), const UniversalImage( path: "https://c.tenor.com/kHcmsxlKHEAAAAAM/rock-one-eyebrow-raised-rock-staring.gif", @@ -24,58 +25,53 @@ class ConfirmDownloadDialog extends StatelessWidget { ) ], ), - ), - content: Container( - padding: const EdgeInsets.all(15), - constraints: BoxConstraints(maxWidth: Breakpoints.sm), - child: SingleChildScrollView( - child: Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text( - context.l10n.download_warning, - textAlign: TextAlign.justify, - ), - const SizedBox(height: 10), - Text( - context.l10n.download_ip_ban_warning, - style: const TextStyle( - color: Colors.red, - fontWeight: FontWeight.bold, + content: Expanded( + flex: screenSize.smAndUp ? 0 : 1, + child: SingleChildScrollView( + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + context.l10n.download_warning, + textAlign: TextAlign.justify, ), - textAlign: TextAlign.justify, - ), - const SizedBox(height: 10), - Text( - context.l10n.by_clicking_accept_terms, - ), - const SizedBox(height: 10), - BulletPoint(context.l10n.download_agreement_1), - const SizedBox(height: 10), - BulletPoint(context.l10n.download_agreement_2), - const SizedBox(height: 10), - BulletPoint(context.l10n.download_agreement_3), - ], + const SizedBox(height: 10), + Text( + context.l10n.download_ip_ban_warning, + style: const TextStyle( + color: Colors.red, + fontWeight: FontWeight.bold, + ), + textAlign: TextAlign.justify, + ), + const SizedBox(height: 10), + Text( + context.l10n.by_clicking_accept_terms, + ), + const SizedBox(height: 10), + BulletPoint(context.l10n.download_agreement_1), + const SizedBox(height: 10), + BulletPoint(context.l10n.download_agreement_2), + const SizedBox(height: 10), + BulletPoint(context.l10n.download_agreement_3), + ], + ), ), ), + actions: [ + Button.outline( + child: Text(context.l10n.decline), + onPressed: () { + Navigator.pop(context, false); + }, + ), + Button.destructive( + onPressed: () => Navigator.of(context).pop(true), + child: Text(context.l10n.accept), + ), + ], ), - actions: [ - OutlinedButton( - child: Text(context.l10n.decline), - onPressed: () { - Navigator.pop(context, false); - }, - ), - FilledButton( - style: FilledButton.styleFrom( - foregroundColor: Colors.white, - backgroundColor: Colors.red, - ), - onPressed: () => Navigator.of(context).pop(true), - child: Text(context.l10n.accept), - ), - ], ); } } diff --git a/lib/components/dialogs/link_open_permission_dialog.dart b/lib/components/dialogs/link_open_permission_dialog.dart new file mode 100644 index 00000000..a7212d0a --- /dev/null +++ b/lib/components/dialogs/link_open_permission_dialog.dart @@ -0,0 +1,69 @@ +import 'package:flutter/services.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:url_launcher/url_launcher_string.dart'; + +class LinkOpenPermissionDialog extends StatelessWidget { + final String? href; + const LinkOpenPermissionDialog({super.key, this.href}); + + @override + Widget build(BuildContext context) { + return ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 450), + child: AlertDialog( + title: Row( + spacing: 8, + children: [ + const Icon(SpotubeIcons.warning), + Text(context.l10n.open_link_in_browser), + ], + ), + content: Text.rich( + TextSpan( + children: [ + TextSpan( + text: + "${context.l10n.do_you_want_to_open_the_following_link}:\n", + ), + if (href != null) + TextSpan( + text: "$href\n\n", + style: const TextStyle(color: Colors.blue), + ), + TextSpan(text: context.l10n.unsafe_url_warning), + ], + ), + ), + actions: [ + Button.ghost( + onPressed: () => Navigator.of(context).pop(false), + child: Text(context.l10n.cancel), + ), + Button.ghost( + onPressed: () { + if (href != null) { + Clipboard.setData(ClipboardData(text: href!)); + } + Navigator.of(context).pop(false); + }, + child: Text(context.l10n.copy_link), + ), + Button.destructive( + onPressed: () { + if (href != null) { + launchUrlString( + href!, + mode: LaunchMode.externalApplication, + ); + } + Navigator.of(context).pop(true); + }, + child: Text(context.l10n.open), + ), + ], + ), + ); + } +} diff --git a/lib/components/dialogs/piped_down_dialog.dart b/lib/components/dialogs/piped_down_dialog.dart deleted file mode 100644 index b1717a2a..00000000 --- a/lib/components/dialogs/piped_down_dialog.dart +++ /dev/null @@ -1,46 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -class PipedDownDialog extends HookConsumerWidget { - const PipedDownDialog({super.key}); - - @override - Widget build(BuildContext context, ref) { - final pipedInstance = - ref.watch(userPreferencesProvider.select((s) => s.pipedInstance)); - final ThemeData(:colorScheme) = Theme.of(context); - - return AlertDialog( - insetPadding: const EdgeInsets.all(6), - contentPadding: const EdgeInsets.all(6), - icon: Icon( - SpotubeIcons.error, - color: colorScheme.error, - ), - title: Text( - context.l10n.piped_api_down, - style: TextStyle(color: colorScheme.error), - ), - content: Card( - child: Padding( - padding: const EdgeInsets.all(8.0), - child: - Text(context.l10n.piped_down_error_instructions(pipedInstance)), - ), - ), - actions: [ - TextButton( - onPressed: () => Navigator.pop(context), - child: Text(context.l10n.ok), - ), - FilledButton( - onPressed: () => Navigator.pop(context), - child: Text(context.l10n.settings), - ), - ], - ); - } -} diff --git a/lib/components/dialogs/playlist_add_track_dialog.dart b/lib/components/dialogs/playlist_add_track_dialog.dart index 5af9c9e4..09d831ea 100644 --- a/lib/components/dialogs/playlist_add_track_dialog.dart +++ b/lib/components/dialogs/playlist_add_track_dialog.dart @@ -1,19 +1,18 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/models/metadata/metadata.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/modules/playlist/playlist_create_dialog.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; class PlaylistAddTrackDialog extends HookConsumerWidget { /// The id of the playlist this dialog was opened from final String? openFromPlaylist; - final List tracks; + final List tracks; const PlaylistAddTrackDialog({ required this.tracks, required this.openFromPlaylist, @@ -22,25 +21,24 @@ class PlaylistAddTrackDialog extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); - final userPlaylists = ref.watch(favoritePlaylistsProvider); + final typography = Theme.of(context).typography; + final userPlaylists = ref.watch(metadataPluginSavedPlaylistsProvider); final favoritePlaylistsNotifier = - ref.watch(favoritePlaylistsProvider.notifier); + ref.watch(metadataPluginSavedPlaylistsProvider.notifier); - final me = ref.watch(meProvider); + final me = ref.watch(metadataPluginUserProvider); final filteredPlaylists = useMemoized( () => userPlaylists.asData?.value.items .where( (playlist) => - playlist.owner?.id != null && - playlist.owner!.id == me.asData?.value.id && + playlist.owner.id == me.asData?.value?.id && playlist.id != openFromPlaylist, ) .toList() ?? [], - [userPlaylists.asData?.value, me.asData?.value.id, openFromPlaylist], + [userPlaylists.asData?.value, me.asData?.value?.id, openFromPlaylist], ); final playlistsCheck = useState({}); @@ -61,70 +59,88 @@ class PlaylistAddTrackDialog extends HookConsumerWidget { selectedPlaylists.map( (playlistId) => favoritePlaylistsNotifier.addTracks( playlistId, - tracks.map((e) => e.id!).toList(), + tracks.map((e) => e.id).toList(), ), ), - ).then((_) => Navigator.pop(context, true)); + ).then((_) => context.mounted ? Navigator.pop(context, true) : null); } - return AlertDialog( - insetPadding: EdgeInsets.zero, - title: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text( - context.l10n.add_to_playlist, - style: textTheme.titleMedium, + return ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 400), + child: AlertDialog( + title: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + mainAxisSize: MainAxisSize.min, + children: [ + Text( + context.l10n.add_to_playlist, + style: typography.large, + ), + const Spacer(), + const PlaylistCreateDialogButton(), + ], + ), + actions: [ + OutlineButton( + child: Text(context.l10n.cancel), + onPressed: () { + Navigator.pop(context, false); + }, + ), + PrimaryButton( + onPressed: onAdd, + child: Text(context.l10n.add), ), - const Gap(20), - const PlaylistCreateDialogButton(), ], - ), - actions: [ - OutlinedButton( - child: Text(context.l10n.cancel), - onPressed: () { - Navigator.pop(context, false); - }, - ), - FilledButton( - onPressed: onAdd, - child: Text(context.l10n.add), - ), - ], - content: SizedBox( - height: 300, - width: 300, - child: userPlaylists.isLoading - ? const Center(child: CircularProgressIndicator()) - : ListView.builder( - shrinkWrap: true, - itemCount: filteredPlaylists.length, - itemBuilder: (context, index) { - final playlist = filteredPlaylists.elementAt(index); - return CheckboxListTile( - secondary: CircleAvatar( - backgroundImage: UniversalImage.imageProvider( - playlist.images.asUrlString( - placeholder: ImagePlaceholder.collection, + content: SizedBox( + height: 300, + child: userPlaylists.isLoading + ? const Center(child: CircularProgressIndicator()) + : ListView.builder( + shrinkWrap: true, + itemCount: filteredPlaylists.length, + itemBuilder: (context, index) { + final playlist = filteredPlaylists.elementAt(index); + return Button.ghost( + style: ButtonVariance.ghost.copyWith( + padding: (context, _, __) { + return const EdgeInsets.symmetric(vertical: 8); + }, + ), + leading: Avatar( + initials: Avatar.getInitials(playlist.name), + provider: UniversalImage.imageProvider( + playlist.images.asUrlString( + placeholder: ImagePlaceholder.collection, + ), ), ), - ), - contentPadding: EdgeInsets.zero, - title: Padding( - padding: const EdgeInsets.only(left: 8.0), - child: Text(playlist.name!), - ), - value: playlistsCheck.value[playlist.id] ?? false, - onChanged: (val) { - playlistsCheck.value = { - ...playlistsCheck.value, - playlist.id!: val == true - }; - }, - ); - }, - ), + trailing: Checkbox( + state: (playlistsCheck.value[playlist.id] ?? false) + ? CheckboxState.checked + : CheckboxState.unchecked, + onChanged: (val) { + playlistsCheck.value = { + ...playlistsCheck.value, + playlist.id: val == CheckboxState.checked, + }; + }, + ), + onPressed: () { + playlistsCheck.value = { + ...playlistsCheck.value, + playlist.id: + !(playlistsCheck.value[playlist.id] ?? false), + }; + }, + child: Padding( + padding: const EdgeInsets.only(left: 8.0), + child: Text(playlist.name), + ), + ); + }, + ), + ), ), ); } diff --git a/lib/components/dialogs/prompt_dialog.dart b/lib/components/dialogs/prompt_dialog.dart index 30a63bcf..3498bf02 100644 --- a/lib/components/dialogs/prompt_dialog.dart +++ b/lib/components/dialogs/prompt_dialog.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/extensions/context.dart'; Future showPromptDialog({ @@ -16,13 +16,13 @@ Future showPromptDialog({ content: Text(message), actions: [ if (cancelText != null) - OutlinedButton( + Button.outline( onPressed: () => Navigator.of(context).pop(false), child: Text( cancelText == "Cancel" ? context.l10n.cancel : cancelText, ), ), - FilledButton( + Button.primary( child: Text(okText == "Ok" ? context.l10n.ok : okText), onPressed: () => Navigator.of(context).pop(true), ), diff --git a/lib/components/dialogs/replace_downloaded_dialog.dart b/lib/components/dialogs/replace_downloaded_dialog.dart index 00461d34..5b5b194e 100644 --- a/lib/components/dialogs/replace_downloaded_dialog.dart +++ b/lib/components/dialogs/replace_downloaded_dialog.dart @@ -1,57 +1,46 @@ -import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; final replaceDownloadedFileState = StateProvider((ref) => null); class ReplaceDownloadedDialog extends ConsumerWidget { - final Track track; + final SpotubeTrackObject track; const ReplaceDownloadedDialog({required this.track, super.key}); @override Widget build(BuildContext context, ref) { - final groupValue = ref.watch(replaceDownloadedFileState); - final theme = Theme.of(context); final replaceAll = ref.watch(replaceDownloadedFileState); return AlertDialog( - title: Text(context.l10n.track_exists(track.name ?? "")), - content: Column( - mainAxisSize: MainAxisSize.min, - children: [ - Text(context.l10n.do_you_want_to_replace), - RadioListTile( - dense: true, - contentPadding: EdgeInsets.zero, - activeColor: theme.colorScheme.primary, - value: true, - groupValue: groupValue, - onChanged: (value) { - if (value != null) { - ref.read(replaceDownloadedFileState.notifier).state = true; - } - }, - title: Text(context.l10n.replace_downloaded_tracks), - ), - RadioListTile( - dense: true, - contentPadding: EdgeInsets.zero, - activeColor: theme.colorScheme.primary, - value: false, - groupValue: groupValue, - onChanged: (value) { - if (value != null) { - ref.read(replaceDownloadedFileState.notifier).state = false; - } - }, - title: Text(context.l10n.skip_download_tracks), - ), - ], + title: Text(context.l10n.track_exists(track.name)), + content: RadioGroup( + value: replaceAll, + onChanged: (value) { + ref.read(replaceDownloadedFileState.notifier).state = value; + }, + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(context.l10n.do_you_want_to_replace), + const Gap(16), + RadioItem( + value: true, + trailing: Text(context.l10n.replace_downloaded_tracks), + ), + const Gap(8), + RadioItem( + value: false, + trailing: Text(context.l10n.skip_download_tracks), + ), + ], + ), ), actions: [ - OutlinedButton( + Button.outline( onPressed: replaceAll == true ? null : () { @@ -59,7 +48,7 @@ class ReplaceDownloadedDialog extends ConsumerWidget { }, child: Text(context.l10n.skip), ), - FilledButton( + Button.primary( onPressed: replaceAll == false ? null : () { diff --git a/lib/components/dialogs/select_device_dialog.dart b/lib/components/dialogs/select_device_dialog.dart index 3a3bde60..5392a403 100644 --- a/lib/components/dialogs/select_device_dialog.dart +++ b/lib/components/dialogs/select_device_dialog.dart @@ -1,6 +1,6 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/connect/clients.dart'; @@ -16,31 +16,31 @@ class SelectDeviceDialog extends HookConsumerWidget { return AlertDialog( title: Text(context.l10n.choose_the_device), - insetPadding: const EdgeInsets.all(16), - content: Column( - mainAxisSize: MainAxisSize.min, - children: [ - Text(context.l10n.multiple_device_connected), - RadioListTile.adaptive( - title: Text(remoteService.name), - value: true, - groupValue: isRemoteService.value, - onChanged: (value) { - isRemoteService.value = value!; - }, - ), - RadioListTile.adaptive( - title: Text(context.l10n.this_device), - value: false, - groupValue: isRemoteService.value, - onChanged: (value) { - isRemoteService.value = !value!; - }, - ), - ], + content: RadioGroup( + value: isRemoteService.value, + onChanged: (value) { + isRemoteService.value = value; + }, + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(context.l10n.multiple_device_connected), + const Gap(16), + RadioItem( + trailing: Text(remoteService.name), + value: true, + ), + const Gap(8), + RadioItem( + trailing: Text(context.l10n.this_device), + value: false, + ), + ], + ), ), actions: [ - TextButton( + Button.primary( onPressed: () { Navigator.of(context).pop(isRemoteService.value); }, @@ -51,7 +51,8 @@ class SelectDeviceDialog extends HookConsumerWidget { } } -Future showSelectDeviceDialog(BuildContext context, WidgetRef ref) async { +Future showSelectDeviceDialog( + BuildContext context, WidgetRef ref) async { final connectClients = ref.read(connectClientsProvider); if (connectClients.asData?.value.resolvedService == null) { @@ -63,5 +64,5 @@ Future showSelectDeviceDialog(BuildContext context, WidgetRef ref) async { builder: (context) => const SelectDeviceDialog(), ); - return isRemote ?? false; + return isRemote; } diff --git a/lib/components/dialogs/track_details_dialog.dart b/lib/components/dialogs/track_details_dialog.dart index 61bca7b1..9d35a6fb 100644 --- a/lib/components/dialogs/track_details_dialog.dart +++ b/lib/components/dialogs/track_details_dialog.dart @@ -1,53 +1,49 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:spotify/spotify.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/links/artist_link.dart'; import 'package:spotube/components/links/hyper_link.dart'; -import 'package:spotube/components/links/link_text.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; import 'package:spotube/extensions/duration.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; -class TrackDetailsDialog extends HookWidget { - final Track track; +class TrackDetailsDialog extends HookConsumerWidget { + final SpotubeFullTrackObject track; const TrackDetailsDialog({ super.key, required this.track, }); @override - Widget build(BuildContext context) { + Widget build(BuildContext context, ref) { final theme = Theme.of(context); final mediaQuery = MediaQuery.of(context); + final sourcedTrack = ref.read(sourcedTrackProvider(track)); final detailsMap = { - context.l10n.title: track.name!, + context.l10n.title: track.name, context.l10n.artist: ArtistLink( - artists: track.artists ?? [], + artists: track.artists, mainAxisAlignment: WrapAlignment.start, textStyle: const TextStyle(color: Colors.blue), hideOverflowArtist: false, ), - context.l10n.album: LinkText( - track.album!.name!, - "/album/${track.album?.id}", - extra: track.album, - overflow: TextOverflow.ellipsis, - style: const TextStyle(color: Colors.blue), - ), - context.l10n.duration: (track is SourcedTrack - ? (track as SourcedTrack).sourceInfo.duration - : track.duration!) - .toHumanReadableString(), - if (track.album!.releaseDate != null) - context.l10n.released: track.album!.releaseDate, - context.l10n.popularity: track.popularity?.toString() ?? "0", + // context.l10n.album: LinkText( + // track.album!.name!, + // AlbumRoute(album: track.album!, id: track.album!.id!), + // overflow: TextOverflow.ellipsis, + // style: const TextStyle(color: Colors.blue), + // ), + context.l10n.duration: sourcedTrack.asData != null + ? sourcedTrack.asData!.value.info.duration.toHumanReadableString() + : Duration(milliseconds: track.durationMs).toHumanReadableString(), + if (track.album.releaseDate != null) + context.l10n.released: track.album.releaseDate, }; - final sourceInfo = - track is SourcedTrack ? (track as SourcedTrack).sourceInfo : null; + final sourceInfo = sourcedTrack.asData?.value.info; final ytTracksDetailsMap = sourceInfo == null ? {} @@ -58,32 +54,26 @@ class TrackDetailsDialog extends HookWidget { maxLines: 2, overflow: TextOverflow.ellipsis, ), - context.l10n.channel: Hyperlink( - sourceInfo.artist, - sourceInfo.artistUrl, - maxLines: 2, - overflow: TextOverflow.ellipsis, - ), - context.l10n.streamUrl: Hyperlink( - (track as SourcedTrack).url, - (track as SourcedTrack).url, - maxLines: 2, - overflow: TextOverflow.ellipsis, - ), + context.l10n.channel: Text(sourceInfo.artists.join(", ")), + if (sourcedTrack.asData?.value.url != null) + context.l10n.streamUrl: Hyperlink( + sourcedTrack.asData!.value.url ?? "", + sourcedTrack.asData!.value.url ?? "", + maxLines: 2, + overflow: TextOverflow.ellipsis, + ), }; return AlertDialog( - contentPadding: const EdgeInsets.all(16), - insetPadding: const EdgeInsets.symmetric(horizontal: 16, vertical: 100), - scrollable: true, + surfaceBlur: 0, + surfaceOpacity: 1, title: Row( - mainAxisAlignment: MainAxisAlignment.center, + spacing: 8, children: [ const Icon(SpotubeIcons.info), - const SizedBox(width: 8), Text( context.l10n.details, - style: theme.textTheme.titleMedium, + style: theme.typography.h4, ), ], ), @@ -91,65 +81,64 @@ class TrackDetailsDialog extends HookWidget { width: mediaQuery.mdAndUp ? double.infinity : 700, child: Table( columnWidths: const { - 0: FixedColumnWidth(95), - 1: FixedColumnWidth(10), - 2: FlexColumnWidth(1), + 0: FixedTableSize(95), + 1: FixedTableSize(10), + 2: FlexTableSize(), }, - defaultVerticalAlignment: TableCellVerticalAlignment.middle, - children: [ + theme: const TableTheme( + backgroundColor: Colors.transparent, + cellTheme: TableCellTheme( + backgroundColor: WidgetStatePropertyAll(Colors.transparent), + ), + ), + rowHeights: const {0: FixedTableSize(40)}, + rows: [ for (final entry in detailsMap.entries) TableRow( - children: [ + cells: [ TableCell( - verticalAlignment: TableCellVerticalAlignment.top, child: Text( entry.key, - style: theme.textTheme.titleMedium, + style: theme.typography.bold, ), ), const TableCell( - verticalAlignment: TableCellVerticalAlignment.top, child: Text(":"), ), - if (entry.value is Widget) - entry.value as Widget - else if (entry.value is String) - Text( - entry.value as String, - style: theme.textTheme.bodyMedium, - ), + TableCell( + child: entry.value is Widget + ? entry.value as Widget + : (entry.value is String) + ? Text( + entry.value as String, + style: theme.typography.normal, + ) + : const Text(""), + ), ], ), - const TableRow( - children: [ - SizedBox(height: 16), - SizedBox(height: 16), - SizedBox(height: 16), - ], - ), for (final entry in ytTracksDetailsMap.entries) TableRow( - children: [ + cells: [ TableCell( - verticalAlignment: TableCellVerticalAlignment.top, child: Text( entry.key, - style: theme.textTheme.titleMedium, + style: theme.typography.bold, ), ), const TableCell( - verticalAlignment: TableCellVerticalAlignment.top, child: Text(":"), ), - if (entry.value is Widget) - entry.value as Widget - else - Text( - entry.value, - maxLines: 2, - overflow: TextOverflow.ellipsis, - style: theme.textTheme.bodyMedium, - ), + TableCell( + child: entry.value is Widget + ? entry.value as Widget + : Text( + entry.value, + maxLines: 2, + overflow: TextOverflow.ellipsis, + style: theme.typography.normal, + ), + ), ], ), ], diff --git a/lib/components/expandable_search/expandable_search.dart b/lib/components/expandable_search/expandable_search.dart index 157e180f..279a3e5f 100644 --- a/lib/components/expandable_search/expandable_search.dart +++ b/lib/components/expandable_search/expandable_search.dart @@ -1,5 +1,5 @@ -import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/extensions/context.dart'; @@ -39,11 +39,10 @@ class ExpandableSearchField extends StatelessWidget { child: TextField( focusNode: searchFocus, controller: searchController, - decoration: InputDecoration( - hintText: context.l10n.search_tracks, - isDense: true, - prefixIcon: const Icon(SpotubeIcons.search), - ), + placeholder: Text(context.l10n.search_tracks), + features: const [ + InputFeature.leading(Icon(SpotubeIcons.search)) + ], ), ), ), @@ -69,16 +68,9 @@ class ExpandableSearchButton extends StatelessWidget { @override Widget build(BuildContext context) { - final theme = Theme.of(context); - return IconButton( icon: icon, - style: IconButton.styleFrom( - backgroundColor: - isFiltering ? theme.colorScheme.secondaryContainer : null, - foregroundColor: isFiltering ? theme.colorScheme.secondary : null, - minimumSize: const Size(25, 25), - ), + variance: isFiltering ? ButtonVariance.secondary : ButtonVariance.outline, onPressed: () { if (isFiltering) { searchFocus.requestFocus(); diff --git a/lib/components/fallbacks/anonymous_fallback.dart b/lib/components/fallbacks/anonymous_fallback.dart index 62ed8ddd..cb6028a7 100644 --- a/lib/components/fallbacks/anonymous_fallback.dart +++ b/lib/components/fallbacks/anonymous_fallback.dart @@ -1,10 +1,13 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/settings/settings.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:spotube/utils/platform.dart'; class AnonymousFallback extends ConsumerWidget { final Widget? child; @@ -15,22 +18,29 @@ class AnonymousFallback extends ConsumerWidget { @override Widget build(BuildContext context, ref) { - final isLoggedIn = ref.watch(authenticationProvider); + final isLoggedIn = ref.watch(metadataPluginAuthenticatedProvider); if (isLoggedIn.isLoading) { return const Center(child: CircularProgressIndicator()); } - if (isLoggedIn.asData?.value != null && child != null) return child!; + if (isLoggedIn.asData?.value == true && child != null) return child!; return Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, + spacing: 10, children: [ + Undraw( + illustration: kIsMobile + ? UndrawIllustration.accessDenied + : UndrawIllustration.secureLogin, + height: 200 * context.theme.scaling, + color: context.theme.colorScheme.primary, + ), Text(context.l10n.not_logged_in), - const SizedBox(height: 10), - FilledButton( - child: Text(context.l10n.login_with_spotify), - onPressed: () => ServiceUtils.pushNamed(context, SettingsPage.name), + Button.primary( + child: Text(context.l10n.login), + onPressed: () => context.navigateTo(const SettingsRoute()), ) ], ), diff --git a/lib/components/fallbacks/error_box.dart b/lib/components/fallbacks/error_box.dart new file mode 100644 index 00000000..fd56cb58 --- /dev/null +++ b/lib/components/fallbacks/error_box.dart @@ -0,0 +1,138 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/context.dart'; + +class ErrorBox extends StatelessWidget { + final Object error; + final VoidCallback? onRetry; + const ErrorBox({ + super.key, + required this.error, + this.onRetry, + }); + + @override + Widget build(BuildContext context) { + // Make a monospace error log view. Make sure it's only 4 lines + return ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 400), + child: Padding( + padding: const EdgeInsets.all(8.0), + child: Card( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + spacing: 12, + children: [ + Basic( + leading: const Icon(SpotubeIcons.error), + contentSpacing: 8, + title: Text(context.l10n.an_error_occurred), + ), + Card( + padding: const EdgeInsets.all(8.0), + filled: true, + fillColor: context.theme.colorScheme.muted, + child: Text( + error.toString(), + style: TextStyle( + // Use monospace + fontFamily: 'Ubuntu Mono', + color: context.theme.colorScheme.mutedForeground, + fontSize: 14, + ), + maxLines: 6, + overflow: TextOverflow.ellipsis, + ), + ), + // Show a dialog with full log and a retry button as well + Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Button.text( + leading: const Icon(SpotubeIcons.logs), + onPressed: () { + showDialog( + context: context, + builder: (context) { + return ConstrainedBox( + constraints: BoxConstraints( + maxWidth: 480, + maxHeight: + MediaQuery.of(context).size.height * 0.8, + ), + child: AlertDialog( + padding: const EdgeInsets.all(12), + title: Row( + spacing: 8, + children: [ + const Icon(SpotubeIcons.logs), + Text(context.l10n.logs), + const Spacer(), + IconButton.ghost( + icon: const Icon(SpotubeIcons.close), + onPressed: () => context.maybePop(), + ) + ], + ), + actions: [ + HookBuilder(builder: (context) { + final copied = useState(false); + + return Button.ghost( + leading: copied.value + ? const Icon(SpotubeIcons.done) + : const Icon(SpotubeIcons.clipboard), + child: Text(context.l10n.copy_to_clipboard), + onPressed: () { + Clipboard.setData( + ClipboardData(text: error.toString()), + ); + copied.value = true; + }, + ); + }) + ], + content: SingleChildScrollView( + child: Card( + padding: const EdgeInsets.all(8.0), + filled: true, + fillColor: context.theme.colorScheme.muted, + child: SelectableText( + error.toString(), + style: TextStyle( + // Use monospace + fontFamily: 'Ubuntu Mono', + color: context + .theme.colorScheme.mutedForeground, + fontSize: 16, + ), + ), + ), + ), + ), + ); + }, + ); + }, + child: Text(context.l10n.view_logs), + ), + if (onRetry != null) + Button.text( + leading: const Icon(SpotubeIcons.refresh), + onPressed: onRetry, + child: Text(context.l10n.retry), + ), + ], + ), + ], + ), + ), + ), + ); + } +} diff --git a/lib/components/fallbacks/no_default_metadata_plugin.dart b/lib/components/fallbacks/no_default_metadata_plugin.dart new file mode 100644 index 00000000..1cabcdb1 --- /dev/null +++ b/lib/components/fallbacks/no_default_metadata_plugin.dart @@ -0,0 +1,42 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:auto_size_text/auto_size_text.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/context.dart'; + +class NoDefaultMetadataPlugin extends StatelessWidget { + const NoDefaultMetadataPlugin({super.key}); + + @override + Widget build(BuildContext context) { + return Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.stars, + color: context.theme.colorScheme.primary, + ), + AutoSizeText( + context.l10n.no_default_metadata_provider_selected, + style: context.theme.typography.h4, + maxLines: 1, + ), + Button.primary( + leading: const Icon(SpotubeIcons.extensions), + child: Text(context.l10n.manage_metadata_providers), + onPressed: () { + context.pushRoute(const SettingsMetadataProviderRoute()); + }, + ), + ], + ), + ); + } +} diff --git a/lib/components/fallbacks/not_found.dart b/lib/components/fallbacks/not_found.dart index ce168f17..9a994446 100644 --- a/lib/components/fallbacks/not_found.dart +++ b/lib/components/fallbacks/not_found.dart @@ -1,32 +1,27 @@ -import 'package:flutter/material.dart'; -import 'package:spotube/collections/assets.gen.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/extensions/context.dart'; class NotFound extends StatelessWidget { - final bool vertical; - const NotFound({super.key, this.vertical = false}); + const NotFound({super.key}); @override Widget build(BuildContext context) { - final theme = Theme.of(context); - final widgets = [ - SizedBox( - height: 150, - width: 150, - child: Assets.emptyBox.image(), - ), - Column( - crossAxisAlignment: CrossAxisAlignment.start, - mainAxisAlignment: MainAxisAlignment.center, - children: [ - Text(context.l10n.nothing_found, style: theme.textTheme.titleLarge), - Text( - context.l10n.the_box_is_empty, - style: theme.textTheme.titleMedium, - ), - ], - ), - ]; - return vertical ? Column(children: widgets) : Row(children: widgets); + return Column( + mainAxisSize: MainAxisSize.min, + children: [ + Undraw( + illustration: UndrawIllustration.empty, + height: 200 * context.theme.scaling, + color: context.theme.colorScheme.primary, + ), + const Gap(10), + Text( + context.l10n.nothing_found, + textAlign: TextAlign.center, + ).muted().small() + ], + ); } } diff --git a/lib/components/form/checkbox_form_field.dart b/lib/components/form/checkbox_form_field.dart new file mode 100644 index 00000000..0e794833 --- /dev/null +++ b/lib/components/form/checkbox_form_field.dart @@ -0,0 +1,45 @@ +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +class CheckboxFormBuilderField extends StatelessWidget { + final String name; + final FormFieldValidator? validator; + + final ValueChanged? onChanged; + final Widget? leading; + final Widget? trailing; + final bool tristate; + const CheckboxFormBuilderField({ + super.key, + required this.name, + this.validator, + this.onChanged, + this.leading, + this.trailing, + this.tristate = false, + }); + + @override + Widget build(BuildContext context) { + return FormBuilderField( + name: name, + validator: validator, + builder: (field) { + return Checkbox( + state: tristate && field.value == null + ? CheckboxState.indeterminate + : field.value == true + ? CheckboxState.checked + : CheckboxState.unchecked, + onChanged: (state) { + field.didChange(state == CheckboxState.checked); + onChanged?.call(state); + }, + leading: leading, + trailing: trailing, + tristate: tristate, + ); + }, + ); + } +} diff --git a/lib/components/form/text_form_field.dart b/lib/components/form/text_form_field.dart new file mode 100644 index 00000000..dc92c257 --- /dev/null +++ b/lib/components/form/text_form_field.dart @@ -0,0 +1,184 @@ +import 'package:flutter/services.dart'; +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; + +class TextFormBuilderField extends StatelessWidget { + final String name; + final FormFieldValidator? validator; + final Widget? label; + + final TextEditingController? controller; + final bool filled; + final Widget? placeholder; + // final AlignmentGeometry? placeholderAlignment; + // final AlignmentGeometry? leadingAlignment; + // final AlignmentGeometry? trailingAlignment; + final Border? border; + final List features; + final EdgeInsetsGeometry? padding; + final ValueChanged? onSubmitted; + final VoidCallback? onEditingComplete; + final FocusNode? focusNode; + final VoidCallback? onTap; + final bool enabled; + final bool readOnly; + final bool obscureText; + final String obscuringCharacter; + final String? initialValue; + final int? maxLength; + final MaxLengthEnforcement? maxLengthEnforcement; + final int? maxLines; + final int? minLines; + final BorderRadiusGeometry? borderRadius; + final TextAlign textAlign; + final bool expands; + final TextAlignVertical? textAlignVertical; + final UndoHistoryController? undoController; + final ValueChanged? onChanged; + final Iterable? autofillHints; + final void Function(PointerDownEvent event)? onTapOutside; + final List? inputFormatters; + final TextStyle? style; + // final EditableTextContextMenuBuilder? contextMenuBuilder; + // final bool useNativeContextMenu; + // final bool? isCollapsed; + final TextInputType? keyboardType; + final TextInputAction? textInputAction; + final Clip clipBehavior; + final bool autofocus; + final WidgetStatesController? statesController; + + const TextFormBuilderField({ + super.key, + required this.name, + this.label, + this.validator, + this.controller, + this.maxLength, + this.maxLengthEnforcement, + this.maxLines = 1, + this.minLines, + this.filled = false, + this.placeholder, + this.border, + this.padding, + this.onSubmitted, + this.onEditingComplete, + this.focusNode, + this.onTap, + this.enabled = true, + this.readOnly = false, + this.obscureText = false, + this.obscuringCharacter = '•', + this.initialValue, + this.borderRadius, + this.keyboardType, + this.textAlign = TextAlign.start, + this.expands = false, + this.textAlignVertical = TextAlignVertical.center, + this.autofillHints, + this.undoController, + this.onChanged, + this.onTapOutside, + this.inputFormatters, + this.style, + // this.contextMenuBuilder = TextField.defaultContextMenuBuilder, + // this.useNativeContextMenu = false, + // this.isCollapsed, + this.textInputAction, + this.clipBehavior = Clip.hardEdge, + this.autofocus = false, + // this.placeholderAlignment, + // this.leadingAlignment, + // this.trailingAlignment, + this.statesController, + this.features = const [], + }); + + @override + Widget build(BuildContext context) { + return FormBuilderField( + name: name, + validator: validator, + onChanged: (value) { + if (value == null) return; + onChanged?.call(value); + }, + builder: (field) => Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + spacing: 5, + children: [ + if (label != null) + DefaultTextStyle( + style: context.theme.typography.semiBold.copyWith( + color: field.hasError + ? context.theme.colorScheme.destructive + : context.theme.colorScheme.foreground, + ), + child: label!, + ), + TextField( + controller: controller, + maxLength: maxLength, + maxLengthEnforcement: maxLengthEnforcement, + maxLines: maxLines, + minLines: minLines, + filled: filled, + placeholder: placeholder, + border: border, + features: features, + padding: padding, + onSubmitted: (value) { + field.validate(); + field.save(); + onSubmitted?.call(value); + }, + onEditingComplete: () { + field.save(); + onEditingComplete?.call(); + }, + focusNode: focusNode, + onTap: onTap, + enabled: enabled, + readOnly: readOnly, + obscureText: obscureText, + obscuringCharacter: obscuringCharacter, + initialValue: field.value, + borderRadius: borderRadius, + textAlign: textAlign, + expands: expands, + textAlignVertical: textAlignVertical, + autofillHints: autofillHints, + undoController: undoController, + onChanged: (value) { + field.didChange(value); + }, + onTapOutside: onTapOutside, + inputFormatters: inputFormatters, + style: style, + // contextMenuBuilder: contextMenuBuilder, + // useNativeContextMenu: useNativeContextMenu, + // isCollapsed: isCollapsed, + keyboardType: keyboardType, + textInputAction: textInputAction, + clipBehavior: clipBehavior, + autofocus: autofocus, + // placeholderAlignment: placeholderAlignment, + // leadingAlignment: leadingAlignment, + // trailingAlignment: trailingAlignment, + statesController: statesController, + ), + if (field.hasError) + Text( + field.errorText ?? "", + style: TextStyle( + color: context.theme.colorScheme.destructive, + ), + ), + ], + ), + ); + } +} diff --git a/lib/components/framework/app_pop_scope.dart b/lib/components/framework/app_pop_scope.dart index b8e35767..fe923958 100644 --- a/lib/components/framework/app_pop_scope.dart +++ b/lib/components/framework/app_pop_scope.dart @@ -1,6 +1,6 @@ import 'dart:io'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; /// A temporary workaround for [WillPopScope] and [PopScope] not working in GoRouter /// https://github.com/flutter/flutter/issues/140869#issuecomment-2247181468 diff --git a/lib/components/heart_button/heart_button.dart b/lib/components/heart_button/heart_button.dart index fa4318cc..14a0572f 100644 --- a/lib/components/heart_button/heart_button.dart +++ b/lib/components/heart_button/heart_button.dart @@ -1,11 +1,12 @@ -import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/components/heart_button/use_track_toggle_like.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/library/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; class HeartButton extends HookConsumerWidget { final bool isLiked; @@ -13,49 +14,58 @@ class HeartButton extends HookConsumerWidget { final IconData? icon; final Color? color; final String? tooltip; + final AbstractButtonStyle variance; + final ButtonSize size; const HeartButton({ required this.isLiked, required this.onPressed, this.color, this.tooltip, this.icon, + this.variance = ButtonVariance.ghost, + this.size = ButtonSize.normal, super.key, }); @override Widget build(BuildContext context, ref) { - final auth = ref.watch(authenticationProvider); + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); - if (auth.asData?.value == null) return const SizedBox.shrink(); + if (authenticated.asData?.value != true) return const SizedBox.shrink(); - return IconButton( - tooltip: tooltip, - icon: AnimatedSwitcher( - switchInCurve: Curves.fastOutSlowIn, - switchOutCurve: Curves.fastOutSlowIn, - duration: const Duration(milliseconds: 300), - transitionBuilder: (child, animation) { - return ScaleTransition( - scale: animation, - child: child, - ); - }, - child: Icon( - icon ?? - (isLiked - ? Icons.favorite_rounded - : Icons.favorite_outline_rounded), - key: ValueKey(isLiked), - color: color ?? (isLiked ? color ?? Colors.red : null), + return Tooltip( + tooltip: TooltipContainer(child: Text(tooltip ?? "")).call, + child: IconButton( + variance: variance, + size: size, + enabled: onPressed != null, + icon: AnimatedSwitcher( + switchInCurve: Curves.fastOutSlowIn, + switchOutCurve: Curves.fastOutSlowIn, + duration: const Duration(milliseconds: 300), + transitionBuilder: (child, animation) { + return ScaleTransition( + scale: animation, + child: child, + ); + }, + child: Icon( + icon ?? + (isLiked + ? Icons.favorite_rounded + : Icons.favorite_outline_rounded), + key: ValueKey(isLiked), + color: color ?? (isLiked ? color ?? Colors.red : null), + ), ), + onPressed: onPressed, ), - onPressed: onPressed, ); } } class TrackHeartButton extends HookConsumerWidget { - final Track track; + final SpotubeTrackObject track; const TrackHeartButton({ super.key, required this.track, @@ -63,9 +73,10 @@ class TrackHeartButton extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final savedTracks = ref.watch(likedTracksProvider); - final me = ref.watch(meProvider); - final (:isLiked, :toggleTrackLike) = useTrackToggleLike(track, ref); + final savedTracks = ref.watch(metadataPluginSavedTracksProvider); + final me = ref.watch(metadataPluginUserProvider); + final (:isLiked, :isLoading, :toggleTrackLike) = + useTrackToggleLike(track, ref); if (me.isLoading) { return const CircularProgressIndicator(); @@ -76,11 +87,11 @@ class TrackHeartButton extends HookConsumerWidget { ? context.l10n.remove_from_favorites : context.l10n.save_as_favorite, isLiked: isLiked, - onPressed: savedTracks.asData?.value != null - ? () { + onPressed: savedTracks.asData?.value == null || isLoading + ? null + : () { toggleTrackLike(track); - } - : null, + }, ); } } diff --git a/lib/components/heart_button/use_track_toggle_like.dart b/lib/components/heart_button/use_track_toggle_like.dart index ba5cbee1..af961578 100644 --- a/lib/components/heart_button/use_track_toggle_like.dart +++ b/lib/components/heart_button/use_track_toggle_like.dart @@ -1,36 +1,31 @@ -import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/provider/scrobbler/scrobbler.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/library/tracks.dart'; typedef UseTrackToggleLike = ({ bool isLiked, - Future Function(Track track) toggleTrackLike, + bool isLoading, + Future Function(SpotubeTrackObject track) toggleTrackLike, }); -UseTrackToggleLike useTrackToggleLike(Track track, WidgetRef ref) { - final savedTracks = ref.watch(likedTracksProvider); - final savedTracksNotifier = ref.watch(likedTracksProvider.notifier); +UseTrackToggleLike useTrackToggleLike(SpotubeTrackObject track, WidgetRef ref) { + final savedTracksNotifier = + ref.watch(metadataPluginSavedTracksProvider.notifier); - final isLiked = useMemoized( - () => - savedTracks.asData?.value.any((element) => element.id == track.id) ?? - false, - [savedTracks.asData?.value, track.id], - ); - - final scrobblerNotifier = ref.read(scrobblerProvider.notifier); + final isSavedTrack = ref.watch(metadataPluginIsSavedTrackProvider(track.id)); return ( - isLiked: isLiked, + isLiked: isSavedTrack.asData?.value ?? false, + isLoading: isSavedTrack.isLoading, toggleTrackLike: (track) async { - await savedTracksNotifier.toggleFavorite(track); + final isLikedTrack = await ref.read( + metadataPluginIsSavedTrackProvider(track.id).future, + ); - if (!isLiked) { - await scrobblerNotifier.love(track); + if (isLikedTrack) { + await savedTracksNotifier.removeFavorite([track]); } else { - await scrobblerNotifier.unlove(track); + await savedTracksNotifier.addFavorite([track]); } }, ); diff --git a/lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart b/lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart index 16204952..3ac90a06 100644 --- a/lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart +++ b/lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart @@ -1,19 +1,20 @@ import 'dart:ui'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/collections/fake.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/album/album_card.dart'; import 'package:spotube/modules/artist/artist_card.dart'; import 'package:spotube/modules/playlist/playlist_card.dart'; -import 'package:spotube/hooks/utils/use_breakpoint_value.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; class HorizontalPlaybuttonCardView extends HookWidget { final Widget title; final List items; + final Widget? error; final VoidCallback onFetchMore; final bool isLoadingNextPage; final bool hasNextPage; @@ -26,24 +27,22 @@ class HorizontalPlaybuttonCardView extends HookWidget { required this.onFetchMore, required this.isLoadingNextPage, this.titleTrailing, + this.error, super.key, }) : assert( items.every( (item) => - item is PlaylistSimple || item is Artist || item is AlbumSimple, + item is SpotubeSimpleAlbumObject || + item is SpotubeSimplePlaylistObject || + item is SpotubeFullArtistObject, ), ); @override Widget build(BuildContext context) { - final ThemeData(:textTheme) = Theme.of(context); final scrollController = useScrollController(); - final height = useBreakpointValue( - xs: 226, - sm: 226, - md: 236, - others: 266, - ); + final isArtist = items.every((s) => s is SpotubeFullArtistObject); + final scale = context.theme.scaling; return Padding( padding: const EdgeInsets.all(8.0), @@ -54,60 +53,70 @@ class HorizontalPlaybuttonCardView extends HookWidget { Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ - DefaultTextStyle( - style: textTheme.titleMedium!, - child: title, + Flexible( + child: DefaultTextStyle( + style: context.theme.typography.h4.copyWith( + color: context.theme.colorScheme.foreground, + ), + maxLines: 1, + overflow: TextOverflow.ellipsis, + child: title, + ), ), if (titleTrailing != null) titleTrailing!, ], ), - SizedBox( - height: height, - child: NotificationListener( - // disable multiple scrollbar to use this - onNotification: (notification) => true, - child: ScrollConfiguration( - behavior: ScrollConfiguration.of(context).copyWith( - dragDevices: PointerDeviceKind.values.toSet(), - ), - child: items.isEmpty - ? ListView.builder( - scrollDirection: Axis.horizontal, - itemCount: 5, - itemBuilder: (context, index) { - return AlbumCard(FakeData.albumSimple); - }, - ) - : InfiniteList( - scrollController: scrollController, - scrollDirection: Axis.horizontal, - padding: const EdgeInsets.symmetric(vertical: 8.0), - itemCount: items.length, - onFetchData: onFetchMore, - loadingBuilder: (context) => Skeletonizer( - enabled: true, - child: AlbumCard(FakeData.albumSimple), - ), - isLoading: isLoadingNextPage, - hasReachedMax: !hasNextPage, - itemBuilder: (context, index) { - final item = items[index]; - - return switch (item) { - PlaylistSimple() => - PlaylistCard(item as PlaylistSimple), - AlbumSimple() => AlbumCard(item as AlbumSimple), - Artist() => Padding( - padding: const EdgeInsets.symmetric( - horizontal: 12.0), - child: ArtistCard(item as Artist), + if (error != null) + error! + else + SizedBox( + height: isArtist ? 250 : 225, + child: NotificationListener( + // disable multiple scrollbar to use this + onNotification: (notification) => true, + child: ScrollConfiguration( + behavior: ScrollConfiguration.of(context).copyWith( + dragDevices: PointerDeviceKind.values.toSet(), + ), + child: items.isEmpty + ? ListView.builder( + scrollDirection: Axis.horizontal, + itemCount: 5, + itemBuilder: (context, index) { + return AlbumCard(FakeData.albumSimple); + }, + ) + : InfiniteList( + scrollController: scrollController, + scrollDirection: Axis.horizontal, + padding: const EdgeInsets.symmetric(vertical: 8.0), + itemCount: items.length, + onFetchData: onFetchMore, + loadingBuilder: (context) => Skeletonizer( + enabled: true, + child: isArtist + ? ArtistCard(FakeData.artist) + : AlbumCard(FakeData.albumSimple), ), - _ => const SizedBox.shrink(), - }; - }), + isLoading: isLoadingNextPage, + hasReachedMax: !hasNextPage, + separatorBuilder: (context, index) => Gap(12 * scale), + itemBuilder: (context, index) { + final item = items[index]; + + return switch (item) { + SpotubeSimplePlaylistObject() => PlaylistCard( + item as SpotubeSimplePlaylistObject), + SpotubeSimpleAlbumObject() => + AlbumCard(item as SpotubeSimpleAlbumObject), + SpotubeFullArtistObject() => + ArtistCard(item as SpotubeFullArtistObject), + _ => const SizedBox.shrink(), + }; + }), + ), ), ), - ), ], ), ); diff --git a/lib/components/image/universal_image.dart b/lib/components/image/universal_image.dart index d8902e63..e157f96a 100644 --- a/lib/components/image/universal_image.dart +++ b/lib/components/image/universal_image.dart @@ -58,10 +58,10 @@ class UniversalImage extends HookWidget { ), height: height, width: width, - placeholder: AssetImage(placeholder ?? Assets.placeholder.path), + placeholder: AssetImage(placeholder ?? Assets.images.placeholder.path), imageErrorBuilder: (context, error, stackTrace) { return Image.asset( - placeholder ?? Assets.placeholder.path, + placeholder ?? Assets.images.placeholder.path, width: width, height: height, cacheHeight: height?.toInt(), @@ -82,7 +82,7 @@ class UniversalImage extends HookWidget { fit: fit, errorBuilder: (context, error, stackTrace) { return Image.asset( - placeholder ?? Assets.placeholder.path, + placeholder ?? Assets.images.placeholder.path, width: width, height: height, cacheHeight: height?.toInt(), @@ -102,7 +102,7 @@ class UniversalImage extends HookWidget { fit: fit, errorBuilder: (context, error, stackTrace) { return Image.asset( - placeholder ?? Assets.placeholder.path, + placeholder ?? Assets.images.placeholder.path, width: width, height: height, cacheHeight: height?.toInt(), @@ -123,7 +123,7 @@ class UniversalImage extends HookWidget { fit: fit, errorBuilder: (context, error, stackTrace) { return Image.asset( - placeholder ?? Assets.placeholder.path, + placeholder ?? Assets.images.placeholder.path, width: width, height: height, cacheHeight: height?.toInt(), diff --git a/lib/components/inter_scrollbar/inter_scrollbar.dart b/lib/components/inter_scrollbar/inter_scrollbar.dart index 8a86b643..415ba6da 100644 --- a/lib/components/inter_scrollbar/inter_scrollbar.dart +++ b/lib/components/inter_scrollbar/inter_scrollbar.dart @@ -1,5 +1,5 @@ import 'package:draggable_scrollbar/draggable_scrollbar.dart'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:spotube/utils/platform.dart'; diff --git a/lib/components/links/anchor_button.dart b/lib/components/links/anchor_button.dart index c6f0b889..a0b3fa73 100644 --- a/lib/components/links/anchor_button.dart +++ b/lib/components/links/anchor_button.dart @@ -1,5 +1,5 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; class AnchorButton extends HookWidget { final String text; diff --git a/lib/components/links/artist_link.dart b/lib/components/links/artist_link.dart index 9f06f1b3..dc093345 100644 --- a/lib/components/links/artist_link.dart +++ b/lib/components/links/artist_link.dart @@ -1,12 +1,12 @@ -import 'package:flutter/material.dart'; -import 'package:spotify/spotify.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/links/anchor_button.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/artist/artist.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:spotube/models/metadata/metadata.dart'; class ArtistLink extends StatelessWidget { - final List artists; + final List artists; final WrapCrossAlignment crossAxisAlignment; final WrapAlignment mainAxisAlignment; final TextStyle textStyle; @@ -38,24 +38,16 @@ class ArtistLink extends StatelessWidget { .entries .map( (artist) => Builder(builder: (context) { - if (artist.value.name == null) { - return Text("Spotify", style: textStyle); - } return AnchorButton( (artist.key != artists.length - 1) ? "${artist.value.name}, " - : artist.value.name!, + : artist.value.name, onTap: () { if (onRouteChange != null) { onRouteChange?.call("/artist/${artist.value.id}"); } else { - ServiceUtils.pushNamed( - context, - ArtistPage.name, - pathParameters: { - "id": artist.value.id!, - }, - ); + context + .navigateTo(ArtistRoute(artistId: artist.value.id)); } }, overflow: TextOverflow.ellipsis, diff --git a/lib/components/links/hyper_link.dart b/lib/components/links/hyper_link.dart index 32d715e0..647edaca 100644 --- a/lib/components/links/hyper_link.dart +++ b/lib/components/links/hyper_link.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/links/anchor_button.dart'; import 'package:url_launcher/url_launcher_string.dart'; diff --git a/lib/components/links/link_text.dart b/lib/components/links/link_text.dart index 0cab71d0..c64ae93d 100644 --- a/lib/components/links/link_text.dart +++ b/lib/components/links/link_text.dart @@ -1,15 +1,14 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/links/anchor_button.dart'; -import 'package:spotube/utils/service_utils.dart'; class LinkText extends StatelessWidget { final String text; final TextStyle style; final TextAlign? textAlign; final TextOverflow? overflow; - final String route; + final PageRouteInfo route; final int? maxLines; - final T? extra; final bool push; const LinkText( @@ -17,7 +16,6 @@ class LinkText extends StatelessWidget { this.route, { super.key, this.textAlign, - this.extra, this.overflow, this.style = const TextStyle(), this.maxLines, @@ -30,9 +28,9 @@ class LinkText extends StatelessWidget { text, onTap: () { if (push) { - ServiceUtils.push(context, route, extra: extra); + context.navigateTo(route); } else { - ServiceUtils.navigate(context, route, extra: extra); + context.navigateTo(route); } }, key: key, diff --git a/lib/components/markdown/markdown.dart b/lib/components/markdown/markdown.dart new file mode 100644 index 00000000..1fd4ac5b --- /dev/null +++ b/lib/components/markdown/markdown.dart @@ -0,0 +1,42 @@ +import 'package:cached_network_image/cached_network_image.dart'; +import 'package:flutter_markdown_plus/flutter_markdown_plus.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/dialogs/link_open_permission_dialog.dart'; +import 'package:url_launcher/url_launcher_string.dart'; + +class AppMarkdown extends StatelessWidget { + final String data; + const AppMarkdown({ + super.key, + required this.data, + }); + + @override + Widget build(BuildContext context) { + return MarkdownBody( + data: data, + imageBuilder: (uri, title, alt) { + final url = uri.toString(); + return CachedNetworkImage( + imageUrl: url, + fit: BoxFit.cover, + ); + }, + onTapLink: (text, href, title) async { + final allowOpeningLink = await showDialog( + context: context, + builder: (context) { + return LinkOpenPermissionDialog(href: href); + }, + ); + + if (href != null && allowOpeningLink == true) { + launchUrlString( + href, + mode: LaunchMode.externalApplication, + ); + } + }, + ); + } +} diff --git a/lib/components/panels/controller.dart b/lib/components/panels/controller.dart deleted file mode 100644 index 4e367701..00000000 --- a/lib/components/panels/controller.dart +++ /dev/null @@ -1,146 +0,0 @@ -part of 'sliding_up_panel.dart'; - -class PanelController extends ChangeNotifier { - SlidingUpPanelState? _panelState; - - void _addState(SlidingUpPanelState panelState) { - _panelState = panelState; - notifyListeners(); - } - - bool _forceScrollChange = false; - - /// use this function when scroll change in func - /// Example: - /// panelController.forseScrollChange(scrollController.animateTo(100, duration: Duration(milliseconds: 400), curve: Curves.ease)) - Future forceScrollChange(Future func) async { - _forceScrollChange = true; - _panelState!._scrollingEnabled = true; - await func; - // if (_panelState!._sc.offset == 0) { - // _panelState!._scrollingEnabled = true; - // } - if (panelPosition < 1) { - _panelState!._scMinOffset = _panelState!._scrollController.offset; - } - _forceScrollChange = false; - } - - bool __nowTargetForceDraggable = false; - - bool get _nowTargetForceDraggable => __nowTargetForceDraggable; - - set _nowTargetForceDraggable(bool value) { - __nowTargetForceDraggable = value; - notifyListeners(); - } - - /// Determine if the panelController is attached to an instance - /// of the SlidingUpPanel (this property must return true before any other - /// functions can be used) - bool get isAttached => _panelState != null; - - /// Closes the sliding panel to its collapsed state (i.e. to the minHeight) - Future close() async { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - await _panelState!._close(); - notifyListeners(); - } - - /// Opens the sliding panel fully - /// (i.e. to the maxHeight) - Future open() async { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - await _panelState!._open(); - notifyListeners(); - } - - /// Hides the sliding panel (i.e. is invisible) - Future hide() async { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - await _panelState!._hide(); - notifyListeners(); - } - - /// Shows the sliding panel in its collapsed state - /// (i.e. "un-hide" the sliding panel) - Future show() async { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - await _panelState!._show(); - notifyListeners(); - } - - /// Animates the panel position to the value. - /// The value must between 0.0 and 1.0 - /// where 0.0 is fully collapsed and 1.0 is completely open. - /// (optional) duration specifies the time for the animation to complete - /// (optional) curve specifies the easing behavior of the animation. - Future animatePanelToPosition(double value, - {Duration? duration, Curve curve = Curves.linear}) { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - assert(0.0 <= value && value <= 1.0); - return _panelState! - ._animatePanelToPosition(value, duration: duration, curve: curve); - } - - /// Animates the panel position to the snap point - /// Requires that the SlidingUpPanel snapPoint property is not null - /// (optional) duration specifies the time for the animation to complete - /// (optional) curve specifies the easing behavior of the animation. - Future animatePanelToSnapPoint( - {Duration? duration, Curve curve = Curves.linear}) { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - assert(_panelState!.widget.snapPoint != null, - "SlidingUpPanel snapPoint property must not be null"); - return _panelState! - ._animatePanelToSnapPoint(duration: duration, curve: curve); - } - - /// Sets the panel position (without animation). - /// The value must between 0.0 and 1.0 - /// where 0.0 is fully collapsed and 1.0 is completely open. - set panelPosition(double value) { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - assert(0.0 <= value && value <= 1.0); - _panelState!._panelPosition = value; - } - - /// Gets the current panel position. - /// Returns the % offset from collapsed state - /// to the open state - /// as a decimal between 0.0 and 1.0 - /// where 0.0 is fully collapsed and - /// 1.0 is full open. - double get panelPosition { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - return _panelState!._panelPosition; - } - - /// Returns whether or not the panel is - /// currently animating. - bool get isPanelAnimating { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - return _panelState!._isPanelAnimating; - } - - /// Returns whether or not the - /// panel is open. - bool get isPanelOpen { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - return _panelState!._isPanelOpen; - } - - /// Returns whether or not the - /// panel is closed. - bool get isPanelClosed { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - return _panelState!._isPanelClosed; - } - - /// Returns whether or not the - /// panel is shown/hidden. - bool get isPanelShown { - assert(isAttached, "PanelController must be attached to a SlidingUpPanel"); - return _panelState!._isPanelShown; - } -} diff --git a/lib/components/panels/helpers.dart b/lib/components/panels/helpers.dart deleted file mode 100644 index d79fa97c..00000000 --- a/lib/components/panels/helpers.dart +++ /dev/null @@ -1,95 +0,0 @@ -part of "sliding_up_panel.dart"; - -/// if you want to prevent the panel from being dragged using the widget, -/// wrap the widget with this -class IgnoreDraggableWidget extends SingleChildRenderObjectWidget { - const IgnoreDraggableWidget({ - super.key, - required super.child, - }); - - @override - IgnoreDraggableWidgetWidgetRenderBox createRenderObject( - BuildContext context, - ) { - return IgnoreDraggableWidgetWidgetRenderBox(); - } -} - -class IgnoreDraggableWidgetWidgetRenderBox extends RenderPointerListener { - @override - HitTestBehavior get behavior => HitTestBehavior.opaque; -} - -/// if you want to force the panel to be dragged using the widget, -/// wrap the widget with this -/// For example, use [Scrollable] inside to allow the panel to be dragged -/// even if the scroll is not at position 0. -class ForceDraggableWidget extends SingleChildRenderObjectWidget { - const ForceDraggableWidget({ - super.key, - required super.child, - }); - - @override - ForceDraggableWidgetRenderBox createRenderObject( - BuildContext context, - ) { - return ForceDraggableWidgetRenderBox(); - } -} - -class ForceDraggableWidgetRenderBox extends RenderPointerListener { - @override - HitTestBehavior get behavior => HitTestBehavior.opaque; -} - -/// To make [ForceDraggableWidget] work in [Scrollable] widgets -class PanelScrollPhysics extends ScrollPhysics { - final PanelController controller; - const PanelScrollPhysics({required this.controller, super.parent}); - @override - PanelScrollPhysics applyTo(ScrollPhysics? ancestor) { - return PanelScrollPhysics( - controller: controller, parent: buildParent(ancestor)); - } - - @override - double applyPhysicsToUserOffset(ScrollMetrics position, double offset) { - if (controller._nowTargetForceDraggable) return 0.0; - return super.applyPhysicsToUserOffset(position, offset); - } - - @override - Simulation? createBallisticSimulation( - ScrollMetrics position, double velocity) { - if (controller._nowTargetForceDraggable) { - return super.createBallisticSimulation(position, 0); - } - return super.createBallisticSimulation(position, velocity); - } - - @override - bool get allowImplicitScrolling => false; -} - -/// if you want to prevent unwanted panel dragging when scrolling widgets [Scrollable] with horizontal axis -/// wrap the widget with this -class HorizontalScrollableWidget extends SingleChildRenderObjectWidget { - const HorizontalScrollableWidget({ - super.key, - required super.child, - }); - - @override - HorizontalScrollableWidgetRenderBox createRenderObject( - BuildContext context, - ) { - return HorizontalScrollableWidgetRenderBox(); - } -} - -class HorizontalScrollableWidgetRenderBox extends RenderPointerListener { - @override - HitTestBehavior get behavior => HitTestBehavior.opaque; -} diff --git a/lib/components/panels/sliding_up_panel.dart b/lib/components/panels/sliding_up_panel.dart deleted file mode 100644 index e99fe261..00000000 --- a/lib/components/panels/sliding_up_panel.dart +++ /dev/null @@ -1,685 +0,0 @@ -/* -Name: Zotov Vladimir -Date: 18/06/22 -Purpose: Defines the package: sliding_up_panel2 -Copyright: © 2022, Zotov Vladimir. All rights reserved. -Licensing: More information can be found here: https://github.com/Zotov-VD/sliding_up_panel/blob/master/LICENSE - -This product includes software developed by Akshath Jain (https://akshathjain.com) -*/ - -library panels; - -import 'dart:math'; - -import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter/physics.dart'; -import 'package:flutter/rendering.dart'; - -part 'controller.dart'; -part 'helpers.dart'; - -enum SlideDirection { up, down } - -enum PanelState { open, closed } - -class SlidingUpPanel extends StatefulWidget { - /// Returns the Widget that slides into view. When the - /// panel is collapsed and if [collapsed] is null, - /// then top portion of this Widget will be displayed; - /// otherwise, [collapsed] will be displayed overtop - /// of this Widget. - final Widget? Function(double position)? panelBuilder; - - /// The Widget displayed overtop the [panel] when collapsed. - /// This fades out as the panel is opened. - final Widget? collapsed; - - /// The Widget that lies underneath the sliding panel. - /// This Widget automatically sizes itself - /// to fill the screen. - final Widget? body; - - /// Optional persistent widget that floats above the [panel] and attaches - /// to the top of the [panel]. Content at the top of the panel will be covered - /// by this widget. Add padding to the bottom of the `panel` to - /// avoid coverage. - final Widget? header; - - /// Optional persistent widget that floats above the [panel] and - /// attaches to the bottom of the [panel]. Content at the bottom of the panel - /// will be covered by this widget. Add padding to the bottom of the `panel` - /// to avoid coverage. - final Widget? footer; - - /// The height of the sliding panel when fully collapsed. - final double minHeight; - - /// The height of the sliding panel when fully open. - final double maxHeight; - - /// A point between [minHeight] and [maxHeight] that the panel snaps to - /// while animating. A fast swipe on the panel will disregard this point - /// and go directly to the open/close position. This value is represented as a - /// percentage of the total animation distance ([maxHeight] - [minHeight]), - /// so it must be between 0.0 and 1.0, exclusive. - final double? snapPoint; - - /// The amount to inset the children of the sliding panel sheet. - final EdgeInsetsGeometry? padding; - - /// Empty space surrounding the sliding panel sheet. - final EdgeInsetsGeometry? margin; - - /// Set to false to disable the panel from snapping open or closed. - final bool panelSnapping; - - /// Disable panel draggable on scrolling. Defaults to false. - final bool disableDraggableOnScrolling; - - /// If non-null, this can be used to control the state of the panel. - final PanelController? controller; - - /// If non-null, shows a darkening shadow over the [body] as the panel slides open. - final bool backdropEnabled; - - /// Shows a darkening shadow of this [Color] over the [body] as the panel slides open. - final Color backdropColor; - - /// The opacity of the backdrop when the panel is fully open. - /// This value can range from 0.0 to 1.0 where 0.0 is completely transparent - /// and 1.0 is completely opaque. - final double backdropOpacity; - - /// Flag that indicates whether or not tapping the - /// backdrop closes the panel. Defaults to true. - final bool backdropTapClosesPanel; - - /// If non-null, this callback - /// is called as the panel slides around with the - /// current position of the panel. The position is a double - /// between 0.0 and 1.0 where 0.0 is fully collapsed and 1.0 is fully open. - final void Function(double position)? onPanelSlide; - - /// If non-null, this callback is called when the - /// panel is fully opened - final VoidCallback? onPanelOpened; - - /// If non-null, this callback is called when the panel - /// is fully collapsed. - final VoidCallback? onPanelClosed; - - /// If non-null and true, the SlidingUpPanel exhibits a - /// parallax effect as the panel slides up. Essentially, - /// the body slides up as the panel slides up. - final bool parallaxEnabled; - - /// Allows for specifying the extent of the parallax effect in terms - /// of the percentage the panel has slid up/down. Recommended values are - /// within 0.0 and 1.0 where 0.0 is no parallax and 1.0 mimics a - /// one-to-one scrolling effect. Defaults to a 10% parallax. - final double parallaxOffset; - - /// Allows toggling of the draggability of the SlidingUpPanel. - /// Set this to false to prevent the user from being able to drag - /// the panel up and down. Defaults to true. - final bool isDraggable; - - /// Either SlideDirection.UP or SlideDirection.DOWN. Indicates which way - /// the panel should slide. Defaults to UP. If set to DOWN, the panel attaches - /// itself to the top of the screen and is fully opened when the user swipes - /// down on the panel. - final SlideDirection slideDirection; - - /// The default state of the panel; either PanelState.OPEN or PanelState.CLOSED. - /// This value defaults to PanelState.CLOSED which indicates that the panel is - /// in the closed position and must be opened. PanelState.OPEN indicates that - /// by default the Panel is open and must be swiped closed by the user. - final PanelState defaultPanelState; - - /// To attach to a [Scrollable] on a panel that - /// links the panel's position to the scroll position. Useful for implementing - /// infinite scroll behavior - final ScrollController? scrollController; - - final BoxDecoration? panelDecoration; - - const SlidingUpPanel( - {super.key, - this.body, - this.collapsed, - this.minHeight = 100.0, - this.maxHeight = 500.0, - this.snapPoint, - this.padding, - this.margin, - this.panelDecoration, - this.panelSnapping = true, - this.disableDraggableOnScrolling = false, - this.controller, - this.backdropEnabled = false, - this.backdropColor = Colors.black, - this.backdropOpacity = 0.5, - this.backdropTapClosesPanel = true, - this.onPanelSlide, - this.onPanelOpened, - this.onPanelClosed, - this.parallaxEnabled = false, - this.parallaxOffset = 0.1, - this.isDraggable = true, - this.slideDirection = SlideDirection.up, - this.defaultPanelState = PanelState.closed, - this.header, - this.footer, - this.scrollController, - this.panelBuilder}) - : assert(panelBuilder != null), - assert(0 <= backdropOpacity && backdropOpacity <= 1.0), - assert(snapPoint == null || 0 < snapPoint && snapPoint < 1.0); - - @override - SlidingUpPanelState createState() => SlidingUpPanelState(); -} - -class SlidingUpPanelState extends State - with SingleTickerProviderStateMixin { - late AnimationController _animationController; - late final ScrollController _scrollController; - - bool _scrollingEnabled = false; - final VelocityTracker _velocityTracker = - VelocityTracker.withKind(PointerDeviceKind.touch); - - bool _isPanelVisible = true; - - @override - void initState() { - super.initState(); - - _animationController = AnimationController( - vsync: this, - duration: const Duration(milliseconds: 300), - value: widget.defaultPanelState == PanelState.closed - ? 0.0 - : 1.0 //set the default panel state (i.e. set initial value of _ac) - ) - ..addListener(() { - if (widget.onPanelSlide != null) { - widget.onPanelSlide!(_animationController.value); - } - - if (widget.onPanelOpened != null && - (_animationController.value == 1.0 || - _animationController.value == 0.0)) { - widget.onPanelOpened!(); - } - }); - - // prevent the panel content from being scrolled only if the widget is - // draggable and panel scrolling is enabled - _scrollController = widget.scrollController ?? ScrollController(); - _scrollController.addListener(() { - if (widget.isDraggable && - !widget.disableDraggableOnScrolling && - (!_scrollingEnabled || _panelPosition < 1) && - widget.controller?._forceScrollChange != true) { - _scrollController.jumpTo(_scMinOffset); - } - }); - - widget.controller?._addState(this); - } - - @override - Widget build(BuildContext context) { - final mediaQuery = MediaQuery.of(context); - - return Stack( - alignment: widget.slideDirection == SlideDirection.up - ? Alignment.bottomCenter - : Alignment.topCenter, - children: [ - //make the back widget take up the entire back side - if (widget.body != null) - AnimatedBuilder( - animation: _animationController, - builder: (context, child) { - return Positioned( - top: widget.parallaxEnabled ? _getParallax() : 0.0, - child: child ?? const SizedBox(), - ); - }, - child: SizedBox( - height: mediaQuery.size.height, - width: mediaQuery.size.width, - child: widget.body, - ), - ), - - //the backdrop to overlay on the body - if (widget.backdropEnabled) - GestureDetector( - onVerticalDragEnd: widget.backdropTapClosesPanel - ? (DragEndDetails details) { - // only trigger a close if the drag is towards panel close position - if ((widget.slideDirection == SlideDirection.up ? 1 : -1) * - details.velocity.pixelsPerSecond.dy > - 0) _close(); - } - : null, - onTap: widget.backdropTapClosesPanel ? () => _close() : null, - child: AnimatedBuilder( - animation: _animationController, - builder: (context, _) { - return Container( - height: mediaQuery.size.height, - width: mediaQuery.size.width, - - //set color to null so that touch events pass through - //to the body when the panel is closed, otherwise, - //if a color exists, then touch events won't go through - color: _animationController.value == 0.0 - ? null - : widget.backdropColor.withOpacity( - widget.backdropOpacity * _animationController.value, - ), - ); - }), - ), - - //the actual sliding part - if (_isPanelVisible) - _gestureHandler( - child: AnimatedBuilder( - animation: _animationController, - builder: (context, child) { - return Container( - height: _animationController.value * - (widget.maxHeight - widget.minHeight) + - widget.minHeight, - margin: widget.margin, - padding: widget.padding, - decoration: widget.panelDecoration, - child: child, - ); - }, - child: Stack( - children: [ - //open panel - Positioned( - top: - widget.slideDirection == SlideDirection.up ? 0.0 : null, - bottom: widget.slideDirection == SlideDirection.down - ? 0.0 - : null, - width: mediaQuery.size.width - - (widget.margin != null - ? widget.margin!.horizontal - : 0) - - (widget.padding != null - ? widget.padding!.horizontal - : 0), - child: SizedBox( - height: widget.maxHeight, - child: widget.panelBuilder!( - _animationController.value, - ), - ), - ), - - // footer - if (widget.footer != null) - Positioned( - top: widget.slideDirection == SlideDirection.up - ? null - : 0.0, - bottom: widget.slideDirection == SlideDirection.down - ? null - : 0.0, - child: widget.footer ?? const SizedBox()), - - // header - if (widget.header != null) - Positioned( - top: widget.slideDirection == SlideDirection.up - ? 0.0 - : null, - bottom: widget.slideDirection == SlideDirection.down - ? 0.0 - : null, - child: widget.header ?? const SizedBox(), - ), - - // collapsed panel - Positioned( - top: - widget.slideDirection == SlideDirection.up ? 0.0 : null, - bottom: widget.slideDirection == SlideDirection.down - ? 0.0 - : null, - width: mediaQuery.size.width - - (widget.margin != null - ? widget.margin!.horizontal - : 0) - - (widget.padding != null - ? widget.padding!.horizontal - : 0), - child: AnimatedContainer( - duration: const Duration(milliseconds: 250), - height: widget.minHeight, - child: widget.collapsed == null - ? null - : FadeTransition( - opacity: Tween(begin: 1.0, end: 0.0) - .animate(_animationController), - - // if the panel is open ignore pointers (touch events) on the collapsed - // child so that way touch events go through to whatever is underneath - child: IgnorePointer( - ignoring: _animationController.value == 1.0, - child: widget.collapsed, - ), - ), - ), - ), - ], - ), - ), - ), - ], - ); - } - - @override - void dispose() { - _animationController.dispose(); - super.dispose(); - } - - double _getParallax() { - if (widget.slideDirection == SlideDirection.up) { - return -_animationController.value * - (widget.maxHeight - widget.minHeight) * - widget.parallaxOffset; - } else { - return _animationController.value * - (widget.maxHeight - widget.minHeight) * - widget.parallaxOffset; - } - } - - bool _ignoreScrollable = false; - bool _isHorizontalScrollableWidget = false; - Axis? _scrollableAxis; - - // returns a gesture detector if panel is used - // and a listener if panelBuilder is used. - // this is because the listener is designed only for use with linking the scrolling of - // panels and using it for panels that don't want to linked scrolling yields odd results - Widget _gestureHandler({required Widget child}) { - if (!widget.isDraggable) return child; - - return Listener( - onPointerDown: (PointerDownEvent e) { - var rb = context.findRenderObject() as RenderBox; - var result = BoxHitTestResult(); - rb.hitTest(result, position: e.position); - - if (_panelPosition == 1) { - _scMinOffset = 0.0; - } - // if there any widget in the path that must force graggable, - // stop it right here - if (result.path.any((entry) => - entry.target.runtimeType == ForceDraggableWidgetRenderBox)) { - widget.controller?._nowTargetForceDraggable = true; - _scMinOffset = _scrollController.offset; - _isHorizontalScrollableWidget = false; - } else if (result.path.any((entry) => - entry.target.runtimeType == HorizontalScrollableWidgetRenderBox)) { - _isHorizontalScrollableWidget = true; - widget.controller?._nowTargetForceDraggable = false; - } else if (result.path.any((entry) => - entry.target.runtimeType == IgnoreDraggableWidgetWidgetRenderBox)) { - _ignoreScrollable = true; - widget.controller?._nowTargetForceDraggable = false; - _isHorizontalScrollableWidget = false; - return; - } else { - widget.controller?._nowTargetForceDraggable = false; - _isHorizontalScrollableWidget = false; - } - _ignoreScrollable = false; - _velocityTracker.addPosition(e.timeStamp, e.position); - }, - onPointerMove: (PointerMoveEvent e) { - if (_scrollableAxis == null) { - if (e.delta.dx.abs() > e.delta.dy.abs()) { - _scrollableAxis = Axis.horizontal; - } else { - _scrollableAxis = Axis.vertical; - } - } - - if (_isHorizontalScrollableWidget && - _scrollableAxis == Axis.horizontal) { - return; - } - - if (_ignoreScrollable) return; - _velocityTracker.addPosition( - e.timeStamp, - e.position, - ); // add current position for velocity tracking - _onGestureSlide(e.delta.dy); - }, - onPointerUp: (PointerUpEvent e) { - if (_ignoreScrollable) return; - _scrollableAxis = null; - _onGestureEnd(_velocityTracker.getVelocity()); - }, - child: child, - ); - } - - double _scMinOffset = 0.0; - - // handles the sliding gesture - void _onGestureSlide(double dy) { - // only slide the panel if scrolling is not enabled - if (widget.controller?._nowTargetForceDraggable == false && - widget.disableDraggableOnScrolling) { - return; - } - if ((!_scrollingEnabled) || - _panelPosition < 1 || - widget.controller?._nowTargetForceDraggable == true) { - if (widget.slideDirection == SlideDirection.up) { - _animationController.value -= - dy / (widget.maxHeight - widget.minHeight); - } else { - _animationController.value += - dy / (widget.maxHeight - widget.minHeight); - } - } - - // if the panel is open and the user hasn't scrolled, we need to determine - // whether to enable scrolling if the user swipes up, or disable closing and - // begin to close the panel if the user swipes down - if (_isPanelOpen && - _scrollController.hasClients && - _scrollController.offset <= _scMinOffset) { - setState(() { - if (dy < 0) { - _scrollingEnabled = true; - } else { - _scrollingEnabled = false; - } - }); - } - } - - // handles when user stops sliding - void _onGestureEnd(Velocity v) { - if (widget.controller?._nowTargetForceDraggable == false && - widget.disableDraggableOnScrolling) { - return; - } - double minFlingVelocity = 365.0; - double kSnap = 8; - - //let the current animation finish before starting a new one - if (_animationController.isAnimating) return; - - // if scrolling is allowed and the panel is open, we don't want to close - // the panel if they swipe up on the scrollable - if (_isPanelOpen && _scrollingEnabled) return; - - //check if the velocity is sufficient to constitute fling to end - double visualVelocity = - -v.pixelsPerSecond.dy / (widget.maxHeight - widget.minHeight); - - // reverse visual velocity to account for slide direction - if (widget.slideDirection == SlideDirection.down) { - visualVelocity = -visualVelocity; - } - - // get minimum distances to figure out where the panel is at - double d2Close = _animationController.value; - double d2Open = 1 - _animationController.value; - double d2Snap = ((widget.snapPoint ?? 3) - _animationController.value) - .abs(); // large value if null results in not every being the min - double minDistance = min(d2Close, min(d2Snap, d2Open)); - - // check if velocity is sufficient for a fling - if (v.pixelsPerSecond.dy.abs() >= minFlingVelocity) { - // snapPoint exists - if (widget.panelSnapping && widget.snapPoint != null) { - if (v.pixelsPerSecond.dy.abs() >= kSnap * minFlingVelocity || - minDistance == d2Snap) { - _animationController.fling(velocity: visualVelocity); - } else { - _flingPanelToPosition(widget.snapPoint!, visualVelocity); - } - - // no snap point exists - } else if (widget.panelSnapping) { - _animationController.fling(velocity: visualVelocity); - - // panel snapping disabled - } else { - _animationController.animateTo( - _animationController.value + visualVelocity * 0.16, - duration: const Duration(milliseconds: 410), - curve: Curves.decelerate, - ); - } - - return; - } - - // check if the controller is already halfway there - if (widget.panelSnapping) { - if (minDistance == d2Close) { - _close(); - } else if (minDistance == d2Snap) { - _flingPanelToPosition(widget.snapPoint!, visualVelocity); - } else { - _open(); - } - } - } - - void _flingPanelToPosition(double targetPos, double velocity) { - final Simulation simulation = SpringSimulation( - SpringDescription.withDampingRatio( - mass: 1.0, - stiffness: 500.0, - ratio: 1.0, - ), - _animationController.value, - targetPos, - velocity); - - _animationController.animateWith(simulation); - } - - //--------------------------------- - //PanelController related functions - //--------------------------------- - - //close the panel - Future _close() { - return _animationController.fling(velocity: -1.0); - } - - //open the panel - Future _open() { - return _animationController.fling(velocity: 1.0); - } - - //hide the panel (completely offscreen) - Future _hide() { - return _animationController.fling(velocity: -1.0).then((x) { - setState(() { - _isPanelVisible = false; - }); - }); - } - - //show the panel (in collapsed mode) - Future _show() { - return _animationController.fling(velocity: -1.0).then((x) { - setState(() { - _isPanelVisible = true; - }); - }); - } - - //animate the panel position to value - must - //be between 0.0 and 1.0 - Future _animatePanelToPosition(double value, - {Duration? duration, Curve curve = Curves.linear}) { - assert(0.0 <= value && value <= 1.0); - return _animationController.animateTo(value, - duration: duration, curve: curve); - } - - //animate the panel position to the snap point - //REQUIRES that widget.snapPoint != null - Future _animatePanelToSnapPoint( - {Duration? duration, Curve curve = Curves.linear}) { - assert(widget.snapPoint != null); - return _animationController.animateTo(widget.snapPoint!, - duration: duration, curve: curve); - } - - //set the panel position to value - must - //be between 0.0 and 1.0 - set _panelPosition(double value) { - assert(0.0 <= value && value <= 1.0); - _animationController.value = value; - } - - //get the current panel position - //returns the % offset from collapsed state - //as a decimal between 0.0 and 1.0 - double get _panelPosition => _animationController.value; - - //returns whether or not - //the panel is still animating - bool get _isPanelAnimating => _animationController.isAnimating; - - //returns whether or not the - //panel is open - bool get _isPanelOpen => _animationController.value == 1.0; - - //returns whether or not the - //panel is closed - bool get _isPanelClosed => _animationController.value == 0.0; - - //returns whether or not the - //panel is shown/hidden - bool get _isPanelShown => _isPanelVisible; -} diff --git a/lib/components/playbutton_card.dart b/lib/components/playbutton_card.dart deleted file mode 100644 index ae9050d8..00000000 --- a/lib/components/playbutton_card.dart +++ /dev/null @@ -1,220 +0,0 @@ -import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/hover_builder.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/string.dart'; -import 'package:spotube/hooks/utils/use_breakpoint_value.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; - -class PlaybuttonCard extends HookWidget { - final void Function()? onTap; - final void Function()? onPlaybuttonPressed; - final void Function()? onAddToQueuePressed; - final String? description; - final EdgeInsetsGeometry? margin; - final String imageUrl; - final bool isPlaying; - final bool isLoading; - final String title; - final bool isOwner; - - const PlaybuttonCard({ - required this.imageUrl, - required this.isPlaying, - required this.isLoading, - required this.title, - this.margin, - this.description, - this.onPlaybuttonPressed, - this.onAddToQueuePressed, - this.onTap, - this.isOwner = false, - super.key, - }); - - @override - Widget build(BuildContext context) { - final textsKey = useMemoized(() => GlobalKey(), []); - final theme = Theme.of(context); - final mediaQuery = MediaQuery.of(context); - final radius = BorderRadius.circular(15); - - final double size = useBreakpointValue( - xs: 130, - sm: 130, - md: 150, - others: 170, - ); - - final end = useBreakpointValue( - xs: 7, - sm: 7, - others: 15, - ); - - final unescapeHtml = description?.unescapeHtml().cleanHtml(); - return Container( - constraints: BoxConstraints(maxWidth: size), - margin: margin, - child: Material( - color: Color.lerp( - theme.colorScheme.surfaceContainerHighest, - theme.colorScheme.surface, - useBrightnessValue(.9, .7), - ), - borderRadius: radius, - shadowColor: theme.colorScheme.surface, - elevation: 3, - child: InkWell( - mouseCursor: SystemMouseCursors.click, - onTap: onTap, - borderRadius: radius, - splashFactory: theme.splashFactory, - child: Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: CrossAxisAlignment.stretch, - children: [ - Stack( - clipBehavior: Clip.none, - children: [ - Container( - margin: const EdgeInsets.fromLTRB(8, 8, 8, 0), - padding: const EdgeInsets.only( - left: 8, - right: 8, - top: 8, - ), - height: mediaQuery.smAndDown - ? 120 - : mediaQuery.mdAndDown - ? 130 - : 150, - decoration: BoxDecoration( - borderRadius: radius, - image: DecorationImage( - image: UniversalImage.imageProvider(imageUrl), - fit: BoxFit.cover, - ), - ), - ), - if (isOwner) - Positioned( - top: 15, - left: 15, - child: AnimatedSize( - duration: const Duration(milliseconds: 150), - alignment: Alignment.centerLeft, - curve: Curves.easeInExpo, - child: HoverBuilder(builder: (context, isHovered) { - return Container( - padding: const EdgeInsets.all(4), - decoration: BoxDecoration( - color: Colors.blueAccent, - borderRadius: BorderRadius.circular(20), - ), - child: Row( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon( - SpotubeIcons.user, - color: Colors.white, - size: 16, - ), - if (isHovered) - Text( - context.l10n.owned_by_you, - style: theme.textTheme.bodySmall?.copyWith( - color: Colors.white, - ), - ), - ], - ), - ); - }), - ), - ), - Positioned( - right: end, - bottom: -15, - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - if (!isPlaying) - Skeleton.keep( - child: IconButton( - style: IconButton.styleFrom( - backgroundColor: theme.colorScheme.surface, - foregroundColor: theme.colorScheme.primary, - minimumSize: const Size.square(10), - ), - icon: const Icon(SpotubeIcons.queueAdd), - onPressed: isLoading ? null : onAddToQueuePressed, - ), - ), - const Gap(5), - IconButton( - style: IconButton.styleFrom( - backgroundColor: theme.colorScheme.primaryContainer, - foregroundColor: theme.colorScheme.primary, - minimumSize: const Size.square(10), - ), - icon: Skeleton.keep( - child: isLoading - ? SizedBox.fromSize( - size: const Size.square(15), - child: const CircularProgressIndicator( - strokeWidth: 2), - ) - : isPlaying - ? const Icon(SpotubeIcons.pause) - : const Icon(SpotubeIcons.play), - ), - onPressed: isLoading ? null : onPlaybuttonPressed, - ), - ], - ), - ), - ], - ), - Column( - key: textsKey, - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - const SizedBox(height: 15), - Padding( - padding: const EdgeInsets.symmetric(horizontal: 12.0), - child: AutoSizeText( - title, - maxLines: 1, - minFontSize: theme.textTheme.bodyMedium!.fontSize!, - overflow: TextOverflow.ellipsis, - ), - ), - if (description != null) - Padding( - padding: const EdgeInsets.symmetric(horizontal: 12.0), - child: AutoSizeText( - unescapeHtml!, - maxLines: 2, - style: theme.textTheme.bodySmall?.copyWith( - color: theme.colorScheme.onSurface.withOpacity(.5), - ), - overflow: TextOverflow.ellipsis, - ), - ), - const SizedBox(height: 10), - ], - ), - ], - ), - ), - ), - ); - } -} diff --git a/lib/components/playbutton_view/playbutton_card.dart b/lib/components/playbutton_view/playbutton_card.dart new file mode 100644 index 00000000..ea28c738 --- /dev/null +++ b/lib/components/playbutton_view/playbutton_card.dart @@ -0,0 +1,170 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/image/universal_image.dart'; +import 'package:spotube/extensions/string.dart'; +import 'package:spotube/utils/platform.dart'; + +class PlaybuttonCard extends StatelessWidget { + final void Function()? onTap; + final void Function()? onPlaybuttonPressed; + final void Function()? onAddToQueuePressed; + final String? description; + + final String? imageUrl; + final Widget? image; + final bool isPlaying; + final bool isLoading; + final String title; + final bool isOwner; + + const PlaybuttonCard({ + required this.isPlaying, + required this.isLoading, + required this.title, + this.description, + this.onPlaybuttonPressed, + this.onAddToQueuePressed, + this.onTap, + this.isOwner = false, + this.imageUrl, + this.image, + super.key, + }) : assert( + imageUrl != null || image != null, + "imageUrl and image can't be null at the same time", + ); + + @override + Widget build(BuildContext context) { + final unescapeHtml = description?.unescapeHtml().cleanHtml() ?? ""; + final scale = context.theme.scaling; + + return SizedBox( + width: 150 * scale, + child: CardImage( + image: Stack( + children: [ + if (imageUrl != null) + Container( + width: 150 * scale, + height: 150 * scale, + decoration: BoxDecoration( + borderRadius: context.theme.borderRadiusMd, + image: DecorationImage( + image: UniversalImage.imageProvider( + imageUrl!, + height: 200 * scale, + width: 200 * scale, + ), + fit: BoxFit.cover, + ), + ), + ) + else + SizedBox( + width: 150 * scale, + height: 150 * scale, + child: ClipRRect( + borderRadius: context.theme.borderRadiusMd, + child: image!, + ), + ), + StatedWidget.builder( + builder: (context, states) { + return Positioned( + right: 8, + bottom: 8, + child: Column( + children: [ + AnimatedScale( + curve: Curves.easeOutBack, + duration: const Duration(milliseconds: 300), + scale: (states.contains(WidgetState.hovered) || + kIsMobile) && + !isLoading + ? 1 + : 0.7, + child: AnimatedOpacity( + duration: const Duration(milliseconds: 300), + opacity: (states.contains(WidgetState.hovered) || + kIsMobile) && + !isLoading + ? 1 + : 0, + child: IconButton.secondary( + icon: const Icon(SpotubeIcons.queueAdd), + onPressed: onAddToQueuePressed, + size: ButtonSize.small, + ), + ), + ), + const Gap(5), + AnimatedScale( + curve: Curves.easeOutBack, + duration: const Duration(milliseconds: 150), + scale: states.contains(WidgetState.hovered) || + kIsMobile || + isPlaying || + isLoading + ? 1 + : 0.7, + child: AnimatedOpacity( + duration: const Duration(milliseconds: 150), + opacity: states.contains(WidgetState.hovered) || + kIsMobile || + isPlaying || + isLoading + ? 1 + : 0, + child: IconButton.secondary( + icon: switch ((isLoading, isPlaying)) { + (true, _) => const CircularProgressIndicator( + size: 15, + ), + (false, false) => const Icon(SpotubeIcons.play), + (false, true) => const Icon(SpotubeIcons.pause) + }, + enabled: !isLoading, + onPressed: onPlaybuttonPressed, + size: ButtonSize.small, + ), + ), + ), + ], + ), + ); + }, + ), + if (isOwner) + const Positioned( + right: 5, + top: 5, + child: SecondaryBadge( + style: ButtonStyle.secondaryIcon( + shape: ButtonShape.circle, + size: ButtonSize.small, + ), + child: Icon(SpotubeIcons.user), + ), + ), + ], + ), + title: Tooltip( + tooltip: TooltipContainer(child: Text(title)).call, + child: Text( + title, + maxLines: 1, + overflow: TextOverflow.ellipsis, + ), + ), + subtitle: Text( + unescapeHtml.isEmpty ? "\n" : unescapeHtml, + maxLines: 2, + overflow: TextOverflow.ellipsis, + ), + onPressed: onTap, + ), + ); + } +} diff --git a/lib/components/playbutton_view/playbutton_tile.dart b/lib/components/playbutton_view/playbutton_tile.dart new file mode 100644 index 00000000..7470105d --- /dev/null +++ b/lib/components/playbutton_view/playbutton_tile.dart @@ -0,0 +1,115 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/image/universal_image.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/extensions/string.dart'; + +class PlaybuttonTile extends StatelessWidget { + final void Function()? onTap; + final void Function()? onPlaybuttonPressed; + final void Function()? onAddToQueuePressed; + final String? description; + + final String? imageUrl; + final Widget? image; + final bool isPlaying; + final bool isLoading; + final String title; + final bool isOwner; + + const PlaybuttonTile({ + required this.isPlaying, + required this.isLoading, + required this.title, + this.description, + this.onPlaybuttonPressed, + this.onAddToQueuePressed, + this.onTap, + this.isOwner = false, + this.imageUrl, + this.image, + super.key, + }) : assert( + imageUrl != null || image != null, + "imageUrl and image can't be null at the same time", + ); + + @override + Widget build(BuildContext context) { + final cleanDescription = description?.unescapeHtml().cleanHtml() ?? ""; + final scale = context.theme.scaling; + + return Button( + leading: imageUrl != null + ? Container( + width: 50 * scale, + height: 50 * scale, + decoration: BoxDecoration( + borderRadius: context.theme.borderRadiusMd, + image: DecorationImage( + image: UniversalImage.imageProvider(imageUrl!), + fit: BoxFit.cover, + ), + ), + ) + : SizedBox( + width: 50 * scale, + height: 50 * scale, + child: ClipRRect( + borderRadius: context.theme.borderRadiusMd, + child: image, + ), + ), + style: ButtonVariance.ghost.copyWith( + padding: (context, states, value) { + return (ButtonVariance.ghost.padding(context, states) as EdgeInsets) + .copyWith(right: 0, left: 0); + }, + ), + trailing: Row( + mainAxisSize: MainAxisSize.min, + children: [ + Tooltip( + tooltip: TooltipContainer(child: Text(context.l10n.add_to_queue)).call, + child: IconButton.outline( + icon: const Icon(SpotubeIcons.queueAdd), + onPressed: onAddToQueuePressed, + enabled: !isLoading, + ), + ), + const Gap(8), + Tooltip( + tooltip: TooltipContainer(child: Text(context.l10n.play)).call, + child: IconButton.secondary( + icon: switch ((isLoading, isPlaying)) { + (true, _) => const CircularProgressIndicator( + size: 22, + ), + (false, false) => const Icon(SpotubeIcons.play), + (false, true) => const Icon(SpotubeIcons.pause) + }, + onPressed: onPlaybuttonPressed, + enabled: !isLoading, + ), + ), + ], + ), + enabled: !isLoading, + onPressed: onTap, + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(title), + if (cleanDescription.isNotEmpty) + Text( + description!, + maxLines: 2, + overflow: TextOverflow.ellipsis, + ).xSmall().muted(), + ], + ), + ); + } +} diff --git a/lib/components/playbutton_view/playbutton_view.dart b/lib/components/playbutton_view/playbutton_view.dart new file mode 100644 index 00000000..7880bb8c --- /dev/null +++ b/lib/components/playbutton_view/playbutton_view.dart @@ -0,0 +1,204 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/playbutton_view/playbutton_card.dart'; +import 'package:spotube/components/playbutton_view/playbutton_tile.dart'; +import 'package:spotube/components/waypoint.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:very_good_infinite_list/very_good_infinite_list.dart'; + +const _dummyPlaybuttonCard = PlaybuttonCard( + imageUrl: 'https://placehold.co/150x150.png', + isLoading: false, + isPlaying: false, + title: "Playbutton", + description: "A really cool playbutton", + isOwner: false, +); + +const _dummyPlaybuttonTile = PlaybuttonTile( + imageUrl: 'https://placehold.co/150x150.png', + isLoading: false, + isPlaying: false, + title: "Playbutton", + description: "A really cool playbutton", + isOwner: false, +); + +/// A [PlaybuttonCard] grid/list view (selectable) sliver widget +/// with support for infinite scrolling +class PlaybuttonView extends StatelessWidget { + final int itemCount; + final Widget Function(BuildContext context, int index) gridItemBuilder; + final Widget Function(BuildContext context, int index) listItemBuilder; + final bool hasMore; + final bool isLoading; + final VoidCallback onRequestMore; + final ScrollController controller; + + final Widget? leading; + + const PlaybuttonView({ + super.key, + required this.itemCount, + required this.gridItemBuilder, + required this.listItemBuilder, + required this.hasMore, + required this.isLoading, + required this.onRequestMore, + required this.controller, + this.leading, + }); + + @override + Widget build(BuildContext context) { + final scale = context.theme.scaling; + + return SliverLayoutBuilder( + builder: (context, constrains) => HookBuilder(builder: (context) { + final isGrid = useState(constrains.mdAndUp); + final hasUserInteracted = useRef(false); + + useEffect(() { + if (hasUserInteracted.value) return null; + if (isGrid.value != constrains.mdAndUp) { + isGrid.value = constrains.mdAndUp; + } + return null; + }, [constrains]); + + return SliverMainAxisGroup( + slivers: [ + SliverToBoxAdapter( + child: Row( + mainAxisAlignment: MainAxisAlignment.end, + children: [ + if (leading != null) leading!, + Toggle( + value: isGrid.value, + style: + const ButtonStyle.outline(density: ButtonDensity.icon), + onChanged: (value) { + isGrid.value = value; + hasUserInteracted.value = true; + }, + child: const Icon(SpotubeIcons.grid), + ), + const SizedBox(width: 8), + Toggle( + value: !isGrid.value, + style: + const ButtonStyle.outline(density: ButtonDensity.icon), + onChanged: (value) { + isGrid.value = !value; + hasUserInteracted.value = true; + }, + child: const Icon(SpotubeIcons.list), + ), + ], + ), + ), + const SliverGap(10), + // Toggle between grid and list view + switch ((isGrid.value, isLoading)) { + (true, _) => !isLoading && itemCount == 0 + ? SliverPadding( + padding: const EdgeInsets.symmetric(horizontal: 8), + sliver: SliverToBoxAdapter( + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.taken, + color: Theme.of(context).colorScheme.primary, + ), + Text( + context.l10n.nothing_found, + textAlign: TextAlign.center, + ).muted().small() + ], + ), + ), + ) + : SliverGrid.builder( + itemCount: isLoading ? 6 : itemCount + 1, + gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( + maxCrossAxisExtent: 150 * scale, + mainAxisExtent: 225 * scale, + crossAxisSpacing: 12 * scale, + mainAxisSpacing: 12 * scale, + ), + itemBuilder: (context, index) { + if (isLoading) { + return const Skeletonizer( + enabled: true, + child: _dummyPlaybuttonCard, + ); + } + + if (index == itemCount) { + if (!hasMore) return const SizedBox.shrink(); + return Waypoint( + controller: controller, + isGrid: true, + onTouchEdge: onRequestMore, + child: const Skeletonizer( + enabled: true, + child: _dummyPlaybuttonCard, + ), + ); + } + + return gridItemBuilder(context, index); + }, + ), + (false, true) => Skeletonizer.sliver( + enabled: true, + child: SliverList( + delegate: SliverChildBuilderDelegate( + (context, index) => _dummyPlaybuttonTile, + childCount: 6, + ), + ), + ), + (false, false) => SliverInfiniteList( + itemCount: itemCount, + loadingBuilder: (context) => const Skeletonizer( + enabled: true, + child: _dummyPlaybuttonTile, + ), + itemBuilder: listItemBuilder, + onFetchData: onRequestMore, + hasReachedMax: !hasMore, + isLoading: isLoading, + emptyBuilder: (context) { + return Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.taken, + color: Theme.of(context).colorScheme.primary, + ), + Text( + context.l10n.nothing_found, + textAlign: TextAlign.center, + ).muted().small() + ], + ); + }, + ), + } + ], + ); + }), + ); + } +} diff --git a/lib/components/shimmers/shimmer_lyrics.dart b/lib/components/shimmers/shimmer_lyrics.dart index 03816202..9312865e 100644 --- a/lib/components/shimmers/shimmer_lyrics.dart +++ b/lib/components/shimmers/shimmer_lyrics.dart @@ -1,6 +1,5 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:skeletonizer/skeletonizer.dart'; diff --git a/lib/components/sort_tracks_dropdown.dart b/lib/components/sort_tracks_dropdown.dart deleted file mode 100644 index 16727013..00000000 --- a/lib/components/sort_tracks_dropdown.dart +++ /dev/null @@ -1,88 +0,0 @@ -import 'package:flutter/material.dart'; - -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/library/user_local_tracks.dart'; -import 'package:spotube/components/adaptive/adaptive_pop_sheet_list.dart'; -import 'package:spotube/extensions/context.dart'; - -class SortTracksDropdown extends StatelessWidget { - final SortBy? value; - final void Function(SortBy)? onChanged; - const SortTracksDropdown({ - this.onChanged, - this.value, - super.key, - }); - - @override - Widget build(BuildContext context) { - var theme = Theme.of(context); - return ListTileTheme( - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(8), - ), - child: AdaptivePopSheetList( - children: [ - PopSheetEntry( - value: SortBy.none, - enabled: value != SortBy.none, - title: Text(context.l10n.none), - ), - PopSheetEntry( - value: SortBy.ascending, - enabled: value != SortBy.ascending, - title: Text(context.l10n.sort_a_z), - ), - PopSheetEntry( - value: SortBy.descending, - enabled: value != SortBy.descending, - title: Text(context.l10n.sort_z_a), - ), - PopSheetEntry( - value: SortBy.newest, - enabled: value != SortBy.newest, - title: Text(context.l10n.sort_newest), - ), - PopSheetEntry( - value: SortBy.oldest, - enabled: value != SortBy.oldest, - title: Text(context.l10n.sort_oldest), - ), - PopSheetEntry( - value: SortBy.duration, - enabled: value != SortBy.duration, - title: Text(context.l10n.sort_duration), - ), - PopSheetEntry( - value: SortBy.artist, - enabled: value != SortBy.artist, - title: Text(context.l10n.sort_artist), - ), - PopSheetEntry( - value: SortBy.album, - enabled: value != SortBy.album, - title: Text(context.l10n.sort_album), - ), - ], - headings: [ - Text(context.l10n.sort_tracks), - ], - onSelected: onChanged, - tooltip: context.l10n.sort_tracks, - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 8.0, vertical: 4.0), - child: DefaultTextStyle( - style: theme.textTheme.titleSmall!, - child: Row( - children: [ - const Icon(SpotubeIcons.sort), - const SizedBox(width: 8), - Text(context.l10n.sort_tracks), - ], - ), - ), - ), - ), - ); - } -} diff --git a/lib/components/spotube_page_route.dart b/lib/components/spotube_page_route.dart deleted file mode 100644 index 22e4d2f1..00000000 --- a/lib/components/spotube_page_route.dart +++ /dev/null @@ -1,25 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:go_router/go_router.dart'; - -class SpotubePage extends MaterialPage { - const SpotubePage({required super.child}); -} - -class SpotubeSlidePage extends CustomTransitionPage { - SpotubeSlidePage({ - required super.child, - super.key, - }) : super( - reverseTransitionDuration: const Duration(milliseconds: 150), - transitionDuration: const Duration(milliseconds: 150), - transitionsBuilder: (context, animation, secondaryAnimation, child) { - return SlideTransition( - position: Tween( - begin: const Offset(1, 0), - end: Offset.zero, - ).animate(animation), - child: child, - ); - }, - ); -} diff --git a/lib/components/themed_button_tab_bar.dart b/lib/components/themed_button_tab_bar.dart deleted file mode 100644 index c245e5f4..00000000 --- a/lib/components/themed_button_tab_bar.dart +++ /dev/null @@ -1,50 +0,0 @@ -import 'package:buttons_tabbar/buttons_tabbar.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; - -class ThemedButtonsTabBar extends HookWidget implements PreferredSizeWidget { - final List tabs; - final TabController? controller; - const ThemedButtonsTabBar({super.key, required this.tabs, this.controller}); - - @override - Widget build(BuildContext context) { - final theme = Theme.of(context); - final bgColor = useBrightnessValue( - theme.colorScheme.primaryContainer, - Color.lerp(theme.colorScheme.primary, Colors.black, 0.7)!, - ); - - return Padding( - padding: const EdgeInsets.only( - top: 8, - bottom: 8, - ), - child: ButtonsTabBar( - controller: controller, - radius: 100, - decoration: BoxDecoration( - color: bgColor, - borderRadius: BorderRadius.circular(15), - ), - labelStyle: theme.textTheme.labelLarge?.copyWith( - color: theme.colorScheme.primary, - fontWeight: FontWeight.bold, - ), - borderWidth: 0, - unselectedDecoration: BoxDecoration( - color: theme.colorScheme.surface, - borderRadius: BorderRadius.circular(15), - ), - unselectedLabelStyle: theme.textTheme.labelLarge?.copyWith( - color: theme.colorScheme.primary, - ), - tabs: tabs, - ), - ); - } - - @override - Size get preferredSize => const Size.fromHeight(50); -} diff --git a/lib/components/titlebar/mouse_state.dart b/lib/components/titlebar/mouse_state.dart deleted file mode 100644 index 9af2a8b0..00000000 --- a/lib/components/titlebar/mouse_state.dart +++ /dev/null @@ -1,73 +0,0 @@ -import 'package:flutter/material.dart'; - -typedef MouseStateBuilderCB = Widget Function( - BuildContext context, MouseState mouseState); - -class MouseState { - bool isMouseOver = false; - bool isMouseDown = false; - MouseState(); - @override - String toString() { - return "isMouseDown: $isMouseDown - isMouseOver: $isMouseOver"; - } -} - -T? _ambiguate(T? value) => value; - -class MouseStateBuilder extends StatefulWidget { - final MouseStateBuilderCB builder; - final VoidCallback? onPressed; - const MouseStateBuilder({super.key, required this.builder, this.onPressed}); - @override - // ignore: library_private_types_in_public_api - _MouseStateBuilderState createState() => _MouseStateBuilderState(); -} - -class _MouseStateBuilderState extends State { - late MouseState _mouseState; - _MouseStateBuilderState() { - _mouseState = MouseState(); - } - - @override - Widget build(BuildContext context) { - return MouseRegion( - onEnter: (event) { - setState(() { - _mouseState.isMouseOver = true; - }); - }, - onExit: (event) { - setState(() { - _mouseState.isMouseOver = false; - }); - }, - child: GestureDetector( - onTapDown: (_) { - setState(() { - _mouseState.isMouseDown = true; - }); - }, - onTapCancel: () { - setState(() { - _mouseState.isMouseDown = false; - }); - }, - onTap: () { - setState(() { - _mouseState.isMouseDown = false; - _mouseState.isMouseOver = false; - }); - _ambiguate(WidgetsBinding.instance)!.addPostFrameCallback((_) { - if (widget.onPressed != null) { - widget.onPressed!(); - } - }); - }, - onTapUp: (_) {}, - child: widget.builder(context, _mouseState), - ), - ); - } -} diff --git a/lib/components/titlebar/titlebar.dart b/lib/components/titlebar/titlebar.dart index 76a5ec8a..778f0b09 100644 --- a/lib/components/titlebar/titlebar.dart +++ b/lib/components/titlebar/titlebar.dart @@ -1,88 +1,60 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/components/button/back_button.dart'; import 'package:spotube/components/titlebar/titlebar_buttons.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/utils/platform.dart'; - import 'package:window_manager/window_manager.dart'; -class PageWindowTitleBar extends StatefulHookConsumerWidget - implements PreferredSizeWidget { - final Widget? leading; +final kTitlebarVisible = kIsWindows || kIsLinux; + +class TitleBar extends HookConsumerWidget implements PreferredSizeWidget { final bool automaticallyImplyLeading; - final List? actions; + final List trailing; + final List leading; + final Widget? child; + final Widget? title; + final Widget? header; // small widget placed on top of title + final Widget? subtitle; // small widget placed below title + final bool + trailingExpanded; // expand the trailing instead of the main content + final AlignmentGeometry alignment; final Color? backgroundColor; final Color? foregroundColor; - final IconThemeData? actionsIconTheme; - final bool? centerTitle; - final double? titleSpacing; - final double toolbarOpacity; - final double? leadingWidth; - final TextStyle? toolbarTextStyle; - final TextStyle? titleTextStyle; - final double? titleWidth; - final Widget? title; + final double? leadingGap; + final double? trailingGap; + final EdgeInsetsGeometry? padding; + final double? height; + final bool useSafeArea; + final double? surfaceBlur; + final double? surfaceOpacity; - final bool _sliver; - - const PageWindowTitleBar({ + const TitleBar({ super.key, - this.actions, + this.automaticallyImplyLeading = true, + this.trailing = const [], + this.leading = const [], this.title, - this.toolbarOpacity = 1, + this.header, + this.subtitle, + this.child, + this.trailingExpanded = false, + this.alignment = Alignment.center, + this.padding, this.backgroundColor, - this.actionsIconTheme, - this.automaticallyImplyLeading = false, - this.centerTitle, this.foregroundColor, - this.leading, - this.leadingWidth, - this.titleSpacing, - this.titleTextStyle, - this.titleWidth, - this.toolbarTextStyle, - }) : _sliver = false, - pinned = false, - floating = false, - snap = false, - stretch = false; + this.leadingGap, + this.trailingGap, + this.height, + this.surfaceBlur, + this.surfaceOpacity, + this.useSafeArea = false, + }); - final bool pinned; - final bool floating; - final bool snap; - final bool stretch; - - const PageWindowTitleBar.sliver({ - super.key, - this.actions, - this.title, - this.backgroundColor, - this.actionsIconTheme, - this.automaticallyImplyLeading = false, - this.centerTitle, - this.foregroundColor, - this.leading, - this.leadingWidth, - this.titleSpacing, - this.titleTextStyle, - this.titleWidth, - this.toolbarTextStyle, - this.pinned = false, - this.floating = false, - this.snap = false, - this.stretch = false, - }) : _sliver = true, - toolbarOpacity = 1; - - @override - Size get preferredSize => const Size.fromHeight(kToolbarHeight); - - @override - ConsumerState createState() => _PageWindowTitleBarState(); -} - -class _PageWindowTitleBarState extends ConsumerState { - void onDrag(details) { + void onDrag(WidgetRef ref) { final systemTitleBar = ref.read(userPreferencesProvider.select((s) => s.systemTitleBar)); if (kIsDesktop && !systemTitleBar) { @@ -91,89 +63,73 @@ class _PageWindowTitleBarState extends ConsumerState { } @override - Widget build(BuildContext context) { - final mediaQuery = MediaQuery.of(context); + Widget build(BuildContext context, ref) { + final hasLeadingOrCanPop = leading.isNotEmpty || Navigator.canPop(context); + final lastClicked = useRef(DateTime.now().millisecondsSinceEpoch); - if (widget._sliver) { - return SliverLayoutBuilder( + return SizedBox( + height: height ?? (48 * context.theme.scaling), + child: LayoutBuilder( builder: (context, constraints) { final hasFullscreen = - mediaQuery.size.width == constraints.crossAxisExtent; - final hasLeadingOrCanPop = - widget.leading != null || Navigator.canPop(context); + MediaQuery.sizeOf(context).width == constraints.maxWidth; - return SliverPadding( - padding: EdgeInsets.only( - left: kIsMacOS && hasFullscreen && hasLeadingOrCanPop ? 65 : 0, - ), - sliver: SliverAppBar( - leading: widget.leading, - automaticallyImplyLeading: widget.automaticallyImplyLeading, - actions: [ - ...?widget.actions, - WindowTitleBarButtons(foregroundColor: widget.foregroundColor), + final canPop = leading.isEmpty && + automaticallyImplyLeading && + (Navigator.canPop(context) || context.watchRouter.canPop()); + + return GestureDetector( + onHorizontalDragStart: (_) => onDrag(ref), + onVerticalDragStart: (_) => onDrag(ref), + onTapDown: (details) async { + final systemTitlebar = ref.read( + userPreferencesProvider.select((s) => s.systemTitleBar)); + if (!kIsDesktop || systemTitlebar) return; + + int currMills = DateTime.now().millisecondsSinceEpoch; + + if ((currMills - lastClicked.value) < 500) { + if (await windowManager.isMaximized()) { + await windowManager.unmaximize(); + } else { + await windowManager.maximize(); + } + } else { + lastClicked.value = currMills; + } + }, + child: AppBar( + leading: canPop ? [const BackButton()] : leading, + trailing: [ + ...trailing, + Align( + alignment: Alignment.topRight, + child: + WindowTitleBarButtons(foregroundColor: foregroundColor), + ), ], - backgroundColor: widget.backgroundColor, - foregroundColor: widget.foregroundColor, - actionsIconTheme: widget.actionsIconTheme, - centerTitle: widget.centerTitle, - titleSpacing: widget.titleSpacing, - leadingWidth: widget.leadingWidth, - toolbarTextStyle: widget.toolbarTextStyle, - titleTextStyle: widget.titleTextStyle, - title: SizedBox( - width: double.infinity, // workaround to force dragging - child: widget.title ?? const Text(""), - ), - pinned: widget.pinned, - floating: widget.floating, - snap: widget.snap, - stretch: widget.stretch, - ), + title: title, + header: header, + subtitle: subtitle, + trailingExpanded: trailingExpanded, + alignment: alignment, + padding: padding ?? EdgeInsets.zero, + backgroundColor: backgroundColor, + leadingGap: leadingGap, + trailingGap: trailingGap, + height: height ?? (48 * context.theme.scaling), + surfaceBlur: surfaceBlur, + surfaceOpacity: surfaceOpacity, + useSafeArea: useSafeArea, + child: child, + ).withPadding( + left: kIsMacOS && hasFullscreen && hasLeadingOrCanPop ? 65 : 0), ); }, - ); - } - - return LayoutBuilder(builder: (context, constrains) { - final hasFullscreen = mediaQuery.size.width == constrains.maxWidth; - final hasLeadingOrCanPop = - widget.leading != null || Navigator.canPop(context); - - return GestureDetector( - onHorizontalDragStart: onDrag, - onVerticalDragStart: onDrag, - child: Padding( - padding: EdgeInsets.only( - left: kIsMacOS && hasFullscreen && hasLeadingOrCanPop ? 65 : 0, - ), - child: AppBar( - leading: widget.leading, - automaticallyImplyLeading: widget.automaticallyImplyLeading, - actions: [ - ...?widget.actions, - WindowTitleBarButtons(foregroundColor: widget.foregroundColor), - ], - backgroundColor: widget.backgroundColor, - foregroundColor: widget.foregroundColor, - actionsIconTheme: widget.actionsIconTheme, - centerTitle: widget.centerTitle, - titleSpacing: widget.titleSpacing, - toolbarOpacity: widget.toolbarOpacity, - leadingWidth: widget.leadingWidth, - toolbarTextStyle: widget.toolbarTextStyle, - titleTextStyle: widget.titleTextStyle, - title: SizedBox( - width: double.infinity, // workaround to force dragging - child: widget.title ?? const Text(""), - ), - scrolledUnderElevation: 0, - shadowColor: Colors.transparent, - forceMaterialTransparency: true, - elevation: 0, - ), - ), - ); - }); + ), + ); } + + @override + Size get preferredSize => Size.fromHeight(height ?? 48); } diff --git a/lib/components/titlebar/titlebar_buttons.dart b/lib/components/titlebar/titlebar_buttons.dart index 35cdf08e..30d88508 100644 --- a/lib/components/titlebar/titlebar_buttons.dart +++ b/lib/components/titlebar/titlebar_buttons.dart @@ -1,8 +1,12 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/components/hover_builder.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/components/titlebar/titlebar_icon_buttons.dart'; -import 'package:spotube/components/titlebar/window_button.dart'; + +import 'package:spotube/hooks/configurators/use_window_listener.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/utils/platform.dart'; import 'package:titlebar_buttons/titlebar_buttons.dart'; @@ -25,6 +29,15 @@ class WindowTitleBarButtons extends HookConsumerWidget { await windowManager.close(); } + useWindowListener( + onWindowMaximize: () { + isMaximized.value = true; + }, + onWindowUnmaximize: () { + isMaximized.value = false; + }, + ); + useEffect(() { if (kIsDesktop) { windowManager.isMaximized().then((value) { @@ -34,91 +47,73 @@ class WindowTitleBarButtons extends HookConsumerWidget { return null; }, []); - if (!kIsDesktop || kIsMacOS || preferences.systemTitleBar) { + if (!kTitlebarVisible || preferences.systemTitleBar) { return const SizedBox.shrink(); } if (kIsWindows) { - final theme = Theme.of(context); - final colors = WindowButtonColors( - normal: Colors.transparent, - iconNormal: foregroundColor ?? theme.colorScheme.onSurface, - mouseOver: theme.colorScheme.onSurface.withOpacity(0.1), - mouseDown: theme.colorScheme.onSurface.withOpacity(0.2), - iconMouseOver: theme.colorScheme.onSurface, - iconMouseDown: theme.colorScheme.onSurface, - ); - - final closeColors = WindowButtonColors( - normal: Colors.transparent, - iconNormal: foregroundColor ?? theme.colorScheme.onSurface, - mouseOver: Colors.red, - mouseDown: Colors.red[800]!, - iconMouseOver: Colors.white, - iconMouseDown: Colors.black, - ); - - return Padding( - padding: const EdgeInsets.only(bottom: 25), - child: Row( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - MinimizeWindowButton( - onPressed: windowManager.minimize, - colors: colors, + return Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + ShadcnWindowButton( + icon: MinimizeIcon(color: context.theme.colorScheme.foreground), + onPressed: windowManager.minimize, + ), + if (isMaximized.value != true) + ShadcnWindowButton( + icon: MaximizeIcon(color: context.theme.colorScheme.foreground), + onPressed: () { + windowManager.maximize(); + isMaximized.value = true; + }, + ) + else + ShadcnWindowButton( + icon: RestoreIcon(color: context.theme.colorScheme.foreground), + onPressed: () { + windowManager.unmaximize(); + isMaximized.value = false; + }, ), - if (isMaximized.value != true) - MaximizeWindowButton( - colors: colors, - onPressed: () { - windowManager.maximize(); - isMaximized.value = true; - }, - ) - else - RestoreWindowButton( - colors: colors, - onPressed: () { - windowManager.unmaximize(); - isMaximized.value = false; - }, + HoverBuilder(builder: (context, isHovered) { + return ShadcnWindowButton( + icon: CloseIcon( + color: isHovered + ? Colors.white + : context.theme.colorScheme.foreground, ), - CloseWindowButton( - colors: closeColors, onPressed: onClose, - ), - ], - ), + hoverBackgroundColor: const Color(0xFFD32F2F), + ); + }), + ], ); } - return Padding( - padding: const EdgeInsets.only(bottom: 20, left: 10), - child: Row( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - DecoratedMinimizeButton( - type: type, - onPressed: windowManager.minimize, - ), - DecoratedMaximizeButton( - type: type, - onPressed: () async { - if (await windowManager.isMaximized()) { - await windowManager.unmaximize(); - isMaximized.value = false; - } else { - await windowManager.maximize(); - isMaximized.value = true; - } - }, - ), - DecoratedCloseButton( - type: type, - onPressed: onClose, - ), - ], - ), + return Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + DecoratedMinimizeButton( + type: type, + onPressed: windowManager.minimize, + ), + DecoratedMaximizeButton( + type: type, + onPressed: () async { + if (await windowManager.isMaximized()) { + await windowManager.unmaximize(); + isMaximized.value = false; + } else { + await windowManager.maximize(); + isMaximized.value = true; + } + }, + ), + DecoratedCloseButton( + type: type, + onPressed: onClose, + ), + ], ); } } diff --git a/lib/components/titlebar/titlebar_icon_buttons.dart b/lib/components/titlebar/titlebar_icon_buttons.dart index 70170262..0a3f6178 100644 --- a/lib/components/titlebar/titlebar_icon_buttons.dart +++ b/lib/components/titlebar/titlebar_icon_buttons.dart @@ -1,56 +1,49 @@ import 'dart:math'; -import 'package:flutter/material.dart'; -import 'package:spotube/components/titlebar/window_button.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; -class MinimizeWindowButton extends WindowButton { - MinimizeWindowButton( - {super.key, super.colors, super.onPressed, bool? animate}) - : super( - animate: animate ?? false, - iconBuilder: (buttonContext) => - MinimizeIcon(color: buttonContext.iconColor), - ); +class ShadcnWindowButton extends StatelessWidget { + final Widget icon; + final VoidCallback onPressed; + final Color? hoverBackgroundColor; + + const ShadcnWindowButton({ + super.key, + required this.icon, + required this.onPressed, + this.hoverBackgroundColor, + }); + + @override + Widget build(BuildContext context) { + return SizedBox( + width: 45, + height: 32, + child: IconButton( + variance: ButtonVariance.ghost.copyWith( + decoration: (context, states, value) { + final decoration = ButtonVariance.ghost.decoration(context, states) + as BoxDecoration; + if (hoverBackgroundColor != null && + states.contains(WidgetState.hovered)) { + return decoration.copyWith( + borderRadius: BorderRadius.zero, + color: hoverBackgroundColor, + ); + } + + return decoration.copyWith( + borderRadius: BorderRadius.zero, + ); + }, + ), + icon: icon, + onPressed: onPressed, + ), + ); + } } -class MaximizeWindowButton extends WindowButton { - MaximizeWindowButton( - {super.key, super.colors, super.onPressed, bool? animate}) - : super( - animate: animate ?? false, - iconBuilder: (buttonContext) => - MaximizeIcon(color: buttonContext.iconColor), - ); -} - -class RestoreWindowButton extends WindowButton { - RestoreWindowButton({super.key, super.colors, super.onPressed, bool? animate}) - : super( - animate: animate ?? false, - iconBuilder: (buttonContext) => - RestoreIcon(color: buttonContext.iconColor), - ); -} - -final _defaultCloseButtonColors = WindowButtonColors( - mouseOver: const Color(0xFFD32F2F), - mouseDown: const Color(0xFFB71C1C), - iconNormal: const Color(0xFF805306), - iconMouseOver: const Color(0xFFFFFFFF)); - -class CloseWindowButton extends WindowButton { - CloseWindowButton( - {super.key, WindowButtonColors? colors, super.onPressed, bool? animate}) - : super( - colors: colors ?? _defaultCloseButtonColors, - animate: animate ?? false, - iconBuilder: (buttonContext) => - CloseIcon(color: buttonContext.iconColor), - ); -} - -// Switched to CustomPaint icons by https://github.com/esDotDev - /// Close class CloseIcon extends StatelessWidget { final Color color; @@ -149,8 +142,9 @@ class _AlignedPaint extends StatelessWidget { @override Widget build(BuildContext context) { return Align( - alignment: Alignment.center, - child: CustomPaint(size: const Size(10, 10), painter: painter)); + alignment: Alignment.center, + child: CustomPaint(size: const Size(10, 10), painter: painter), + ); } } diff --git a/lib/components/titlebar/window_button.dart b/lib/components/titlebar/window_button.dart deleted file mode 100644 index 3201d191..00000000 --- a/lib/components/titlebar/window_button.dart +++ /dev/null @@ -1,133 +0,0 @@ -import 'dart:io'; - -import 'package:flutter/foundation.dart'; -import 'package:flutter/material.dart'; -import 'package:spotube/components/titlebar/mouse_state.dart'; - -typedef WindowButtonIconBuilder = Widget Function( - WindowButtonContext buttonContext); -typedef WindowButtonBuilder = Widget Function( - WindowButtonContext buttonContext, Widget icon); - -class WindowButtonContext { - BuildContext context; - MouseState mouseState; - Color? backgroundColor; - Color iconColor; - WindowButtonContext( - {required this.context, - required this.mouseState, - this.backgroundColor, - required this.iconColor}); -} - -class WindowButtonColors { - late Color normal; - late Color mouseOver; - late Color mouseDown; - late Color iconNormal; - late Color iconMouseOver; - late Color iconMouseDown; - WindowButtonColors( - {Color? normal, - Color? mouseOver, - Color? mouseDown, - Color? iconNormal, - Color? iconMouseOver, - Color? iconMouseDown}) { - this.normal = normal ?? _defaultButtonColors.normal; - this.mouseOver = mouseOver ?? _defaultButtonColors.mouseOver; - this.mouseDown = mouseDown ?? _defaultButtonColors.mouseDown; - this.iconNormal = iconNormal ?? _defaultButtonColors.iconNormal; - this.iconMouseOver = iconMouseOver ?? _defaultButtonColors.iconMouseOver; - this.iconMouseDown = iconMouseDown ?? _defaultButtonColors.iconMouseDown; - } -} - -final _defaultButtonColors = WindowButtonColors( - normal: Colors.transparent, - iconNormal: const Color(0xFF805306), - mouseOver: const Color(0xFF404040), - mouseDown: const Color(0xFF202020), - iconMouseOver: const Color(0xFFFFFFFF), - iconMouseDown: const Color(0xFFF0F0F0), -); - -class WindowButton extends StatelessWidget { - final WindowButtonBuilder? builder; - final WindowButtonIconBuilder? iconBuilder; - late final WindowButtonColors colors; - final bool animate; - final EdgeInsets? padding; - final VoidCallback? onPressed; - - WindowButton( - {super.key, - WindowButtonColors? colors, - this.builder, - @required this.iconBuilder, - this.padding, - this.onPressed, - this.animate = false}) { - this.colors = colors ?? _defaultButtonColors; - } - - Color getBackgroundColor(MouseState mouseState) { - if (mouseState.isMouseDown) return colors.mouseDown; - if (mouseState.isMouseOver) return colors.mouseOver; - return colors.normal; - } - - Color getIconColor(MouseState mouseState) { - if (mouseState.isMouseDown) return colors.iconMouseDown; - if (mouseState.isMouseOver) return colors.iconMouseOver; - return colors.iconNormal; - } - - @override - Widget build(BuildContext context) { - if (kIsWeb) { - return Container(); - } else { - // Don't show button on macOS - if (Platform.isMacOS) { - return Container(); - } - } - - return MouseStateBuilder( - builder: (context, mouseState) { - WindowButtonContext buttonContext = WindowButtonContext( - mouseState: mouseState, - context: context, - backgroundColor: getBackgroundColor(mouseState), - iconColor: getIconColor(mouseState)); - - var icon = - (iconBuilder != null) ? iconBuilder!(buttonContext) : Container(); - - var fadeOutColor = - getBackgroundColor(MouseState()..isMouseOver = true).withOpacity(0); - var padding = this.padding ?? const EdgeInsets.all(10); - var animationMs = - mouseState.isMouseOver ? (animate ? 100 : 0) : (animate ? 200 : 0); - Widget iconWithPadding = Padding(padding: padding, child: icon); - iconWithPadding = AnimatedContainer( - curve: Curves.easeOut, - duration: Duration(milliseconds: animationMs), - color: buttonContext.backgroundColor ?? fadeOutColor, - child: iconWithPadding); - var button = - (builder != null) ? builder!(buttonContext, icon) : iconWithPadding; - return SizedBox( - width: 45, - height: 32, - child: button, - ); - }, - onPressed: () { - if (onPressed != null) onPressed!(); - }, - ); - } -} diff --git a/lib/components/track_presentation/presentation_actions.dart b/lib/components/track_presentation/presentation_actions.dart new file mode 100644 index 00000000..61202a48 --- /dev/null +++ b/lib/components/track_presentation/presentation_actions.dart @@ -0,0 +1,234 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/adaptive/adaptive_pop_sheet_list.dart'; +import 'package:spotube/components/dialogs/confirm_download_dialog.dart'; +import 'package:spotube/components/dialogs/playlist_add_track_dialog.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/presentation_state.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/download_manager_provider.dart'; +import 'package:spotube/provider/history/history.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; + +ToastOverlay showToastForAction( + BuildContext context, + String action, + int count, +) { + final message = switch (action) { + "download" => (context.l10n.download_count(count), SpotubeIcons.download), + "add-to-playlist" => ( + context.l10n.add_count_to_playlist(count), + SpotubeIcons.playlistAdd + ), + "add-to-queue" => ( + context.l10n.add_count_to_queue(count), + SpotubeIcons.queueAdd + ), + "play-next" => ( + context.l10n.play_count_next(count), + SpotubeIcons.lightning + ), + _ => ("", SpotubeIcons.error), + }; + + return showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + leading: Icon(message.$2), + title: Text(message.$1), + leadingAlignment: Alignment.center, + trailing: IconButton.ghost( + size: ButtonSize.small, + icon: const Icon(SpotubeIcons.close), + onPressed: () { + overlay.close(); + }, + ), + ), + ); + }, + ); +} + +class TrackPresentationActionsSection extends HookConsumerWidget { + const TrackPresentationActionsSection({super.key}); + + @override + Widget build(BuildContext context, ref) { + final options = TrackPresentationOptions.of(context); + + ref.watch(downloadManagerProvider); + final downloader = ref.watch(downloadManagerProvider.notifier); + final playlistNotifier = ref.watch(audioPlayerProvider.notifier); + final historyNotifier = ref.watch(playbackHistoryActionsProvider); + + final state = ref.watch(presentationStateProvider(options.collection)); + final notifier = + ref.watch(presentationStateProvider(options.collection).notifier); + final selectedTracks = state.selectedTracks; + + Future actionDownloadTracks({ + required BuildContext context, + required List tracks, + required String action, + }) async { + final fullTrackObjects = + tracks.whereType().toList(); + final confirmed = await showDialog( + context: context, + builder: (context) { + return const ConfirmDownloadDialog(); + }, + ) ?? + false; + if (confirmed != true) return; + downloader.addAllToQueue(fullTrackObjects); + notifier.deselectAllTracks(); + if (!context.mounted) return; + showToastForAction(context, action, fullTrackObjects.length); + } + + return AdaptivePopSheetList( + tooltip: context.l10n.more_actions, + headings: [ + Text( + context.l10n.more_actions, + style: context.theme.typography.large, + ), + ], + onSelected: (action) async { + var tracks = selectedTracks; + + if (selectedTracks.isEmpty) { + tracks = await options.pagination.onFetchAll(); + + notifier.selectAllTracks(); + } + + if (!context.mounted) return; + + switch (action) { + case "download": + await actionDownloadTracks( + context: context, + tracks: tracks, + action: action, + ); + break; + case "add-to-playlist": + { + if (context.mounted) { + final worked = await showDialog( + context: context, + builder: (context) { + return PlaylistAddTrackDialog( + openFromPlaylist: options.collectionId, + tracks: tracks.toList(), + ); + }, + ); + + if (!context.mounted || worked != true) return; + showToastForAction(context, action, tracks.length); + } + break; + } + case "play-next": + { + playlistNotifier.addTracksAtFirst(tracks); + playlistNotifier.addCollection(options.collectionId); + if (options.collection is SpotubeSimpleAlbumObject) { + historyNotifier.addAlbums( + [options.collection as SpotubeSimpleAlbumObject]); + } else { + historyNotifier.addPlaylists( + [options.collection as SpotubeSimplePlaylistObject]); + } + notifier.deselectAllTracks(); + if (!context.mounted) return; + showToastForAction(context, action, tracks.length); + break; + } + case "add-to-queue": + { + playlistNotifier.addTracks(tracks); + playlistNotifier.addCollection(options.collectionId); + if (options.collection is SpotubeSimpleAlbumObject) { + historyNotifier.addAlbums( + [options.collection as SpotubeSimpleAlbumObject]); + } else { + historyNotifier.addPlaylists( + [options.collection as SpotubeSimplePlaylistObject]); + } + notifier.deselectAllTracks(); + if (!context.mounted) return; + showToastForAction(context, action, tracks.length); + break; + } + default: + } + + if (!context.mounted) return; + }, + icon: const Icon(SpotubeIcons.moreVertical), + variance: ButtonVariance.outline, + items: (context) => [ + AdaptiveMenuButton( + value: "download", + leading: const Icon(SpotubeIcons.download), + child: selectedTracks.isEmpty || + selectedTracks.length == options.tracks.length + ? Text( + context.l10n.download_all, + ) + : Text( + context.l10n.download_count(selectedTracks.length), + ), + ), + AdaptiveMenuButton( + value: "add-to-playlist", + leading: const Icon(SpotubeIcons.playlistAdd), + child: selectedTracks.isEmpty || + selectedTracks.length == options.tracks.length + ? Text( + context.l10n.add_all_to_playlist, + ) + : Text( + context.l10n.add_count_to_playlist(selectedTracks.length), + ), + ), + AdaptiveMenuButton( + value: "add-to-queue", + leading: const Icon(SpotubeIcons.queueAdd), + child: selectedTracks.isEmpty || + selectedTracks.length == options.tracks.length + ? Text( + context.l10n.add_all_to_queue, + ) + : Text( + context.l10n.add_count_to_queue(selectedTracks.length), + ), + ), + AdaptiveMenuButton( + value: "play-next", + leading: const Icon(SpotubeIcons.lightning), + child: selectedTracks.isEmpty || + selectedTracks.length == options.tracks.length + ? Text( + context.l10n.play_all_next, + ) + : Text( + context.l10n.play_count_next(selectedTracks.length), + ), + ), + ], + ); + } +} diff --git a/lib/components/track_presentation/presentation_list.dart b/lib/components/track_presentation/presentation_list.dart new file mode 100644 index 00000000..19772c7c --- /dev/null +++ b/lib/components/track_presentation/presentation_list.dart @@ -0,0 +1,129 @@ +import 'package:flutter/services.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/fake.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/presentation_state.dart'; +import 'package:spotube/components/track_presentation/use_track_tile_play_callback.dart'; +import 'package:spotube/components/track_tile/track_tile.dart'; +import 'package:spotube/components/track_presentation/use_is_user_playlist.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:very_good_infinite_list/very_good_infinite_list.dart'; + +class PresentationListSection extends HookConsumerWidget { + const PresentationListSection({super.key}); + + @override + Widget build(BuildContext context, ref) { + final options = TrackPresentationOptions.of(context); + final playlist = ref.watch(audioPlayerProvider); + final state = ref.watch(presentationStateProvider(options.collection)); + final notifier = + ref.read(presentationStateProvider(options.collection).notifier); + final isUserPlaylist = useIsUserPlaylist(ref, options.collectionId); + + final onTileTap = useTrackTilePlayCallback(ref); + + if (state.presentationTracks.isEmpty && !options.pagination.isLoading) { + if (options.error != null) { + return SliverToBoxAdapter( + child: Center( + child: Padding( + padding: const EdgeInsets.all(8.0), + child: ErrorBox( + error: options.error!, + onRetry: options.pagination.onRefresh, + ), + ), + ), + ); + } + return SliverToBoxAdapter( + child: Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Undraw( + illustration: UndrawIllustration.dreamer, + color: context.theme.colorScheme.primary, + height: 200 * context.theme.scaling, + ), + Text( + isUserPlaylist + ? context.l10n.no_tracks_added_yet + : context.l10n.no_tracks, + textAlign: TextAlign.center, + ).muted().small(), + ], + ), + ), + ); + } + + return SliverInfiniteList( + isLoading: options.pagination.isLoading, + onFetchData: options.pagination.onFetchMore, + itemCount: state.presentationTracks.length, + hasReachedMax: !options.pagination.hasNextPage, + loadingBuilder: (context) { + return Skeletonizer( + enabled: true, + child: TrackTile( + index: 0, + playlist: playlist, + track: FakeData.track, + ), + ); + }, + emptyBuilder: (context) => Skeletonizer( + enabled: true, + child: Column( + children: List.generate( + 10, + (index) => TrackTile( + track: FakeData.track, + index: index, + playlist: playlist, + ), + ), + ), + ), + itemBuilder: (context, index) => HookBuilder(builder: (context) { + final track = state.presentationTracks[index]; + final isSelected = useMemoized( + () => state.selectedTracks.any((e) => e.id == track.id), + [track.id, state.selectedTracks], + ); + return TrackTile( + userPlaylist: isUserPlaylist, + playlistId: options.collectionId, + index: index, + playlist: playlist, + track: track, + selected: isSelected, + onTap: () => onTileTap(track, index), + onChanged: state.selectedTracks.isEmpty + ? null + : (isSelected) { + if (isSelected == true) { + notifier.selectTrack(track); + } else { + notifier.deselectTrack(track); + } + }, + onLongPress: () { + notifier.selectTrack(track); + HapticFeedback.selectionClick(); + }, + ); + }), + ); + } +} diff --git a/lib/components/track_presentation/presentation_modifiers.dart b/lib/components/track_presentation/presentation_modifiers.dart new file mode 100644 index 00000000..42c3cb4f --- /dev/null +++ b/lib/components/track_presentation/presentation_modifiers.dart @@ -0,0 +1,131 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/track_presentation/sort_tracks_dropdown.dart'; +import 'package:spotube/components/track_presentation/presentation_actions.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/presentation_state.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart'; + +class TrackPresentationModifiersSection extends HookConsumerWidget { + final FocusNode? focusNode; + const TrackPresentationModifiersSection({ + super.key, + this.focusNode, + }); + + @override + Widget build(BuildContext context, ref) { + final options = TrackPresentationOptions.of(context); + final state = ref.watch(presentationStateProvider(options.collection)); + final notifier = ref.watch( + presentationStateProvider(options.collection).notifier, + ); + + final controller = useShadcnTextEditingController(); + final scale = context.theme.scaling; + + return LayoutBuilder(builder: (context, constrains) { + return Padding( + padding: EdgeInsets.symmetric( + horizontal: (constrains.mdAndUp ? 16 : 8) * scale, + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Row( + mainAxisSize: MainAxisSize.min, + children: [ + Checkbox( + state: state.selectedTracks.length == options.tracks.length + ? CheckboxState.checked + : CheckboxState.unchecked, + onChanged: (value) { + if (value == CheckboxState.checked) { + notifier.selectAllTracks(); + } else { + notifier.deselectAllTracks(); + } + }, + ), + ], + ), + Flexible( + child: Row( + mainAxisSize: MainAxisSize.min, + spacing: 8, + children: [ + Flexible( + child: ConstrainedBox( + constraints: BoxConstraints( + maxWidth: 320 * scale, + maxHeight: 38 * scale, + ), + child: TextField( + controller: controller, + focusNode: focusNode, + placeholder: Text(context.l10n.search_tracks), + onChanged: (value) { + if (value.isEmpty) { + notifier.clearFilter(); + } else { + notifier.filterTracks(value); + } + }, + features: [ + InputFeature.leading( + Icon( + SpotubeIcons.search, + color: context.theme.colorScheme.mutedForeground, + ), + ), + InputFeature.trailing( + ListenableBuilder( + listenable: controller, + builder: (context, _) { + return AnimatedCrossFade( + duration: const Duration(milliseconds: 300), + crossFadeState: controller.text.isEmpty + ? CrossFadeState.showFirst + : CrossFadeState.showSecond, + firstChild: + const SizedBox.square(dimension: 20), + secondChild: AnimatedScale( + duration: + const Duration(milliseconds: 300), + scale: controller.text.isEmpty ? 0 : 1, + child: IconButton.ghost( + size: const ButtonSize(.6), + icon: const Icon(SpotubeIcons.close), + onPressed: () { + controller.clear(); + notifier.clearFilter(); + }, + ), + ), + ); + }), + ) + ], + ), + ), + ), + SortTracksDropdown( + value: state.sortBy, + onChanged: (value) { + notifier.sortTracks(value); + }, + ), + const TrackPresentationActionsSection(), + ], + ), + ), + ], + ), + ); + }); + } +} diff --git a/lib/components/track_presentation/presentation_props.dart b/lib/components/track_presentation/presentation_props.dart new file mode 100644 index 00000000..1992487f --- /dev/null +++ b/lib/components/track_presentation/presentation_props.dart @@ -0,0 +1,113 @@ +import 'dart:async'; + +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class PaginationProps { + final bool hasNextPage; + final bool isLoading; + final VoidCallback onFetchMore; + final Future Function() onRefresh; + final Future> Function() onFetchAll; + + const PaginationProps({ + required this.hasNextPage, + required this.isLoading, + required this.onFetchMore, + required this.onFetchAll, + required this.onRefresh, + }); + + @override + operator ==(Object other) { + return other is PaginationProps && + other.hasNextPage == hasNextPage && + other.isLoading == isLoading && + other.onFetchMore == onFetchMore && + other.onFetchAll == onFetchAll && + other.onRefresh == onRefresh; + } + + @override + int get hashCode => + super.hashCode ^ + hasNextPage.hashCode ^ + isLoading.hashCode ^ + onFetchMore.hashCode ^ + onFetchAll.hashCode ^ + onRefresh.hashCode; +} + +class TrackPresentationOptions { + final Object collection; + final String title; + final String? description; + final String? owner; + final String? ownerImage; + final String image; + final String routePath; + final List tracks; + final PaginationProps pagination; + final bool isLiked; + final String? shareUrl; + final Object? error; + + // events + final FutureOr Function()? onHeart; // if null heart button will hidden + + const TrackPresentationOptions({ + required this.collection, + required this.title, + this.description, + this.owner, + this.ownerImage, + required this.image, + required this.tracks, + required this.pagination, + required this.routePath, + this.shareUrl, + this.isLiked = false, + this.onHeart, + this.error, + }) : assert(collection is SpotubeSimpleAlbumObject || + collection is SpotubeSimplePlaylistObject); + + String get collectionId => collection is SpotubeSimpleAlbumObject + ? (collection as SpotubeSimpleAlbumObject).id + : (collection as SpotubeSimplePlaylistObject).id; + + static TrackPresentationOptions of(BuildContext context) { + return Data.of(context); + } + + @override + operator ==(Object other) { + return other is TrackPresentationOptions && + other.collection == collection && + other.title == title && + other.description == description && + other.image == image && + other.routePath == routePath && + other.tracks == tracks && + other.pagination == pagination && + other.isLiked == isLiked && + other.shareUrl == shareUrl && + other.onHeart == onHeart && + other.error == error; + } + + @override + int get hashCode => + super.hashCode ^ + collection.hashCode ^ + title.hashCode ^ + description.hashCode ^ + image.hashCode ^ + routePath.hashCode ^ + tracks.hashCode ^ + pagination.hashCode ^ + isLiked.hashCode ^ + shareUrl.hashCode ^ + onHeart.hashCode ^ + error.hashCode; +} diff --git a/lib/components/track_presentation/presentation_state.dart b/lib/components/track_presentation/presentation_state.dart new file mode 100644 index 00000000..32b7353a --- /dev/null +++ b/lib/components/track_presentation/presentation_state.dart @@ -0,0 +1,180 @@ +import 'package:collection/collection.dart'; +import 'package:fuzzywuzzy/fuzzywuzzy.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/pages/library/user_local_tracks/user_local_tracks.dart'; +import 'package:spotube/provider/metadata_plugin/library/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/album.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/playlist.dart'; +import 'package:spotube/utils/service_utils.dart'; + +class PresentationState { + final List selectedTracks; + final List presentationTracks; + final SortBy sortBy; + + const PresentationState({ + required this.selectedTracks, + required this.presentationTracks, + required this.sortBy, + }); + + PresentationState copyWith({ + List? selectedTracks, + List? presentationTracks, + SortBy? sortBy, + }) { + return PresentationState( + selectedTracks: selectedTracks ?? this.selectedTracks, + presentationTracks: presentationTracks ?? this.presentationTracks, + sortBy: sortBy ?? this.sortBy, + ); + } +} + +class PresentationStateNotifier + extends AutoDisposeFamilyNotifier { + @override + PresentationState build(collection) { + if (arg case SpotubeSimplePlaylistObject() || SpotubeSimpleAlbumObject()) { + if (isSavedTrackPlaylist) { + ref.listen( + metadataPluginSavedTracksProvider, + (previous, next) { + next.whenData((value) { + state = state.copyWith( + presentationTracks: ServiceUtils.sortTracks( + value.items, + state.sortBy, + ), + ); + }); + }, + ); + } else { + ref.listen( + arg is SpotubeSimplePlaylistObject + ? metadataPluginPlaylistTracksProvider( + (arg as SpotubeSimplePlaylistObject).id) + : metadataPluginAlbumTracksProvider( + (arg as SpotubeSimpleAlbumObject).id), + (previous, next) { + next.whenData((value) { + state = state.copyWith( + presentationTracks: ServiceUtils.sortTracks( + value.items, + state.sortBy, + ), + ); + }); + }, + ); + } + } + + return PresentationState( + selectedTracks: [], + presentationTracks: tracks, + sortBy: SortBy.none, + ); + } + + bool get isSavedTrackPlaylist => + arg is SpotubeSimplePlaylistObject && + (arg as SpotubeSimplePlaylistObject).id == "user-liked-tracks"; + + List get tracks { + assert( + arg is SpotubeSimplePlaylistObject || arg is SpotubeSimpleAlbumObject, + "arg must be SpotubeSimplePlaylistObject or SpotubeSimpleAlbumObject", + ); + + final isPlaylist = arg is SpotubeSimplePlaylistObject; + + final tracks = switch ((isPlaylist, isSavedTrackPlaylist)) { + (true, true) => + ref.read(metadataPluginSavedTracksProvider).asData?.value.items, + (true, false) => ref + .read(metadataPluginPlaylistTracksProvider( + (arg as SpotubeSimplePlaylistObject).id)) + .asData + ?.value + .items, + _ => ref + .read(metadataPluginAlbumTracksProvider( + (arg as SpotubeSimpleAlbumObject).id)) + .asData + ?.value + .items, + } ?? + []; + + return tracks; + } + + void selectTrack(SpotubeTrackObject track) { + if (state.selectedTracks.any((e) => e.id == track.id)) { + return; + } + + state = state.copyWith( + selectedTracks: [...state.selectedTracks, track], + ); + } + + void selectAllTracks() { + state = state.copyWith( + selectedTracks: tracks, + ); + } + + void deselectTrack(SpotubeTrackObject track) { + state = state.copyWith( + selectedTracks: state.selectedTracks.where((e) => e != track).toList(), + ); + } + + void deselectAllTracks() { + state = state.copyWith( + selectedTracks: [], + ); + } + + void filterTracks(String query) { + if (query.isEmpty) { + return; + } + + state = state.copyWith( + presentationTracks: ServiceUtils.sortTracks( + tracks + .map((e) => (weightedRatio(e.name, query), e)) + .sorted((a, b) => b.$1.compareTo(a.$1)) + .where((e) => e.$1 > 50) + .map((e) => e.$2) + .toList(), + state.sortBy, + ), + ); + } + + void clearFilter() { + state = state.copyWith( + presentationTracks: ServiceUtils.sortTracks(tracks, state.sortBy), + ); + } + + void sortTracks(SortBy sortBy) { + state = state.copyWith( + presentationTracks: sortBy == SortBy.none + ? tracks + : ServiceUtils.sortTracks(state.presentationTracks, sortBy), + sortBy: sortBy, + ); + } +} + +final presentationStateProvider = AutoDisposeNotifierProviderFamily< + PresentationStateNotifier, PresentationState, Object>( + () => PresentationStateNotifier(), +); diff --git a/lib/components/track_presentation/presentation_top.dart b/lib/components/track_presentation/presentation_top.dart new file mode 100644 index 00000000..d2576cc0 --- /dev/null +++ b/lib/components/track_presentation/presentation_top.dart @@ -0,0 +1,261 @@ +import 'package:auto_size_text/auto_size_text.dart'; +import 'package:flutter/services.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/heart_button/heart_button.dart'; +import 'package:spotube/components/image/universal_image.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/use_action_callbacks.dart'; +import 'package:spotube/components/track_presentation/use_is_user_playlist.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/playlist/playlist_create_dialog.dart'; + +class TrackPresentationTopSection extends HookConsumerWidget { + const TrackPresentationTopSection({super.key}); + + @override + Widget build(BuildContext context, ref) { + final mediaQuery = MediaQuery.sizeOf(context); + final options = TrackPresentationOptions.of(context); + final scale = context.theme.scaling; + final isUserPlaylist = useIsUserPlaylist(ref, options.collectionId); + + final decorationImage = DecorationImage( + image: UniversalImage.imageProvider(options.image), + fit: BoxFit.cover, + ); + + final imageDimension = mediaQuery.mdAndUp ? 200 : 120; + + final (:isLoading, :isActive, :onPlay, :onShuffle, :onAddToQueue) = + useActionCallbacks(ref); + + final playbackActions = Row( + spacing: 8 * scale, + children: [ + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.shuffle_playlist), + ).call, + child: IconButton.secondary( + icon: isLoading + ? const Center( + child: + CircularProgressIndicator(onSurface: false, size: 20), + ) + : const Icon(SpotubeIcons.shuffle), + enabled: !isLoading && !isActive, + onPressed: onShuffle, + ), + ), + if (mediaQuery.width <= 320) + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.add_to_queue), + ).call, + child: IconButton.secondary( + icon: const Icon(SpotubeIcons.queueAdd), + enabled: !isLoading && !isActive, + onPressed: onAddToQueue, + ), + ) + else + Button.secondary( + leading: const Icon(SpotubeIcons.add), + enabled: !isLoading && !isActive, + onPressed: onAddToQueue, + child: Text(context.l10n.queue), + ), + Button.primary( + alignment: Alignment.center, + leading: switch ((isActive, isLoading)) { + (true, false) => const Icon(SpotubeIcons.pause), + (false, true) => const Center( + child: CircularProgressIndicator(onSurface: true, size: 18), + ), + _ => const Icon(SpotubeIcons.play), + }, + onPressed: onPlay, + enabled: !isLoading && !isActive, + child: isActive ? Text(context.l10n.pause) : Text(context.l10n.play), + ), + ], + ); + + final additionalActions = Row( + spacing: 8 * scale, + children: [ + if (isUserPlaylist) + IconButton.outline( + size: ButtonSize.small, + icon: const Icon(SpotubeIcons.edit), + onPressed: () { + showDialog( + context: context, + builder: (context) { + return PlaylistCreateDialog( + playlistId: options.collectionId, + trackIds: options.tracks.map((e) => e.id).toList(), + ); + }, + ); + }, + ), + if (options.shareUrl != null) + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.share), + ).call, + child: IconButton.outline( + icon: const Icon(SpotubeIcons.share), + size: ButtonSize.small, + onPressed: () async { + await Clipboard.setData( + ClipboardData(text: options.shareUrl!), + ); + + if (!context.mounted) return; + + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n + .copied_shareurl_to_clipboard(options.shareUrl!), + ).small(), + ); + }, + ); + }, + ), + ), + if (options.onHeart != null) + HeartButton( + isLiked: options.isLiked, + tooltip: options.isLiked + ? context.l10n.remove_from_favorites + : context.l10n.save_as_favorite, + variance: ButtonVariance.outline, + size: ButtonSize.small, + onPressed: options.onHeart, + ), + ], + ); + + return SliverMainAxisGroup( + slivers: [ + if (mediaQuery.mdAndUp) SliverGap(16 * scale), + SliverPadding( + padding: EdgeInsets.symmetric( + horizontal: (mediaQuery.mdAndUp ? 16 : 8.0) * scale, + ), + sliver: SliverList.list( + children: [ + DecoratedBox( + decoration: BoxDecoration( + image: decorationImage, + borderRadius: BorderRadius.circular(45), + ), + child: OutlinedContainer( + surfaceOpacity: context.theme.surfaceOpacity, + surfaceBlur: context.theme.surfaceBlur, + padding: EdgeInsets.all(24 * scale), + borderRadius: BorderRadius.circular(22 * scale), + borderWidth: 2, + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 16 * scale, + children: [ + Row( + spacing: 16 * scale, + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + Container( + height: imageDimension * scale, + width: imageDimension * scale, + decoration: BoxDecoration( + borderRadius: context.theme.borderRadiusXl, + image: decorationImage, + ), + ), + Flexible( + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + AutoSizeText( + options.title, + maxLines: 2, + minFontSize: 16, + style: context.theme.typography.h3, + ), + if (options.description != null) + AutoSizeText( + options.description!, + maxLines: 2, + minFontSize: 14, + maxFontSize: 18, + overflow: TextOverflow.ellipsis, + style: TextStyle( + color: context + .theme.colorScheme.mutedForeground, + fontSize: 18, + ), + ), + const Gap(16), + Flex( + crossAxisAlignment: CrossAxisAlignment.start, + direction: mediaQuery.smAndUp + ? Axis.horizontal + : Axis.vertical, + spacing: 8 * scale, + children: [ + if (options.owner != null) + OutlineBadge( + leading: options.ownerImage != null + ? Avatar( + initials: + options.owner?[0] ?? "U", + provider: UniversalImage + .imageProvider( + options.ownerImage!, + ), + size: 20 * scale, + ) + : null, + child: Text( + options.owner!, + maxLines: 1, + overflow: TextOverflow.ellipsis, + ).small(), + ), + additionalActions, + ], + ), + if (mediaQuery.mdAndUp) ...[ + const Gap(16), + playbackActions + ], + ], + ), + ), + ], + ), + if (mediaQuery.smAndDown) playbackActions, + ], + ), + ), + ), + ], + ), + ) + ], + ); + } +} diff --git a/lib/components/track_presentation/sort_tracks_dropdown.dart b/lib/components/track_presentation/sort_tracks_dropdown.dart new file mode 100644 index 00000000..0a07cbad --- /dev/null +++ b/lib/components/track_presentation/sort_tracks_dropdown.dart @@ -0,0 +1,70 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/pages/library/user_local_tracks/user_local_tracks.dart'; +import 'package:spotube/components/adaptive/adaptive_pop_sheet_list.dart'; +import 'package:spotube/extensions/context.dart'; + +class SortTracksDropdown extends StatelessWidget { + final SortBy? value; + final void Function(SortBy)? onChanged; + const SortTracksDropdown({ + this.onChanged, + this.value, + super.key, + }); + + @override + Widget build(BuildContext context) { + return AdaptivePopSheetList( + variance: ButtonVariance.outline, + headings: [ + Text(context.l10n.sort_tracks), + ], + onSelected: onChanged, + tooltip: context.l10n.sort_tracks, + icon: const Icon(SpotubeIcons.sort), + items: (context) => [ + AdaptiveMenuButton( + value: SortBy.none, + enabled: value != SortBy.none, + child: Text(context.l10n.none), + ), + AdaptiveMenuButton( + value: SortBy.ascending, + enabled: value != SortBy.ascending, + child: Text(context.l10n.sort_a_z), + ), + AdaptiveMenuButton( + value: SortBy.descending, + enabled: value != SortBy.descending, + child: Text(context.l10n.sort_z_a), + ), + AdaptiveMenuButton( + value: SortBy.newest, + enabled: value != SortBy.newest, + child: Text(context.l10n.sort_newest), + ), + AdaptiveMenuButton( + value: SortBy.oldest, + enabled: value != SortBy.oldest, + child: Text(context.l10n.sort_oldest), + ), + AdaptiveMenuButton( + value: SortBy.duration, + enabled: value != SortBy.duration, + child: Text(context.l10n.sort_duration), + ), + AdaptiveMenuButton( + value: SortBy.artist, + enabled: value != SortBy.artist, + child: Text(context.l10n.sort_artist), + ), + AdaptiveMenuButton( + value: SortBy.album, + enabled: value != SortBy.album, + child: Text(context.l10n.sort_album), + ), + ], + ); + } +} diff --git a/lib/components/track_presentation/track_presentation.dart b/lib/components/track_presentation/track_presentation.dart new file mode 100644 index 00000000..2b2a9f6f --- /dev/null +++ b/lib/components/track_presentation/track_presentation.dart @@ -0,0 +1,96 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/components/track_presentation/presentation_list.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/presentation_top.dart'; +import 'package:spotube/components/track_presentation/presentation_modifiers.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/utils/platform.dart'; + +class TrackPresentation extends HookConsumerWidget { + final TrackPresentationOptions options; + const TrackPresentation({ + super.key, + required this.options, + }); + + @override + Widget build(BuildContext context, ref) { + final scrollController = useScrollController(); + final focusNode = useFocusNode(); + final scale = context.theme.scaling; + + useEffect(() { + if (!kIsMobile) return null; + void listener() { + if (!scrollController.hasClients) return; + + if (focusNode.hasFocus) { + scrollController.animateTo( + 300 * scale, + duration: const Duration(milliseconds: 300), + curve: Curves.easeInOut, + ); + } + } + + focusNode.addListener(listener); + return () { + focusNode.removeListener(listener); + }; + }, [focusNode, scrollController, scale]); + + return Data.inherit( + data: options, + child: SafeArea( + bottom: false, + child: Scaffold( + headers: const [TitleBar()], + child: CustomScrollView( + controller: scrollController, + slivers: [ + const TrackPresentationTopSection(), + const SliverGap(16), + SliverList.list( + children: [ + TrackPresentationModifiersSection( + focusNode: focusNode, + ), + LayoutBuilder(builder: (context, constrains) { + return Basic( + padding: const EdgeInsets.symmetric( + vertical: 8, + horizontal: 16, + ), + leading: constrains.mdAndUp ? const Text(" #") : null, + title: Row( + children: [ + Expanded( + flex: constrains.lgAndUp ? 5 : 6, + child: Text(context.l10n.title), + ), + if (constrains.mdAndUp) + Expanded( + flex: 3, + child: Text(context.l10n.album), + ), + Text(context.l10n.duration), + ], + ), + ).small().muted(); + }), + ], + ), + const PresentationListSection(), + const SliverSafeArea(sliver: SliverGap(10)), + ], + ), + ), + ), + ); + } +} diff --git a/lib/components/track_presentation/use_action_callbacks.dart b/lib/components/track_presentation/use_action_callbacks.dart new file mode 100644 index 00000000..6707dd36 --- /dev/null +++ b/lib/components/track_presentation/use_action_callbacks.dart @@ -0,0 +1,173 @@ +import 'dart:math'; + +import 'package:flutter/widgets.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/components/dialogs/select_device_dialog.dart'; +import 'package:spotube/components/track_presentation/presentation_actions.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; + +import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/connect/connect.dart'; +import 'package:spotube/provider/history/history.dart'; +import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/services/logger/logger.dart'; + +typedef UseActionCallbacks = ({ + bool isActive, + bool isLoading, + Future Function() onShuffle, + Future Function() onPlay, + VoidCallback onAddToQueue, +}); + +UseActionCallbacks useActionCallbacks(WidgetRef ref) { + final isLoading = useState(false); + final context = useContext(); + final options = TrackPresentationOptions.of(context); + final playlist = ref.watch(audioPlayerProvider); + final playlistNotifier = ref.watch(audioPlayerProvider.notifier); + final historyNotifier = ref.watch(playbackHistoryActionsProvider); + + final isActive = useMemoized( + () => playlist.collections.contains(options.collectionId), + [playlist.collections, options.collectionId], + ); + + final onShuffle = useCallback(() async { + try { + isLoading.value = true; + + final initialTracks = options.tracks; + if (!context.mounted) return; + + final isRemoteDevice = await showSelectDeviceDialog(context, ref); + if (isRemoteDevice == null) return; + if (isRemoteDevice) { + final allTracks = await options.pagination.onFetchAll(); + final remotePlayback = ref.read(connectProvider.notifier); + await remotePlayback.load( + options.collection is SpotubeSimpleAlbumObject + ? WebSocketLoadEventData.album( + tracks: allTracks, + collection: options.collection as SpotubeSimpleAlbumObject, + initialIndex: Random().nextInt(allTracks.length)) + : WebSocketLoadEventData.playlist( + tracks: allTracks, + collection: options.collection as SpotubeSimplePlaylistObject, + initialIndex: Random().nextInt(allTracks.length), + ), + ); + await remotePlayback.setShuffle(true); + } else { + await playlistNotifier.load( + initialTracks, + autoPlay: true, + initialIndex: Random().nextInt(initialTracks.length), + ); + await audioPlayer.setShuffle(true); + playlistNotifier.addCollection(options.collectionId); + if (options.collection is SpotubeSimpleAlbumObject) { + historyNotifier + .addAlbums([options.collection as SpotubeSimpleAlbumObject]); + } else { + historyNotifier.addPlaylists( + [options.collection as SpotubeSimplePlaylistObject]); + } + + final allTracks = await options.pagination.onFetchAll(); + + await playlistNotifier.addTracks( + allTracks.sublist(initialTracks.length), + ); + } + } catch (e, stack) { + AppLogger.reportError(e, stack); + rethrow; + } finally { + isLoading.value = false; + } + }, [options, playlistNotifier, historyNotifier]); + + final onPlay = useCallback(() async { + try { + isLoading.value = true; + + final initialTracks = options.tracks; + + if (!context.mounted) return; + + final isRemoteDevice = await showSelectDeviceDialog(context, ref); + if (isRemoteDevice == null) return; + if (isRemoteDevice) { + final allTracks = await options.pagination.onFetchAll(); + + final remotePlayback = ref.read(connectProvider.notifier); + await remotePlayback.load( + options.collection is SpotubeSimpleAlbumObject + ? WebSocketLoadEventData.album( + tracks: allTracks, + collection: options.collection as SpotubeSimpleAlbumObject, + ) + : WebSocketLoadEventData.playlist( + tracks: allTracks, + collection: options.collection as SpotubeSimplePlaylistObject, + ), + ); + } else { + if (initialTracks.isEmpty) return; + + await playlistNotifier.load(initialTracks, autoPlay: true); + playlistNotifier.addCollection(options.collectionId); + + if (options.collection is SpotubeSimpleAlbumObject) { + historyNotifier.addAlbums( + [options.collection as SpotubeSimpleAlbumObject], + ); + } else { + historyNotifier.addPlaylists( + [options.collection as SpotubeSimplePlaylistObject], + ); + } + + final allTracks = await options.pagination.onFetchAll(); + + await playlistNotifier.addTracks( + allTracks.sublist(initialTracks.length), + ); + } + } catch (e, stack) { + AppLogger.reportError(e, stack); + rethrow; + } finally { + if (context.mounted) { + isLoading.value = false; + } + } + }, [options, playlistNotifier, historyNotifier]); + + final onAddToQueue = useCallback(() { + final tracks = options.tracks; + playlistNotifier.addTracks(tracks); + playlistNotifier.addCollection(options.collectionId); + if (options.collection is SpotubeSimpleAlbumObject) { + historyNotifier + .addAlbums([options.collection as SpotubeSimpleAlbumObject]); + } else { + historyNotifier + .addPlaylists([options.collection as SpotubeSimplePlaylistObject]); + } + if (!context.mounted) return; + showToastForAction(context, "add-to-queue", tracks.length); + }, [options, playlistNotifier, historyNotifier]); + + return ( + isActive: isActive, + isLoading: isLoading.value, + onShuffle: onShuffle, + onPlay: onPlay, + onAddToQueue: onAddToQueue, + ); +} diff --git a/lib/components/tracks_view/sections/body/use_is_user_playlist.dart b/lib/components/track_presentation/use_is_user_playlist.dart similarity index 56% rename from lib/components/tracks_view/sections/body/use_is_user_playlist.dart rename to lib/components/track_presentation/use_is_user_playlist.dart index 2f87ccc8..8792f6e7 100644 --- a/lib/components/tracks_view/sections/body/use_is_user_playlist.dart +++ b/lib/components/track_presentation/use_is_user_playlist.dart @@ -1,17 +1,18 @@ import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; bool useIsUserPlaylist(WidgetRef ref, String playlistId) { - final userPlaylistsQuery = ref.watch(favoritePlaylistsProvider); - final me = ref.watch(meProvider); + final userPlaylistsQuery = ref.watch(metadataPluginSavedPlaylistsProvider); + final me = ref.watch(metadataPluginUserProvider); return useMemoized( () => userPlaylistsQuery.asData?.value.items.any((e) => e.id == playlistId && me.asData?.value != null && - e.owner?.id == me.asData?.value.id) ?? + e.owner.id == me.asData?.value?.id) ?? false, [userPlaylistsQuery.asData?.value, playlistId, me.asData?.value], ); diff --git a/lib/components/track_presentation/use_track_tile_play_callback.dart b/lib/components/track_presentation/use_track_tile_play_callback.dart new file mode 100644 index 00000000..99f44f1e --- /dev/null +++ b/lib/components/track_presentation/use_track_tile_play_callback.dart @@ -0,0 +1,93 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; + +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:spotube/components/dialogs/select_device_dialog.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/presentation_state.dart'; +import 'package:spotube/extensions/list.dart'; + +import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/connect/connect.dart'; +import 'package:spotube/provider/history/history.dart'; + +Future Function(SpotubeTrackObject track, int index) + useTrackTilePlayCallback( + WidgetRef ref, +) { + final context = useContext(); + final options = TrackPresentationOptions.of(context); + final playlist = ref.watch(audioPlayerProvider); + final playlistNotifier = ref.watch(audioPlayerProvider.notifier); + final historyNotifier = ref.watch(playbackHistoryActionsProvider); + + final isActive = useMemoized( + () => playlist.collections.contains(options.collectionId), + [playlist.collections, options.collectionId], + ); + + final onTapTrackTile = + useCallback((SpotubeTrackObject track, int index) async { + final state = ref.read(presentationStateProvider(options.collection)); + final notifier = + ref.read(presentationStateProvider(options.collection).notifier); + + if (state.selectedTracks.isNotEmpty) { + if (state.selectedTracks.contains(track)) { + notifier.deselectTrack(track); + } else { + notifier.selectTrack(track); + } + return; + } + + final isRemoteDevice = await showSelectDeviceDialog(context, ref); + if (isRemoteDevice == null) return; + + if (isRemoteDevice) { + final remotePlayback = ref.read(connectProvider.notifier); + final remoteQueue = ref.read(queueProvider); + if (remoteQueue.collections.contains(options.collectionId) || + remoteQueue.tracks.any((s) => s.id == track.id)) { + await playlistNotifier.jumpToTrack(track); + } else { + final tracks = await options.pagination.onFetchAll(); + await remotePlayback.load( + options.collection is SpotubeSimpleAlbumObject + ? WebSocketLoadEventData.album( + tracks: tracks, + collection: options.collection as SpotubeSimpleAlbumObject, + initialIndex: index, + ) + : WebSocketLoadEventData.playlist( + tracks: tracks, + collection: options.collection as SpotubeSimplePlaylistObject, + initialIndex: index, + ), + ); + } + } else { + if (isActive || playlist.tracks.containsBy(track, (a) => a.id)) { + await playlistNotifier.jumpToTrack(track); + } else { + final tracks = await options.pagination.onFetchAll(); + await playlistNotifier.load( + tracks, + initialIndex: index, + autoPlay: true, + ); + playlistNotifier.addCollection(options.collectionId); + if (options.collection is SpotubeSimpleAlbumObject) { + historyNotifier + .addAlbums([options.collection as SpotubeSimpleAlbumObject]); + } else { + historyNotifier.addPlaylists( + [options.collection as SpotubeSimplePlaylistObject]); + } + } + } + }, [isActive, playlist, options, playlistNotifier, historyNotifier]); + + return onTapTrackTile; +} diff --git a/lib/components/track_tile/track_options.dart b/lib/components/track_tile/track_options.dart index fe3a1324..3e12d783 100644 --- a/lib/components/track_tile/track_options.dart +++ b/lib/components/track_tile/track_options.dart @@ -1,461 +1,283 @@ -import 'dart:io'; - -import 'package:flutter/material.dart' hide Page; -import 'package:flutter/services.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/assets.gen.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.dart'; import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/adaptive/adaptive_pop_sheet_list.dart'; -import 'package:spotube/components/dialogs/playlist_add_track_dialog.dart'; -import 'package:spotube/components/dialogs/prompt_dialog.dart'; -import 'package:spotube/components/dialogs/track_details_dialog.dart'; -import 'package:spotube/components/heart_button/use_track_toggle_like.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/links/artist_link.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/models/local_track.dart'; -import 'package:spotube/pages/track/track.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/blacklist_provider.dart'; -import 'package:spotube/provider/download_manager_provider.dart'; -import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; -import 'package:spotube/provider/spotify_provider.dart'; -import 'package:spotube/utils/service_utils.dart'; - -import 'package:url_launcher/url_launcher_string.dart'; - -enum TrackOptionValue { - album, - share, - songlink, - addToPlaylist, - addToQueue, - removeFromPlaylist, - removeFromQueue, - blacklist, - delete, - playNext, - favorite, - details, - download, - startRadio, -} +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/track_options/track_options_provider.dart'; +/// [track] must be a [SpotubeFullTrackObject] or [SpotubeLocalTrackObject] class TrackOptions extends HookConsumerWidget { - final Track track; + final SpotubeTrackObject track; final bool userPlaylist; final String? playlistId; - final ObjectRef?>? showMenuCbRef; final Widget? icon; + final VoidCallback? onTapItem; + const TrackOptions({ super.key, required this.track, - this.showMenuCbRef, this.userPlaylist = false, this.playlistId, this.icon, - }); - - void actionShare(BuildContext context, Track track) { - final data = "https://open.spotify.com/track/${track.id}"; - Clipboard.setData(ClipboardData(text: data)).then((_) { - ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - width: 300, - behavior: SnackBarBehavior.floating, - content: Text( - context.l10n.copied_to_clipboard(data), - textAlign: TextAlign.center, - ), - ), - ); - }); - } - - void actionAddToPlaylist( - BuildContext context, - Track track, - ) { - showDialog( - context: context, - builder: (context) => PlaylistAddTrackDialog( - tracks: [track], - openFromPlaylist: playlistId, - ), - ); - } - - void actionStartRadio( - BuildContext context, - WidgetRef ref, - Track track, - ) async { - final playback = ref.read(audioPlayerProvider.notifier); - final playlist = ref.read(audioPlayerProvider); - final spotify = ref.read(spotifyProvider); - final query = "${track.name} Radio"; - final pages = - await spotify.search.get(query, types: [SearchType.playlist]).first(); - - final radios = pages - .expand((e) => e.items?.cast().toList() ?? []) - .toList(); - - final artists = track.artists!.map((e) => e.name); - - final radio = radios.firstWhere( - (e) { - final validPlaylists = - artists.where((a) => e.description!.contains(a!)); - return e.name == "${track.name} Radio" && - (validPlaylists.length >= 2 || - validPlaylists.length == artists.length) && - e.owner?.displayName == "Spotify"; - }, - orElse: () => radios.first, - ); - - bool replaceQueue = false; - - if (context.mounted && playlist.tracks.isNotEmpty) { - replaceQueue = await showPromptDialog( - context: context, - title: context.l10n.how_to_start_radio, - message: context.l10n.replace_queue_question, - okText: context.l10n.replace, - cancelText: context.l10n.add_to_queue, - ); - } - - if (replaceQueue || playlist.tracks.isEmpty) { - await playback.stop(); - await playback.load([track], autoPlay: true); - - // we don't have to add those tracks as useEndlessPlayback will do it for us - return; - } else { - await playback.addTrack(track); - } - - final tracks = - await spotify.playlists.getTracksByPlaylistId(radio.id!).all(); - - await playback.addTracks( - tracks.toList() - ..removeWhere((e) { - final isDuplicate = playlist.tracks.any((t) => t.id == e.id); - return e.id == track.id || isDuplicate; - }), - ); - } + this.onTapItem, + }) : assert( + track is SpotubeFullTrackObject || track is SpotubeLocalTrackObject, + "Track must be a SpotubeFullTrackObject, SpotubeLocalTrackObject", + ); @override Widget build(BuildContext context, ref) { - final scaffoldMessenger = ScaffoldMessenger.of(context); final mediaQuery = MediaQuery.of(context); - final router = GoRouter.of(context); - final ThemeData(:colorScheme) = Theme.of(context); - final playlist = ref.watch(audioPlayerProvider); - final playback = ref.watch(audioPlayerProvider.notifier); - final auth = ref.watch(authenticationProvider); - ref.watch(downloadManagerProvider); - final downloadManager = ref.watch(downloadManagerProvider.notifier); - final blacklist = ref.watch(blacklistProvider); - final me = ref.watch(meProvider); + final trackOptionActions = ref.watch(trackOptionActionsProvider(track)); + final ( + :isBlacklisted, + :isInDownloadQueue, + :isInQueue, + :isActiveTrack, + :isAuthenticated, + :isLiked, + :downloadTask + ) = ref.watch(trackOptionsStateProvider(track)); + final isLocalTrack = track is SpotubeLocalTrackObject; - final favorites = useTrackToggleLike(track, ref); - - final isBlackListed = useMemoized( - () => blacklist.asData?.value.any( - (element) => element.elementId == track.id, - ), - [blacklist, track], - ); - - final removingTrack = useState(null); - final favoritePlaylistsNotifier = - ref.watch(favoritePlaylistsProvider.notifier); - - final isInQueue = useMemoized(() { - if (playlist.activeTrack == null) return false; - return downloadManager.isActive(playlist.activeTrack!); - }, [ - playlist.activeTrack, - downloadManager, - ]); - - final progressNotifier = useMemoized(() { - final spotubeTrack = downloadManager.mapToSourcedTrack(track); - if (spotubeTrack == null) return null; - return downloadManager.getProgressNotifier(spotubeTrack); - }); - - final isLocalTrack = track is LocalTrack; - - final adaptivePopSheetList = AdaptivePopSheetList( - onSelected: (value) async { - switch (value) { - case TrackOptionValue.album: - await router.push( - '/album/${track.album!.id}', - extra: track.album!, - ); - break; - case TrackOptionValue.delete: - await File((track as LocalTrack).path).delete(); - ref.invalidate(localTracksProvider); - break; - case TrackOptionValue.addToQueue: - await playback.addTrack(track); - if (context.mounted) { - scaffoldMessenger.showSnackBar( - SnackBar( - content: Text( - context.l10n.added_track_to_queue(track.name!), - ), - ), - ); - } - break; - case TrackOptionValue.playNext: - playback.addTracksAtFirst([track]); - scaffoldMessenger.showSnackBar( - SnackBar( - content: Text( - context.l10n.track_will_play_next(track.name!), - ), - ), - ); - break; - case TrackOptionValue.removeFromQueue: - playback.removeTrack(track.id!); - scaffoldMessenger.showSnackBar( - SnackBar( - content: Text( - context.l10n.removed_track_from_queue( - track.name!, - ), - ), - ), - ); - break; - case TrackOptionValue.favorite: - favorites.toggleTrackLike(track); - break; - case TrackOptionValue.addToPlaylist: - actionAddToPlaylist(context, track); - break; - case TrackOptionValue.removeFromPlaylist: - removingTrack.value = track.uri; - favoritePlaylistsNotifier - .removeTracks(playlistId ?? "", [track.id!]); - break; - case TrackOptionValue.blacklist: - if (isBlackListed == null) break; - if (isBlackListed == true) { - await ref.read(blacklistProvider.notifier).remove(track.id!); - } else { - await ref.read(blacklistProvider.notifier).add( - BlacklistTableCompanion.insert( - name: track.name!, - elementId: track.id!, - elementType: BlacklistedType.track, - ), - ); - } - break; - case TrackOptionValue.share: - actionShare(context, track); - break; - case TrackOptionValue.songlink: - final url = "https://song.link/s/${track.id}"; - await launchUrlString(url); - break; - case TrackOptionValue.details: - showDialog( - context: context, - builder: (context) => TrackDetailsDialog(track: track), - ); - break; - case TrackOptionValue.download: - await downloadManager.addToQueue(track); - break; - case TrackOptionValue.startRadio: - actionStartRadio(context, ref, track); - break; - } - }, - icon: icon ?? const Icon(SpotubeIcons.moreHorizontal), - headings: [ - ListTile( - dense: true, - leading: AspectRatio( - aspectRatio: 1, - child: ClipRRect( - borderRadius: BorderRadius.circular(10), - child: UniversalImage( - path: track.album!.images - .asUrlString(placeholder: ImagePlaceholder.albumArt), - fit: BoxFit.cover, - ), - ), - ), - title: Text( - track.name!, - maxLines: 1, - overflow: TextOverflow.ellipsis, - style: Theme.of(context).textTheme.titleMedium, - ), - subtitle: Align( - alignment: Alignment.centerLeft, - child: ArtistLink( - artists: track.artists!, - onOverflowArtistClick: () => ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": track.id!, - }, - ), - ), - ), - ), - ], + return Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 8, children: [ if (isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.delete, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.delete, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.trash), title: Text(context.l10n.delete), ), if (mediaQuery.smAndDown && !isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.album, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.album, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.album), - title: Text(context.l10n.go_to_album), - subtitle: Text(track.album!.name!), + title: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(context.l10n.go_to_album), + Text( + track.album.name, + style: context.theme.typography.xSmall, + ), + ], + ), ), - if (!playlist.containsTrack(track)) - PopSheetEntry( - value: TrackOptionValue.addToQueue, + if (!isInQueue) + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.addToQueue, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.queueAdd), title: Text(context.l10n.add_to_queue), ), else - PopSheetEntry( - value: TrackOptionValue.removeFromQueue, - enabled: playlist.activeTrack?.id != track.id, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.removeFromQueue, + playlistId, + ); + onTapItem?.call(); + }, + enabled: !isActiveTrack, leading: const Icon(SpotubeIcons.queueRemove), title: Text(context.l10n.remove_from_queue), ), - PopSheetEntry( - value: TrackOptionValue.playNext, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.playNext, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.lightning), title: Text(context.l10n.play_next), ), - if (me.asData?.value != null && !isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.favorite, - leading: favorites.isLiked + if (isAuthenticated && !isLocalTrack) + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.favorite, + playlistId, + ); + onTapItem?.call(); + }, + leading: isLiked ? const Icon( SpotubeIcons.heartFilled, color: Colors.pink, ) : const Icon(SpotubeIcons.heart), title: Text( - favorites.isLiked + isLiked ? context.l10n.remove_from_favorites : context.l10n.save_as_favorite, ), ), - if (auth.asData?.value != null && !isLocalTrack) ...[ - PopSheetEntry( - value: TrackOptionValue.startRadio, + if (isAuthenticated && !isLocalTrack) ...[ + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.startRadio, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.radio), title: Text(context.l10n.start_a_radio), ), - PopSheetEntry( - value: TrackOptionValue.addToPlaylist, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.addToPlaylist, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.playlistAdd), title: Text(context.l10n.add_to_playlist), ), ], - if (userPlaylist && auth.asData?.value != null && !isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.removeFromPlaylist, + if (userPlaylist && isAuthenticated && !isLocalTrack) + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.removeFromPlaylist, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.removeFilled), title: Text(context.l10n.remove_from_playlist), ), if (!isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.download, - enabled: !isInQueue, - leading: isInQueue - ? HookBuilder(builder: (context) { - final progress = useListenable(progressNotifier!); - return CircularProgressIndicator( - value: progress.value, - ); - }) + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.download, + playlistId, + ); + onTapItem?.call(); + }, + enabled: !isInDownloadQueue, + leading: isInDownloadQueue + ? StreamBuilder( + stream: downloadTask?.downloadedBytesStream, + builder: (context, snapshot) { + final progress = downloadTask?.totalSizeBytes == null || + downloadTask?.totalSizeBytes == 0 + ? 0 + : (snapshot.data ?? 0) / + downloadTask!.totalSizeBytes!; + return CircularProgressIndicator( + value: progress.toDouble(), + ); + }, + ) : const Icon(SpotubeIcons.download), title: Text(context.l10n.download_track), ), if (!isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.blacklist, - leading: const Icon(SpotubeIcons.playlistRemove), - iconColor: isBlackListed != true ? Colors.red[400] : null, - textColor: isBlackListed != true ? Colors.red[400] : null, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.blacklist, + playlistId, + ); + onTapItem?.call(); + }, + leading: Icon( + SpotubeIcons.playlistRemove, + color: isBlacklisted != true ? Colors.red[400] : null, + ), title: Text( - isBlackListed == true + isBlacklisted == true ? context.l10n.remove_from_blacklist : context.l10n.add_to_blacklist, + style: TextStyle( + color: isBlacklisted != true ? Colors.red[400] : null, + ), ), ), if (!isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.share, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.share, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.share), title: Text(context.l10n.share), ), if (!isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.songlink, - leading: Assets.logos.songlinkTransparent.image( - width: 22, - height: 22, - color: colorScheme.onSurface.withOpacity(0.5), - ), - title: Text(context.l10n.song_link), - ), - if (!isLocalTrack) - PopSheetEntry( - value: TrackOptionValue.details, + ButtonTile( + style: ButtonVariance.menu, + onPressed: () async { + await trackOptionActions.action( + rootNavigatorKey.currentContext!, + TrackOptionValue.details, + playlistId, + ); + onTapItem?.call(); + }, leading: const Icon(SpotubeIcons.info), title: Text(context.l10n.details), ), ], ); - - //! This is the most ANTI pattern I've ever done, but it works - showMenuCbRef?.value = (relativeRect) { - adaptivePopSheetList.showPopupMenu(context, relativeRect); - }; - - return ListTileTheme( - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(10), - ), - child: adaptivePopSheetList, - ); } } diff --git a/lib/components/track_tile/track_options_button.dart b/lib/components/track_tile/track_options_button.dart new file mode 100644 index 00000000..51fff5ea --- /dev/null +++ b/lib/components/track_tile/track_options_button.dart @@ -0,0 +1,152 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/image/universal_image.dart'; +import 'package:spotube/components/links/artist_link.dart'; +import 'package:spotube/components/track_tile/track_options.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class TrackOptionsButton extends HookConsumerWidget { + final SpotubeTrackObject track; + final bool userPlaylist; + final String? playlistId; + const TrackOptionsButton({ + super.key, + required this.track, + required this.userPlaylist, + this.playlistId, + }); + + static OverlayCompleter showOptions( + BuildContext context, + Offset offset, + SpotubeTrackObject track, { + bool userPlaylist = false, + String? playlistId, + }) { + return showPopover( + context: context, + position: offset, + alignment: Alignment.bottomRight, + builder: (context) { + return SizedBox( + width: 220 * context.theme.scaling, + child: Card( + padding: const EdgeInsets.all(8), + child: TrackOptions( + track: track, + playlistId: playlistId, + userPlaylist: userPlaylist, + onTapItem: () { + closeOverlay(context); + }, + ), + ), + ); + }, + ); + } + + @override + Widget build(BuildContext context, ref) { + final imageProvider = useMemoized( + () => UniversalImage.imageProvider( + (track.album.images).smallest(ImagePlaceholder.albumArt), + ), + [track.album.images], + ); + + return IconButton.ghost( + icon: const Icon(SpotubeIcons.moreHorizontal), + onPressed: () { + final mediaQuery = MediaQuery.sizeOf(context); + + if (mediaQuery.lgAndUp) { + final renderBox = context.findRenderObject() as RenderBox; + final position = RelativeRect.fromRect( + Rect.fromPoints( + renderBox.localToGlobal(Offset.zero, + ancestor: context.findRenderObject()), + renderBox.localToGlobal(renderBox.size.bottomRight(Offset.zero), + ancestor: context.findRenderObject()), + ), + Offset.zero & mediaQuery, + ); + final offset = Offset(position.left, position.top); + showOptions( + context, + offset, + track, + userPlaylist: userPlaylist, + playlistId: playlistId, + ); + } else { + openDrawer( + context: context, + position: OverlayPosition.bottom, + draggable: true, + showDragHandle: true, + borderRadius: context.theme.borderRadiusMd, + transformBackdrop: false, + builder: (context) { + return Padding( + padding: const EdgeInsets.symmetric( + horizontal: 16.0, + vertical: 8.0, + ), + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 8, + children: [ + Basic( + leading: Container( + width: 40, + height: 40, + decoration: BoxDecoration( + borderRadius: context.theme.borderRadiusMd, + image: DecorationImage( + fit: BoxFit.cover, + image: imageProvider, + ), + ), + ), + title: Text( + track.name, + maxLines: 1, + overflow: TextOverflow.ellipsis, + ).semiBold(), + subtitle: Align( + alignment: Alignment.centerLeft, + child: ArtistLink( + artists: track.artists, + onOverflowArtistClick: () => context.navigateTo( + TrackRoute(trackId: track.id), + ), + ), + ), + ), + const Divider(), + TrackOptions( + track: track, + userPlaylist: userPlaylist, + playlistId: playlistId, + onTapItem: () { + closeDrawer(context); + }, + ), + ], + ), + ); + }, + ); + } + }, + ); + } +} diff --git a/lib/components/track_tile/track_tile.dart b/lib/components/track_tile/track_tile.dart index 8ab889f8..ec3f50f3 100644 --- a/lib/components/track_tile/track_tile.dart +++ b/lib/components/track_tile/track_tile.dart @@ -1,35 +1,45 @@ import 'dart:async'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/hover_builder.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/links/artist_link.dart'; import 'package:spotube/components/links/link_text.dart'; -import 'package:spotube/components/track_tile/track_options.dart'; -import 'package:spotube/extensions/artist_simple.dart'; +import 'package:spotube/components/track_tile/track_options_button.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/duration.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/models/local_track.dart'; -import 'package:spotube/pages/track/track.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/querying_track_info.dart'; import 'package:spotube/provider/audio_player/state.dart'; import 'package:spotube/provider/blacklist_provider.dart'; import 'package:spotube/utils/platform.dart'; -import 'package:spotube/utils/service_utils.dart'; + +final isBlacklistedProvider = + Provider.autoDispose.family( + (ref, track) { + ref.watch(blacklistProvider); + final blacklist = ref.read(blacklistProvider.notifier); + return blacklist.contains(track); + }, +); + +final _overlay = ValueNotifier?>(null); class TrackTile extends HookConsumerWidget { /// [index] will not be shown if null final int? index; - final Track track; + final SpotubeTrackObject track; final bool selected; + final bool selectionMode; final ValueChanged? onChanged; final Future Function()? onTap; final VoidCallback? onLongPress; @@ -44,6 +54,7 @@ class TrackTile extends HookConsumerWidget { this.index, required this.track, this.selected = false, + this.selectionMode = false, required this.playlist, this.onTap, this.onLongPress, @@ -57,15 +68,7 @@ class TrackTile extends HookConsumerWidget { Widget build(BuildContext context, ref) { final theme = Theme.of(context); - final blacklist = ref.watch(blacklistProvider); - final blacklistNotifier = ref.watch(blacklistProvider.notifier); - - final isBlackListed = useMemoized( - () => blacklistNotifier.contains(track), - [blacklist, track], - ); - - final showOptionCbRef = useRef?>(null); + final isBlackListed = ref.watch(isBlacklistedProvider(track)); final isLoading = useState(false); @@ -73,24 +76,41 @@ class TrackTile extends HookConsumerWidget { final isSelected = isPlaying || isLoading.value; + final imageProvider = useMemoized( + () => UniversalImage.imageProvider( + (track.album.images).smallest(ImagePlaceholder.albumArt), + ), + [track.album.images], + ); + + // Treat either explicit selectionMode or presence of onChanged as selection + // context. Some lists enable selection by providing `onChanged` without + // toggling a dedicated `selectionMode` flag (e.g. playlists), so we must + // disable inner navigation in both cases. + final effectiveSelection = selectionMode || onChanged != null; + return LayoutBuilder(builder: (context, constrains) { return Listener( onPointerDown: (event) { if (event.buttons != kSecondaryMouseButton) return; - showOptionCbRef.value?.call( - RelativeRect.fromLTRB( - event.position.dx, - event.position.dy, - constrains.maxWidth - event.position.dx, - constrains.maxHeight - event.position.dy, - ), + if (_overlay.value != null) { + _overlay.value?.remove(); + _overlay.value = null; + } + _overlay.value = TrackOptionsButton.showOptions( + context, + Offset.zero, + track, + userPlaylist: userPlaylist, + playlistId: playlistId, ); }, child: HoverBuilder( permanentState: isSelected || constrains.smAndDown ? true : null, - builder: (context, isHovering) => ListTile( + builder: (context, isHovering) => ButtonTile( selected: isSelected, - onTap: () async { + onPressed: () async { + if (isBlackListed) return; try { isLoading.value = true; await onTap?.call(); @@ -101,46 +121,54 @@ class TrackTile extends HookConsumerWidget { } }, onLongPress: onLongPress, - enabled: !isBlackListed, - contentPadding: EdgeInsets.zero, - tileColor: isBlackListed ? theme.colorScheme.errorContainer : null, - horizontalTitleGap: 12, - leadingAndTrailingTextStyle: theme.textTheme.bodyMedium, + style: (isBlackListed + ? ButtonVariance.destructive + : ButtonVariance.ghost) + .copyWith( + padding: (context, states, value) => + const EdgeInsets.symmetric(vertical: 8, horizontal: 0), + ), leading: Row( mainAxisSize: MainAxisSize.min, children: [ ...?leadingActions, - if (index != null && onChanged == null && constrains.mdAndUp) - SizedBox( - width: 50, - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 6), - child: Text( - '${(index ?? 0) + 1}', - maxLines: 1, - style: theme.textTheme.bodySmall, - textAlign: TextAlign.center, - ), - ), - ) - else if (constrains.smAndDown) - const SizedBox(width: 16), - if (onChanged != null) - Checkbox( - value: selected, - onChanged: onChanged, + AnimatedCrossFade( + duration: const Duration(milliseconds: 300), + crossFadeState: index != null && onChanged == null + ? CrossFadeState.showSecond + : CrossFadeState.showFirst, + firstChild: Checkbox( + state: selected + ? CheckboxState.checked + : CheckboxState.unchecked, + onChanged: (state) => + onChanged?.call(state == CheckboxState.checked), ), + secondChild: constrains.smAndDown + ? const SizedBox(width: 16) + : SizedBox( + width: 50, + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 6), + child: Text( + '${(index ?? 0) + 1}', + maxLines: 1, + style: theme.typography.small, + textAlign: TextAlign.center, + ), + ), + ), + ), Stack( children: [ - ClipRRect( - borderRadius: BorderRadius.circular(4), - child: AspectRatio( - aspectRatio: 1, - child: UniversalImage( - path: (track.album?.images).asUrlString( - placeholder: ImagePlaceholder.albumArt, - ), + Container( + height: 40, + width: 40, + decoration: BoxDecoration( + borderRadius: theme.borderRadiusMd, + image: DecorationImage( fit: BoxFit.cover, + image: imageProvider, ), ), ), @@ -148,46 +176,48 @@ class TrackTile extends HookConsumerWidget { child: AnimatedContainer( duration: const Duration(milliseconds: 300), decoration: BoxDecoration( - borderRadius: BorderRadius.circular(4), + borderRadius: theme.borderRadiusMd, color: isHovering - ? Colors.black.withOpacity(0.4) + ? Colors.black.withAlpha(102) : Colors.transparent, ), ), ), Positioned.fill( child: Center( - child: IconTheme( - data: theme.iconTheme - .copyWith(size: 26, color: Colors.white), - child: Skeleton.ignore( - child: Consumer( - builder: (context, ref, _) { - final isFetchingActiveTrack = - ref.watch(queryingTrackInfoProvider); - return AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: (isPlaying && isFetchingActiveTrack) || - isLoading.value - ? const SizedBox( - width: 26, - height: 26, - child: CircularProgressIndicator( - strokeWidth: 1.5, - color: Colors.white, - ), - ) - : isPlaying - ? Icon( - SpotubeIcons.pause, - color: theme.colorScheme.primary, - ) - : !isHovering - ? const SizedBox.shrink() - : const Icon(SpotubeIcons.play), - ); - }, - ), + child: Skeleton.ignore( + child: Consumer( + builder: (context, ref, _) { + final isFetchingActiveTrack = + ref.watch(queryingTrackInfoProvider); + return AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + child: switch (( + isPlaying, + isFetchingActiveTrack, + isPlaying, + isHovering, + isLoading.value + )) { + (true, true, _, _, _) || + (_, _, _, _, true) => + const SizedBox( + width: 26, + height: 26, + child: CircularProgressIndicator(), + ), + (_, _, true, _, _) => Icon( + SpotubeIcons.pause, + color: theme.colorScheme.primary, + ), + (_, _, _, true, _) => const Icon( + SpotubeIcons.play, + color: Colors.white, + ), + _ => const SizedBox.shrink(), + }, + ); + }, ), ), ), @@ -200,37 +230,59 @@ class TrackTile extends HookConsumerWidget { children: [ Expanded( flex: 6, - child: switch (track) { - LocalTrack() => Text( - track.name!, + child: AbsorbPointer( + absorbing: selectionMode, + child: switch (track) { + SpotubeLocalTrackObject() => Text( + track.name, maxLines: 1, overflow: TextOverflow.ellipsis, ), - _ => LinkText( - track.name!, - "/track/${track.id}", - push: true, - maxLines: 1, - overflow: TextOverflow.ellipsis, + _ => Row( + mainAxisSize: MainAxisSize.min, + children: [ + Flexible( + child: Button( + style: ButtonVariance.link.copyWith( + padding: (context, states, value) => + EdgeInsets.zero, + ), + onPressed: effectiveSelection + ? null + : () { + context + .navigateTo(TrackRoute(trackId: track.id)); + }, + child: Text( + track.name, + maxLines: 1, + overflow: TextOverflow.ellipsis, + ), + ), + ), + ], ), }, + ), ), if (constrains.mdAndUp) ...[ const SizedBox(width: 8), Expanded( flex: 4, child: switch (track) { - LocalTrack() => Text( - track.album!.name!, + SpotubeLocalTrackObject() => Text( + track.album.name, maxLines: 1, overflow: TextOverflow.ellipsis, ), _ => Align( alignment: Alignment.centerLeft, child: LinkText( - track.album!.name!, - "/album/${track.album?.id}", - extra: track.album, + track.album.name, + AlbumRoute( + album: track.album, + id: track.album.id, + ), push: true, overflow: TextOverflow.ellipsis, ), @@ -242,21 +294,24 @@ class TrackTile extends HookConsumerWidget { ), subtitle: Align( alignment: Alignment.centerLeft, - child: track is LocalTrack + child: track is SpotubeLocalTrackObject ? Text( - track.artists?.asString() ?? '', + track.artists.asString(), ) : ClipRect( child: ConstrainedBox( constraints: const BoxConstraints(maxHeight: 40), - child: ArtistLink( - artists: track.artists ?? [], - onOverflowArtistClick: () => ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": track.id!, - }, + child: AbsorbPointer( + absorbing: effectiveSelection, + child: ArtistLink( + artists: track.artists, + onOverflowArtistClick: effectiveSelection + ? () {} + : () { + context.navigateTo( + TrackRoute(trackId: track.id), + ); + }, ), ), ), @@ -267,16 +322,19 @@ class TrackTile extends HookConsumerWidget { children: [ const SizedBox(width: 8), Text( - Duration(milliseconds: track.durationMs ?? 0) + Duration(milliseconds: track.durationMs) .toHumanReadableString(padZero: false), maxLines: 1, overflow: TextOverflow.ellipsis, ), - TrackOptions( - track: track, - playlistId: playlistId, - userPlaylist: userPlaylist, - showMenuCbRef: showOptionCbRef, + Builder( + builder: (context) { + return TrackOptionsButton( + track: track, + userPlaylist: userPlaylist, + playlistId: playlistId, + ); + }, ), if (kIsDesktop) const Gap(10), ], diff --git a/lib/components/tracks_view/sections/body/track_view_body.dart b/lib/components/tracks_view/sections/body/track_view_body.dart deleted file mode 100644 index 0f161b0c..00000000 --- a/lib/components/tracks_view/sections/body/track_view_body.dart +++ /dev/null @@ -1,192 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter/services.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:gap/gap.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/components/dialogs/select_device_dialog.dart'; -import 'package:spotube/components/expandable_search/expandable_search.dart'; -import 'package:spotube/components/track_tile/track_tile.dart'; -import 'package:spotube/components/tracks_view/sections/body/track_view_body_headers.dart'; -import 'package:spotube/components/tracks_view/sections/body/use_is_user_playlist.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:spotube/components/tracks_view/track_view_provider.dart'; -import 'package:spotube/extensions/list.dart'; -import 'package:spotube/models/connect/connect.dart'; -import 'package:spotube/provider/connect/connect.dart'; -import 'package:spotube/provider/history/history.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/utils/service_utils.dart'; -import 'package:very_good_infinite_list/very_good_infinite_list.dart'; - -class TrackViewBodySection extends HookConsumerWidget { - const TrackViewBodySection({super.key}); - - @override - Widget build(BuildContext context, ref) { - final playlist = ref.watch(audioPlayerProvider); - final playlistNotifier = ref.watch(audioPlayerProvider.notifier); - final historyNotifier = ref.watch(playbackHistoryActionsProvider); - final props = InheritedTrackView.of(context); - final trackViewState = ref.watch(trackViewProvider(props.tracks)); - - final searchController = useTextEditingController(); - final searchFocus = useFocusNode(); - - useValueListenable(searchController); - final searchQuery = searchController.text; - - final isFiltering = useState(false); - - final uniqTracks = useMemoized(() { - final trackIds = props.tracks.map((e) => e.id).toSet(); - return props.tracks.where((e) => trackIds.remove(e.id)).toList(); - }, [props.tracks]); - - final tracks = useMemoized(() { - List filteredTracks; - if (searchQuery.isEmpty) { - filteredTracks = uniqTracks; - } else { - filteredTracks = uniqTracks - .map((e) => (weightedRatio(e.name!, searchQuery), e)) - .sorted((a, b) => b.$1.compareTo(a.$1)) - .where((e) => e.$1 > 50) - .map((e) => e.$2) - .toList(); - } - return ServiceUtils.sortTracks(filteredTracks, trackViewState.sortBy); - }, [trackViewState.sortBy, searchQuery, uniqTracks]); - - final isUserPlaylist = useIsUserPlaylist(ref, props.collectionId); - - final isActive = playlist.collections.contains(props.collectionId); - - final onTapTrackTile = useCallback((Track track, int index) async { - if (trackViewState.isSelecting) { - trackViewState.toggleTrackSelection(track.id!); - return; - } - - final isRemoteDevice = await showSelectDeviceDialog(context, ref); - - if (isRemoteDevice) { - final remotePlayback = ref.read(connectProvider.notifier); - final remoteQueue = ref.read(queueProvider); - if (remoteQueue.collections.contains(props.collectionId) || - remoteQueue.tracks.any((s) => s.id == track.id)) { - await playlistNotifier.jumpToTrack(track); - } else { - final tracks = await props.pagination.onFetchAll(); - await remotePlayback.load( - props.collection is AlbumSimple - ? WebSocketLoadEventData.album( - tracks: tracks, - collection: props.collection as AlbumSimple, - initialIndex: index, - ) - : WebSocketLoadEventData.playlist( - tracks: tracks, - collection: props.collection as PlaylistSimple, - initialIndex: index, - ), - ); - } - } else { - if (isActive || playlist.tracks.containsBy(track, (a) => a.id)) { - await playlistNotifier.jumpToTrack(track); - } else { - final tracks = await props.pagination.onFetchAll(); - await playlistNotifier.load( - tracks, - initialIndex: index, - autoPlay: true, - ); - playlistNotifier.addCollection(props.collectionId); - if (props.collection is AlbumSimple) { - historyNotifier.addAlbums([props.collection as AlbumSimple]); - } else { - historyNotifier.addPlaylists([props.collection as PlaylistSimple]); - } - } - } - }, [isActive, playlist, props, playlistNotifier, historyNotifier]); - - return SliverMainAxisGroup( - slivers: [ - SliverToBoxAdapter( - child: TrackViewBodyHeaders( - isFiltering: isFiltering, - searchFocus: searchFocus, - ), - ), - const SliverGap(8), - SliverToBoxAdapter( - child: ExpandableSearchField( - isFiltering: isFiltering.value, - onChangeFiltering: (value) { - isFiltering.value = value; - }, - searchController: searchController, - searchFocus: searchFocus, - ), - ), - SliverSafeArea( - top: false, - sliver: SliverInfiniteList( - itemCount: tracks.length, - onFetchData: props.pagination.onFetchMore, - isLoading: props.pagination.isLoading, - hasReachedMax: !props.pagination.hasNextPage, - loadingBuilder: (context) => Skeletonizer( - enabled: true, - child: TrackTile( - playlist: playlist, - track: FakeData.track, - index: 0, - ), - ), - emptyBuilder: (context) => Skeletonizer( - enabled: true, - child: Column( - children: List.generate( - 10, - (index) => TrackTile( - track: FakeData.track, - index: index, - playlist: playlist, - ), - ), - ), - ), - itemBuilder: (context, index) { - final track = tracks[index]; - return TrackTile( - playlist: playlist, - track: track, - index: index, - selected: trackViewState.selectedTrackIds.contains(track.id!), - playlistId: props.collectionId, - userPlaylist: isUserPlaylist, - onChanged: !trackViewState.isSelecting - ? null - : (value) { - trackViewState.toggleTrackSelection(track.id!); - }, - onLongPress: () { - trackViewState.selectTrack(track.id!); - HapticFeedback.selectionClick(); - }, - onTap: () => onTapTrackTile(track, index), - ); - }, - ), - ), - ], - ); - } -} diff --git a/lib/components/tracks_view/sections/body/track_view_body_headers.dart b/lib/components/tracks_view/sections/body/track_view_body_headers.dart deleted file mode 100644 index 82cc7706..00000000 --- a/lib/components/tracks_view/sections/body/track_view_body_headers.dart +++ /dev/null @@ -1,105 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/components/expandable_search/expandable_search.dart'; -import 'package:spotube/components/sort_tracks_dropdown.dart'; -import 'package:spotube/components/tracks_view/sections/body/track_view_options.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:spotube/components/tracks_view/track_view_provider.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/utils/platform.dart'; - -class TrackViewBodyHeaders extends HookConsumerWidget { - final ValueNotifier isFiltering; - final FocusNode searchFocus; - - const TrackViewBodyHeaders({ - super.key, - required this.isFiltering, - required this.searchFocus, - }); - - @override - Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); - final props = InheritedTrackView.of(context); - final trackViewState = ref.watch(trackViewProvider(props.tracks)); - return LayoutBuilder( - builder: (context, constrains) { - return Row( - children: [ - AnimatedSwitcher( - duration: const Duration(milliseconds: 200), - transitionBuilder: (child, animation) { - return FadeTransition( - opacity: animation, - child: ScaleTransition( - scale: animation, - child: child, - ), - ); - }, - child: Checkbox( - value: trackViewState.hasSelectedAll, - onChanged: (checked) { - if (checked == true) { - trackViewState.selectAll(); - } else { - trackViewState.deselectAll(); - } - }, - ), - ), - Expanded( - flex: 7, - child: Row( - children: [ - Text( - context.l10n.title, - style: textTheme.bodyLarge, - overflow: TextOverflow.ellipsis, - ), - ], - ), - ), - // used alignment of this table-head - if (constrains.mdAndUp) - Expanded( - flex: 3, - child: Row( - children: [ - Text( - context.l10n.album, - overflow: TextOverflow.ellipsis, - style: textTheme.bodyLarge, - ), - ], - ), - ), - SortTracksDropdown( - value: trackViewState.sortBy, - onChanged: (value) { - trackViewState.sort(value); - }, - ), - ExpandableSearchButton( - isFiltering: isFiltering.value, - searchFocus: searchFocus, - onPressed: (value) { - isFiltering.value = value; - if (value) { - searchFocus.requestFocus(); - } else { - searchFocus.unfocus(); - } - }, - ), - const TrackViewBodyOptions(), - if (kIsDesktop) const Gap(10), - ], - ); - }, - ); - } -} diff --git a/lib/components/tracks_view/sections/body/track_view_options.dart b/lib/components/tracks_view/sections/body/track_view_options.dart deleted file mode 100644 index 23198aec..00000000 --- a/lib/components/tracks_view/sections/body/track_view_options.dart +++ /dev/null @@ -1,140 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/adaptive/adaptive_pop_sheet_list.dart'; -import 'package:spotube/components/dialogs/confirm_download_dialog.dart'; -import 'package:spotube/components/dialogs/playlist_add_track_dialog.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:spotube/components/tracks_view/track_view_provider.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/download_manager_provider.dart'; -import 'package:spotube/provider/history/history.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -class TrackViewBodyOptions extends HookConsumerWidget { - const TrackViewBodyOptions({super.key}); - - @override - Widget build(BuildContext context, ref) { - final props = InheritedTrackView.of(context); - final ThemeData(:textTheme) = Theme.of(context); - - ref.watch(downloadManagerProvider); - final downloader = ref.watch(downloadManagerProvider.notifier); - final playlistNotifier = ref.watch(audioPlayerProvider.notifier); - final historyNotifier = ref.watch(playbackHistoryActionsProvider); - final audioSource = - ref.watch(userPreferencesProvider.select((s) => s.audioSource)); - - final trackViewState = ref.watch(trackViewProvider(props.tracks)); - final selectedTracks = trackViewState.selectedTracks; - - return AdaptivePopSheetList( - tooltip: context.l10n.more_actions, - headings: [ - Text( - context.l10n.more_actions, - style: textTheme.bodyLarge, - ), - ], - onSelected: (action) async { - switch (action) { - case "download": - { - final confirmed = audioSource == AudioSource.piped || - await showDialog( - context: context, - builder: (context) { - return const ConfirmDownloadDialog(); - }, - ); - if (confirmed != true) return; - await downloader.batchAddToQueue(selectedTracks); - trackViewState.deselectAll(); - break; - } - case "add-to-playlist": - { - if (context.mounted) { - await showDialog( - context: context, - builder: (context) { - return PlaylistAddTrackDialog( - openFromPlaylist: props.collectionId, - tracks: selectedTracks.toList(), - ); - }, - ); - } - break; - } - case "play-next": - { - playlistNotifier.addTracksAtFirst(selectedTracks); - playlistNotifier.addCollection(props.collectionId); - if (props.collection is AlbumSimple) { - historyNotifier.addAlbums([props.collection as AlbumSimple]); - } else { - historyNotifier - .addPlaylists([props.collection as PlaylistSimple]); - } - trackViewState.deselectAll(); - break; - } - case "add-to-queue": - { - playlistNotifier.addTracks(selectedTracks); - playlistNotifier.addCollection(props.collectionId); - if (props.collection is AlbumSimple) { - historyNotifier.addAlbums([props.collection as AlbumSimple]); - } else { - historyNotifier - .addPlaylists([props.collection as PlaylistSimple]); - } - trackViewState.deselectAll(); - break; - } - default: - } - }, - icon: const Icon(SpotubeIcons.moreVertical), - children: [ - PopSheetEntry( - value: "download", - leading: const Icon(SpotubeIcons.download), - enabled: selectedTracks.isNotEmpty, - title: Text( - context.l10n.download_count(selectedTracks.length), - ), - ), - PopSheetEntry( - value: "add-to-playlist", - leading: const Icon(SpotubeIcons.playlistAdd), - enabled: selectedTracks.isNotEmpty, - title: Text( - context.l10n.add_count_to_playlist(selectedTracks.length), - ), - ), - PopSheetEntry( - enabled: selectedTracks.isNotEmpty, - value: "add-to-queue", - leading: const Icon(SpotubeIcons.queueAdd), - title: Text( - context.l10n.add_count_to_queue(selectedTracks.length), - ), - ), - PopSheetEntry( - enabled: selectedTracks.isNotEmpty, - value: "play-next", - leading: const Icon(SpotubeIcons.lightning), - title: Text( - context.l10n.play_count_next(selectedTracks.length), - ), - ), - ], - ); - } -} diff --git a/lib/components/tracks_view/sections/header/flexible_header.dart b/lib/components/tracks_view/sections/header/flexible_header.dart deleted file mode 100644 index 508d289c..00000000 --- a/lib/components/tracks_view/sections/header/flexible_header.dart +++ /dev/null @@ -1,167 +0,0 @@ -import 'dart:ui'; - -import 'package:flutter/material.dart'; - -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/assets.gen.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/tracks_view/sections/header/header_actions.dart'; -import 'package:spotube/components/tracks_view/sections/header/header_buttons.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:gap/gap.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/string.dart'; -import 'package:spotube/hooks/utils/use_palette_color.dart'; -import 'package:spotube/utils/platform.dart'; - -class TrackViewFlexHeader extends HookConsumerWidget { - const TrackViewFlexHeader({super.key}); - - @override - Widget build(BuildContext context, ref) { - final props = InheritedTrackView.of(context); - final ThemeData(:colorScheme, :textTheme, :iconTheme) = Theme.of(context); - final defaultTextStyle = DefaultTextStyle.of(context); - final mediaQuery = MediaQuery.of(context); - - final palette = usePaletteColor(props.image, ref); - - return IconTheme( - data: iconTheme.copyWith(color: palette.bodyTextColor), - child: SliverLayoutBuilder( - builder: (context, constrains) { - final isExpanded = constrains.scrollOffset < 350; - - final headingStyle = (mediaQuery.mdAndDown - ? textTheme.headlineSmall - : textTheme.headlineMedium) - ?.copyWith( - color: palette.bodyTextColor, - ); - return SliverAppBar( - iconTheme: iconTheme.copyWith( - color: palette.bodyTextColor, - size: 16, - ), - actions: isExpanded - ? [] - : [ - const TrackViewHeaderActions(), - TrackViewHeaderButtons(compact: true, color: palette), - ], - floating: false, - pinned: true, - expandedHeight: 450, - automaticallyImplyLeading: kIsMobile, - backgroundColor: palette.color, - title: isExpanded ? null : Text(props.title, style: headingStyle), - flexibleSpace: FlexibleSpaceBar( - background: Container( - clipBehavior: Clip.hardEdge, - decoration: BoxDecoration( - image: DecorationImage( - image: UniversalImage.imageProvider(props.image), - fit: BoxFit.cover, - ), - ), - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 10, sigmaY: 10), - child: DecoratedBox( - decoration: BoxDecoration( - gradient: LinearGradient( - colors: [ - Colors.black45, - colorScheme.surface, - ], - begin: const FractionalOffset(0, 0), - end: const FractionalOffset(0, 1), - tileMode: TileMode.clamp, - ), - ), - child: SafeArea( - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Column( - mainAxisSize: MainAxisSize.min, - mainAxisAlignment: MainAxisAlignment.center, - crossAxisAlignment: CrossAxisAlignment.center, - children: [ - ConstrainedBox( - constraints: BoxConstraints( - maxWidth: mediaQuery.mdAndDown - ? mediaQuery.size.width - : 800, - ), - child: Flex( - direction: mediaQuery.mdAndDown - ? Axis.vertical - : Axis.horizontal, - mainAxisSize: MainAxisSize.min, - children: [ - ClipRRect( - borderRadius: BorderRadius.circular(10), - child: UniversalImage( - path: props.image, - width: 200, - height: 200, - placeholder: Assets.albumPlaceholder.path, - ), - ), - const Gap(20), - Flexible( - child: Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: mediaQuery.mdAndDown - ? CrossAxisAlignment.center - : CrossAxisAlignment.start, - children: [ - Text( - props.title, - style: headingStyle, - textAlign: mediaQuery.mdAndDown - ? TextAlign.center - : TextAlign.start, - maxLines: 2, - overflow: TextOverflow.ellipsis, - ), - const SizedBox(height: 10), - if (props.description != null && - props.description!.isNotEmpty) - Text( - props.description! - .unescapeHtml() - .cleanHtml(), - style: - defaultTextStyle.style.copyWith( - color: palette.bodyTextColor, - ), - textAlign: mediaQuery.mdAndDown - ? TextAlign.center - : TextAlign.start, - maxLines: 2, - overflow: TextOverflow.ellipsis, - ), - const Gap(10), - const TrackViewHeaderActions(), - const Gap(10), - TrackViewHeaderButtons(color: palette), - ], - ), - ), - ], - ), - ), - ], - ), - ), - ), - ), - ), - ), - ), - ); - }, - ), - ); - } -} diff --git a/lib/components/tracks_view/sections/header/header_actions.dart b/lib/components/tracks_view/sections/header/header_actions.dart deleted file mode 100644 index 8e378f97..00000000 --- a/lib/components/tracks_view/sections/header/header_actions.dart +++ /dev/null @@ -1,111 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter/services.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/playlist/playlist_create_dialog.dart'; -import 'package:spotube/components/heart_button/heart_button.dart'; -import 'package:spotube/components/tracks_view/sections/body/use_is_user_playlist.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/history/history.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; - -class TrackViewHeaderActions extends HookConsumerWidget { - const TrackViewHeaderActions({super.key}); - - @override - Widget build(BuildContext context, ref) { - final props = InheritedTrackView.of(context); - - final playlist = ref.watch(audioPlayerProvider); - final playlistNotifier = ref.watch(audioPlayerProvider.notifier); - final historyNotifier = ref.watch(playbackHistoryActionsProvider); - - final isActive = playlist.collections.contains(props.collectionId); - - final isUserPlaylist = useIsUserPlaylist(ref, props.collectionId); - - final scaffoldMessenger = ScaffoldMessenger.of(context); - - final auth = ref.watch(authenticationProvider); - - final copiedText = - context.l10n.copied_shareurl_to_clipboard(props.shareUrl); - - return Row( - mainAxisSize: MainAxisSize.min, - children: [ - IconButton( - tooltip: context.l10n.share, - icon: const Icon(SpotubeIcons.share), - onPressed: () async { - await Clipboard.setData( - ClipboardData(text: props.shareUrl), - ); - - scaffoldMessenger.showSnackBar( - SnackBar( - width: 300, - behavior: SnackBarBehavior.floating, - content: Text( - copiedText, - textAlign: TextAlign.center, - ), - ), - ); - }, - ), - IconButton( - icon: const Icon(SpotubeIcons.queueAdd), - tooltip: context.l10n.add_to_queue, - onPressed: isActive || props.tracks.isEmpty - ? null - : () async { - final tracks = await props.pagination.onFetchAll(); - await playlistNotifier.addTracks(tracks); - playlistNotifier.addCollection(props.collectionId); - if (props.collection is AlbumSimple) { - historyNotifier - .addAlbums([props.collection as AlbumSimple]); - } else { - historyNotifier - .addPlaylists([props.collection as PlaylistSimple]); - } - }, - ), - if (props.onHeart != null && auth.asData?.value != null) - HeartButton( - isLiked: props.isLiked, - icon: isUserPlaylist ? SpotubeIcons.trash : null, - tooltip: props.isLiked - ? context.l10n.remove_from_favorites - : context.l10n.save_as_favorite, - onPressed: () async { - final shouldPop = await props.onHeart?.call(); - if (isUserPlaylist && shouldPop == true && context.mounted) { - context.pop(); - } - }, - ), - if (isUserPlaylist) - IconButton( - icon: const Icon(SpotubeIcons.edit), - onPressed: () { - showDialog( - context: context, - builder: (context) { - return PlaylistCreateDialog( - playlistId: props.collectionId, - trackIds: props.tracks.map((e) => e.id!).toList(), - ); - }, - ); - }, - ), - ], - ); - } -} diff --git a/lib/components/tracks_view/sections/header/header_buttons.dart b/lib/components/tracks_view/sections/header/header_buttons.dart deleted file mode 100644 index 54e0f0cf..00000000 --- a/lib/components/tracks_view/sections/header/header_buttons.dart +++ /dev/null @@ -1,206 +0,0 @@ -import 'dart:math'; - -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:palette_generator/palette_generator.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/dialogs/select_device_dialog.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/models/connect/connect.dart'; -import 'package:spotube/provider/connect/connect.dart'; -import 'package:spotube/provider/history/history.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/services/audio_player/audio_player.dart'; - -class TrackViewHeaderButtons extends HookConsumerWidget { - final PaletteColor color; - final bool compact; - const TrackViewHeaderButtons({ - super.key, - required this.color, - this.compact = false, - }); - - @override - Widget build(BuildContext context, ref) { - final props = InheritedTrackView.of(context); - final playlist = ref.watch(audioPlayerProvider); - final playlistNotifier = ref.watch(audioPlayerProvider.notifier); - final historyNotifier = ref.watch(playbackHistoryActionsProvider); - - final isActive = playlist.collections.contains(props.collectionId); - - final isLoading = useState(false); - - const progressIndicator = Center( - child: SizedBox.square( - dimension: 20, - child: CircularProgressIndicator(strokeWidth: .8), - ), - ); - - void onShuffle() async { - try { - isLoading.value = true; - - final initialTracks = props.tracks; - if (!context.mounted) return; - - final isRemoteDevice = await showSelectDeviceDialog(context, ref); - if (isRemoteDevice) { - final allTracks = await props.pagination.onFetchAll(); - final remotePlayback = ref.read(connectProvider.notifier); - await remotePlayback.load( - props.collection is AlbumSimple - ? WebSocketLoadEventData.album( - tracks: allTracks, - collection: props.collection as AlbumSimple, - initialIndex: Random().nextInt(allTracks.length)) - : WebSocketLoadEventData.playlist( - tracks: allTracks, - collection: props.collection as PlaylistSimple, - initialIndex: Random().nextInt(allTracks.length), - ), - ); - await remotePlayback.setShuffle(true); - } else { - await playlistNotifier.load( - initialTracks, - autoPlay: true, - initialIndex: Random().nextInt(initialTracks.length), - ); - await audioPlayer.setShuffle(true); - playlistNotifier.addCollection(props.collectionId); - if (props.collection is AlbumSimple) { - historyNotifier.addAlbums([props.collection as AlbumSimple]); - } else { - historyNotifier.addPlaylists([props.collection as PlaylistSimple]); - } - - final allTracks = await props.pagination.onFetchAll(); - - await playlistNotifier.addTracks( - allTracks.sublist(initialTracks.length), - ); - } - } finally { - isLoading.value = false; - } - } - - void onPlay() async { - try { - isLoading.value = true; - - final initialTracks = props.tracks; - - if (!context.mounted) return; - - final isRemoteDevice = await showSelectDeviceDialog(context, ref); - if (isRemoteDevice) { - final allTracks = await props.pagination.onFetchAll(); - final remotePlayback = ref.read(connectProvider.notifier); - await remotePlayback.load( - props.collection is AlbumSimple - ? WebSocketLoadEventData.album( - tracks: allTracks, - collection: props.collection as AlbumSimple, - ) - : WebSocketLoadEventData.playlist( - tracks: allTracks, - collection: props.collection as PlaylistSimple, - ), - ); - } else { - await playlistNotifier.load(initialTracks, autoPlay: true); - playlistNotifier.addCollection(props.collectionId); - if (props.collection is AlbumSimple) { - historyNotifier.addAlbums([props.collection as AlbumSimple]); - } else { - historyNotifier.addPlaylists([props.collection as PlaylistSimple]); - } - - final allTracks = await props.pagination.onFetchAll(); - - await playlistNotifier.addTracks( - allTracks.sublist(initialTracks.length), - ); - } - } finally { - if (context.mounted) { - isLoading.value = false; - } - } - } - - if (compact) { - return Row( - mainAxisSize: MainAxisSize.min, - children: [ - if (!isActive && !isLoading.value) - IconButton( - icon: const Icon(SpotubeIcons.shuffle), - onPressed: props.tracks.isEmpty ? null : onShuffle, - ), - const Gap(10), - IconButton.filledTonal( - icon: isActive - ? const Icon(SpotubeIcons.pause) - : isLoading.value - ? progressIndicator - : const Icon(SpotubeIcons.play), - onPressed: isActive || props.tracks.isEmpty || isLoading.value - ? null - : onPlay, - ), - const Gap(10), - ], - ); - } - - return Row( - mainAxisSize: MainAxisSize.min, - children: [ - AnimatedOpacity( - duration: const Duration(milliseconds: 300), - opacity: isActive || isLoading.value ? 0 : 1, - child: AnimatedSize( - duration: const Duration(milliseconds: 300), - child: SizedBox.square( - dimension: isActive || isLoading.value ? 0 : null, - child: FilledButton.icon( - style: ElevatedButton.styleFrom( - backgroundColor: Colors.white, - foregroundColor: Colors.black, - minimumSize: const Size(150, 40)), - label: Text(context.l10n.shuffle), - icon: const Icon(SpotubeIcons.shuffle), - onPressed: props.tracks.isEmpty ? null : onShuffle, - ), - ), - ), - ), - const Gap(10), - FilledButton.icon( - style: ElevatedButton.styleFrom( - backgroundColor: color.color, - foregroundColor: color.bodyTextColor, - minimumSize: const Size(150, 40)), - onPressed: isActive || props.tracks.isEmpty || isLoading.value - ? null - : onPlay, - icon: isActive - ? const Icon(SpotubeIcons.pause) - : isLoading.value - ? progressIndicator - : const Icon(SpotubeIcons.play), - label: Text(context.l10n.play), - ), - ], - ); - } -} diff --git a/lib/components/tracks_view/track_view.dart b/lib/components/tracks_view/track_view.dart deleted file mode 100644 index 2a3f5237..00000000 --- a/lib/components/tracks_view/track_view.dart +++ /dev/null @@ -1,52 +0,0 @@ -import 'package:flutter/material.dart'; - -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:sliver_tools/sliver_tools.dart'; -import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/components/tracks_view/sections/header/flexible_header.dart'; -import 'package:spotube/components/tracks_view/sections/body/track_view_body.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; -import 'package:spotube/utils/platform.dart'; - -class TrackView extends HookConsumerWidget { - const TrackView({super.key}); - - @override - Widget build(BuildContext context, ref) { - final props = InheritedTrackView.of(context); - final controller = useScrollController(); - - return Scaffold( - appBar: kIsDesktop - ? const PageWindowTitleBar( - backgroundColor: Colors.transparent, - foregroundColor: Colors.white, - leadingWidth: 400, - leading: Align( - alignment: Alignment.centerLeft, - child: BackButton(color: Colors.white), - ), - ) - : null, - extendBodyBehindAppBar: true, - body: RefreshIndicator( - onRefresh: props.pagination.onRefresh, - child: InterScrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - slivers: const [ - TrackViewFlexHeader(), - SliverAnimatedSwitcher( - duration: Duration(milliseconds: 500), - child: TrackViewBodySection(), - ), - ], - ), - ), - ), - ); - } -} diff --git a/lib/components/tracks_view/track_view_props.dart b/lib/components/tracks_view/track_view_props.dart deleted file mode 100644 index b0a00ae2..00000000 --- a/lib/components/tracks_view/track_view_props.dart +++ /dev/null @@ -1,99 +0,0 @@ -import 'dart:async'; - -import 'package:flutter/material.dart' hide Page; -import 'package:spotify/spotify.dart'; - -class PaginationProps { - final bool hasNextPage; - final bool isLoading; - final VoidCallback onFetchMore; - final Future Function() onRefresh; - final Future> Function() onFetchAll; - - const PaginationProps({ - required this.hasNextPage, - required this.isLoading, - required this.onFetchMore, - required this.onFetchAll, - required this.onRefresh, - }); - - @override - operator ==(Object other) { - return other is PaginationProps && - other.hasNextPage == hasNextPage && - other.isLoading == isLoading && - other.onFetchMore == onFetchMore && - other.onFetchAll == onFetchAll && - other.onRefresh == onRefresh; - } - - @override - int get hashCode => - super.hashCode ^ - hasNextPage.hashCode ^ - isLoading.hashCode ^ - onFetchMore.hashCode ^ - onFetchAll.hashCode ^ - onRefresh.hashCode; -} - -class InheritedTrackView extends InheritedWidget { - final Object collection; - final String title; - final String? description; - final String image; - final String routePath; - final List tracks; - final PaginationProps pagination; - final bool isLiked; - final String shareUrl; - - // events - final FutureOr Function()? onHeart; // if null heart button will hidden - - const InheritedTrackView({ - super.key, - required super.child, - required this.collection, - required this.title, - this.description, - required this.image, - required this.tracks, - required this.pagination, - required this.routePath, - required this.shareUrl, - this.isLiked = false, - this.onHeart, - }) : assert(collection is AlbumSimple || collection is PlaylistSimple); - - String get collectionId => collection is AlbumSimple - ? (collection as AlbumSimple).id! - : (collection as PlaylistSimple).id!; - - @override - bool updateShouldNotify(InheritedTrackView oldWidget) { - return oldWidget.title != title || - oldWidget.description != description || - oldWidget.image != image || - oldWidget.tracks != tracks || - oldWidget.pagination != pagination || - oldWidget.isLiked != isLiked || - oldWidget.onHeart != onHeart || - oldWidget.shareUrl != shareUrl || - oldWidget.routePath != routePath || - oldWidget.collection != collection || - oldWidget.child != child; - } - - static InheritedTrackView of(BuildContext context) { - final widget = - context.dependOnInheritedWidgetOfExactType(); - if (widget == null) { - throw Exception( - 'InheritedTrackView not found. Make sure to wrap [TrackView] with [InheritedTrackView]', - ); - } - return widget; - } -} diff --git a/lib/components/tracks_view/track_view_provider.dart b/lib/components/tracks_view/track_view_provider.dart deleted file mode 100644 index 16aa6d9c..00000000 --- a/lib/components/tracks_view/track_view_provider.dart +++ /dev/null @@ -1,64 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/modules/library/user_local_tracks.dart'; - -class TrackViewNotifier extends ChangeNotifier { - List tracks; - List selectedTrackIds; - SortBy sortBy; - String? searchQuery; - - TrackViewNotifier( - this.tracks, { - this.selectedTrackIds = const [], - this.sortBy = SortBy.none, - this.searchQuery, - }); - - bool get isSelecting => selectedTrackIds.isNotEmpty; - - bool get hasSelectedAll => - selectedTrackIds.length == tracks.length && tracks.isNotEmpty; - - List get selectedTracks => - tracks.where((e) => selectedTrackIds.contains(e.id)).toList(); - - void selectTrack(String trackId) { - selectedTrackIds = [...selectedTrackIds, trackId]; - notifyListeners(); - } - - void unselectTrack(String trackId) { - selectedTrackIds = selectedTrackIds.where((e) => e != trackId).toList(); - notifyListeners(); - } - - void toggleTrackSelection(String trackId) { - if (selectedTrackIds.contains(trackId)) { - unselectTrack(trackId); - } else { - selectTrack(trackId); - } - } - - void selectAll() { - selectedTrackIds = tracks.map((e) => e.id!).toList(); - notifyListeners(); - } - - void deselectAll() { - selectedTrackIds = []; - notifyListeners(); - } - - void sort(SortBy sortBy) { - this.sortBy = sortBy; - notifyListeners(); - } -} - -final trackViewProvider = ChangeNotifierProvider.autoDispose - .family>((ref, tracks) { - return TrackViewNotifier(tracks); -}); diff --git a/lib/components/ui/button_tile.dart b/lib/components/ui/button_tile.dart new file mode 100644 index 00000000..e31a09a5 --- /dev/null +++ b/lib/components/ui/button_tile.dart @@ -0,0 +1,109 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +class ButtonTile extends StatelessWidget { + final Widget? title; + final Widget? subtitle; + final Widget? leading; + final Widget? trailing; + final bool enabled; + final VoidCallback? onPressed; + final VoidCallback? onLongPress; + final bool selected; + final AbstractButtonStyle style; + final EdgeInsets? padding; + + const ButtonTile({ + super.key, + this.title, + this.subtitle, + this.leading, + this.trailing, + this.enabled = true, + this.onPressed, + this.onLongPress, + this.selected = false, + this.padding, + this.style = ButtonVariance.outline, + }); + + @override + Widget build(BuildContext context) { + final ThemeData(:colorScheme, :typography) = Theme.of(context); + + return GestureDetector( + onLongPress: onLongPress, + child: Button( + enabled: enabled, + onPressed: onPressed, + style: style.copyWith( + padding: + padding != null ? (context, states, value) => padding! : null, + decoration: (context, states, value) { + final decoration = + style.decoration(context, states) as BoxDecoration; + + if (selected) { + return switch (style) { + ButtonVariance.outline => decoration.copyWith( + border: Border.all( + color: colorScheme.primary, + width: 1.0, + ), + color: colorScheme.primary.withAlpha(25), + ), + ButtonVariance.ghost || _ => decoration.copyWith( + color: colorScheme.primary.withAlpha(25), + ), + }; + } + + return decoration; + }, + iconTheme: (context, states, value) { + final iconTheme = style.iconTheme(context, states); + + if (selected && style == ButtonVariance.outline) { + return iconTheme.copyWith( + color: colorScheme.primary, + ); + } + + return iconTheme; + }, + textStyle: (context, states, value) { + final textStyle = style.textStyle(context, states); + + if (selected && style == ButtonVariance.outline) { + return textStyle.copyWith( + color: colorScheme.primary, + ); + } + + return textStyle; + }, + ), + alignment: Alignment.centerLeft, + child: SizedBox( + width: double.infinity, + child: Basic( + padding: EdgeInsets.zero, + leadingAlignment: Alignment.center, + trailingAlignment: Alignment.center, + leading: leading, + title: title, + subtitle: + style == ButtonVariance.outline && selected && subtitle != null + ? DefaultTextStyle( + style: typography.xSmall.copyWith( + color: colorScheme.primary, + ), + child: subtitle!, + ) + : subtitle, + trailing: trailing, + ), + ), + ), + ); + } +} diff --git a/lib/extensions/album_simple.dart b/lib/extensions/album_simple.dart deleted file mode 100644 index 5678390c..00000000 --- a/lib/extensions/album_simple.dart +++ /dev/null @@ -1,21 +0,0 @@ -import 'package:spotify/spotify.dart'; - -extension AlbumExtensions on AlbumSimple { - Album toAlbum() { - Album album = Album(); - album.albumType = albumType; - album.artists = artists; - album.availableMarkets = availableMarkets; - album.externalUrls = externalUrls; - album.href = href; - album.id = id; - album.images = images; - album.name = name; - album.releaseDate = releaseDate; - album.releaseDatePrecision = releaseDatePrecision; - album.tracks = tracks; - album.type = type; - album.uri = uri; - return album; - } -} diff --git a/lib/extensions/artist_simple.dart b/lib/extensions/artist_simple.dart deleted file mode 100644 index 7997355d..00000000 --- a/lib/extensions/artist_simple.dart +++ /dev/null @@ -1,7 +0,0 @@ -import 'package:spotify/spotify.dart'; - -extension ArtistExtension on List { - String asString() { - return map((e) => e.name?.replaceAll(",", " ")).join(", "); - } -} diff --git a/lib/extensions/button_variance.dart b/lib/extensions/button_variance.dart new file mode 100644 index 00000000..cf66d528 --- /dev/null +++ b/lib/extensions/button_variance.dart @@ -0,0 +1,21 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +extension CopyWithButtonVarianceExtension on ButtonVariance { + ButtonVariance copyWith({ + ButtonStateProperty? padding, + ButtonStateProperty? decoration, + ButtonStateProperty? mouseCursor, + ButtonStateProperty? iconTheme, + ButtonStateProperty? margin, + ButtonStateProperty? textStyle, + }) { + return ButtonVariance( + padding: padding ?? this.padding, + decoration: decoration ?? this.decoration, + mouseCursor: mouseCursor ?? this.mouseCursor, + iconTheme: iconTheme ?? this.iconTheme, + margin: margin ?? this.margin, + textStyle: textStyle ?? this.textStyle, + ); + } +} diff --git a/lib/extensions/color.dart b/lib/extensions/color.dart index 68cd8ef7..bc7d65a2 100644 --- a/lib/extensions/color.dart +++ b/lib/extensions/color.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; extension ColorAlterer on Color { Color darken(double amount) { diff --git a/lib/extensions/constrains.dart b/lib/extensions/constrains.dart index dc1027e2..b7353c4f 100644 --- a/lib/extensions/constrains.dart +++ b/lib/extensions/constrains.dart @@ -106,3 +106,22 @@ extension ScreenBreakpoints on MediaQueryData { bool get lgAndDown => isXs || isSm || isMd || isLg; bool get xlAndDown => isXs || isSm || isMd || isLg || isXl; } + +extension SizeBreakpoints on Size { + bool get isXs => width <= Breakpoints.xs; + bool get isSm => width > Breakpoints.xs && width <= Breakpoints.sm; + bool get isMd => width > Breakpoints.sm && width <= Breakpoints.md; + bool get isLg => width > Breakpoints.md && width <= Breakpoints.lg; + bool get isXl => width > Breakpoints.lg && width <= Breakpoints.xl; + bool get is2Xl => width > Breakpoints.xl; + + bool get smAndUp => isSm || isMd || isLg || isXl || is2Xl; + bool get mdAndUp => isMd || isLg || isXl || is2Xl; + bool get lgAndUp => isLg || isXl || is2Xl; + bool get xlAndUp => isXl || is2Xl; + + bool get smAndDown => isXs || isSm; + bool get mdAndDown => isXs || isSm || isMd; + bool get lgAndDown => isXs || isSm || isMd || isLg; + bool get xlAndDown => isXs || isSm || isMd || isLg || isXl; +} diff --git a/lib/extensions/context.dart b/lib/extensions/context.dart index 9ca1e237..29fbb7ca 100644 --- a/lib/extensions/context.dart +++ b/lib/extensions/context.dart @@ -1,5 +1,5 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_gen/gen_l10n/app_localizations.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/l10n/l10n.dart'; extension AppLocale on BuildContext { AppLocalizations get l10n => AppLocalizations.of(this)!; diff --git a/lib/extensions/dio.dart b/lib/extensions/dio.dart new file mode 100644 index 00000000..81bb1e70 --- /dev/null +++ b/lib/extensions/dio.dart @@ -0,0 +1,168 @@ +import 'dart:io'; +import 'package:dio/dio.dart'; +import 'package:path/path.dart'; +import 'package:path_provider/path_provider.dart'; + +extension ChunkDownloaderDioExtension on Dio { + Future chunkDownload( + String urlPath, + dynamic savePath, { + ProgressCallback? onReceiveProgress, + Map? queryParameters, + CancelToken? cancelToken, + bool deleteOnError = true, + FileAccessMode fileAccessMode = FileAccessMode.write, + String lengthHeader = Headers.contentLengthHeader, + Object? data, + Options? options, + int connections = 4, + }) async { + final targetFile = File(savePath.toString()); + final tempRootDir = await getTemporaryDirectory(); + final tempSaveDir = Directory( + join( + tempRootDir.path, + 'Spotube', + '.chunk_dl_${targetFile.uri.pathSegments.last}', + ), + ); + if (await tempSaveDir.exists()) await tempSaveDir.delete(recursive: true); + await tempSaveDir.create(recursive: true); + + try { + int? totalLength; + bool supportsRange = false; + + Response? headResp; + try { + headResp = await head( + urlPath, + queryParameters: queryParameters, + options: Options( + headers: {'Range': 'bytes=0-0'}, + followRedirects: true, + ), + ); + } catch (_) { + // Some servers reject HEAD -> ignore + } + + final lengthStr = headResp?.headers[lengthHeader]?.first; + if (lengthStr != null) { + final parsed = int.tryParse(lengthStr); + if (parsed != null && parsed > 1) { + totalLength = parsed; + } + } + + supportsRange = headResp?.statusCode == 206 || + headResp?.headers.value(HttpHeaders.acceptRangesHeader) == 'bytes'; + + if (totalLength == null || totalLength <= 1) { + final resp = await get( + urlPath, + options: Options( + responseType: ResponseType.stream, + ), + queryParameters: queryParameters, + cancelToken: cancelToken, + ); + + final len = int.tryParse(resp.headers[lengthHeader]?.first ?? ''); + if (len == null || len <= 1) { + // can’t safely chunk — fallback + return download( + urlPath, + savePath, + onReceiveProgress: onReceiveProgress, + queryParameters: queryParameters, + cancelToken: cancelToken, + deleteOnError: deleteOnError, + options: options, + data: data, + ); + } + + totalLength = len; + supportsRange = + resp.headers.value(HttpHeaders.acceptRangesHeader)?.toLowerCase() == + 'bytes'; + } + + if (!supportsRange || connections <= 1) { + return download( + urlPath, + savePath, + onReceiveProgress: onReceiveProgress, + queryParameters: queryParameters, + cancelToken: cancelToken, + deleteOnError: deleteOnError, + options: options, + data: data, + ); + } + + final chunkSize = (totalLength / connections).ceil(); + int downloaded = 0; + + final partFiles = List.generate( + connections, + (i) => File(join(tempSaveDir.path, 'part_$i')), + ); + + final futures = List.generate(connections, (i) async { + final start = i * chunkSize; + final end = (i + 1) * chunkSize - 1; + if (start >= totalLength!) return; + + final resp = await get( + urlPath, + options: Options( + responseType: ResponseType.stream, + headers: {'Range': 'bytes=$start-$end'}, + ), + queryParameters: queryParameters, + cancelToken: cancelToken, + ); + + final file = partFiles[i]; + if (await file.exists()) await file.delete(); + await file.create(recursive: true); + final sink = file.openWrite(); + + await for (final chunk in resp.data!.stream) { + sink.add(chunk); + downloaded += chunk.length; + onReceiveProgress?.call(downloaded, totalLength); + } + + await sink.close(); + }); + + await Future.wait(futures); + + final targetSink = targetFile.openWrite(); + for (final f in partFiles) { + await targetSink.addStream(f.openRead()); + } + await targetSink.close(); + + await tempSaveDir.delete(recursive: true); + + return Response( + requestOptions: RequestOptions(path: urlPath), + data: targetFile, + statusCode: 200, + statusMessage: 'Chunked download completed ($connections connections)', + ); + } catch (e) { + if (deleteOnError) { + if (await targetFile.exists()) await targetFile.delete(); + if (await tempSaveDir.exists()) { + await tempSaveDir.delete(recursive: true); + } + } + rethrow; + } + } +} diff --git a/lib/extensions/image.dart b/lib/extensions/image.dart deleted file mode 100644 index ee78653a..00000000 --- a/lib/extensions/image.dart +++ /dev/null @@ -1,34 +0,0 @@ -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/assets.gen.dart'; -import 'package:spotube/utils/primitive_utils.dart'; -import 'package:collection/collection.dart'; - -enum ImagePlaceholder { - albumArt, - artist, - collection, - online, -} - -extension SpotifyImageExtensions on List? { - String asUrlString({ - int index = 1, - required ImagePlaceholder placeholder, - }) { - final String placeholderUrl = { - ImagePlaceholder.albumArt: Assets.albumPlaceholder.path, - ImagePlaceholder.artist: Assets.userPlaceholder.path, - ImagePlaceholder.collection: Assets.placeholder.path, - ImagePlaceholder.online: - "https://avatars.dicebear.com/api/bottts/${PrimitiveUtils.uuid.v4()}.png", - }[placeholder]!; - - final sortedImage = this?.sorted((a, b) => a.width!.compareTo(b.width!)); - - return sortedImage != null && sortedImage.isNotEmpty - ? sortedImage[ - index > sortedImage.length - 1 ? sortedImage.length - 1 : index] - .url! - : placeholderUrl; - } -} diff --git a/lib/extensions/page.dart b/lib/extensions/page.dart deleted file mode 100644 index 34343fb5..00000000 --- a/lib/extensions/page.dart +++ /dev/null @@ -1,61 +0,0 @@ -import 'package:spotify/spotify.dart'; - -extension CursorPageJson on CursorPage { - static CursorPage fromJson( - Map json, - T Function(dynamic json) itemFromJson, - ) { - final metadata = Paging.fromJson(json["metadata"]); - final paging = CursorPaging(); - paging.cursors = Cursor.fromJson(json["metadata"])..after = json["after"]; - paging.href = metadata.href; - paging.itemsNative = paging.itemsNative; - paging.limit = metadata.limit; - paging.next = metadata.next; - return CursorPage( - paging, - itemFromJson, - ); - } - - Map toJson() { - return { - "after": after, - "metadata": metadata.toJson(), - }; - } -} - -extension PagingToJson on Paging { - Map toJson() { - return { - "items": itemsNative, - "total": total, - "next": next, - "previous": previous, - "limit": limit, - "offset": offset, - "href": href, - }; - } -} - -extension PageJson on Page { - static Page fromJson( - Map json, - T Function(dynamic json) itemFromJson, - ) { - return Page( - Paging.fromJson( - Map.castFrom(json["metadata"]), - ), - itemFromJson, - ); - } - - Map toJson() { - return { - "metadata": metadata.toJson(), - }; - } -} diff --git a/lib/extensions/theme.dart b/lib/extensions/theme.dart deleted file mode 100644 index 22a1ce84..00000000 --- a/lib/extensions/theme.dart +++ /dev/null @@ -1,34 +0,0 @@ -import 'package:flutter/material.dart'; - -class ShimmerColorTheme extends ThemeExtension { - final Color? shimmerColor; - final Color? shimmerBackgroundColor; - - ShimmerColorTheme({ - this.shimmerBackgroundColor, - this.shimmerColor, - }); - - @override - ThemeExtension copyWith( - {Color? shimmerColor, Color? shimmerBackgroundColor}) { - return ShimmerColorTheme( - shimmerBackgroundColor: - shimmerBackgroundColor ?? this.shimmerBackgroundColor, - shimmerColor: shimmerColor ?? this.shimmerColor, - ); - } - - @override - ThemeExtension lerp( - ThemeExtension? other, double t) { - if (other is! ShimmerColorTheme) { - return this; - } - return ShimmerColorTheme( - shimmerBackgroundColor: - Color.lerp(shimmerBackgroundColor, other.shimmerBackgroundColor, t), - shimmerColor: Color.lerp(shimmerColor, other.shimmerColor, t), - ); - } -} diff --git a/lib/extensions/track.dart b/lib/extensions/track.dart deleted file mode 100644 index 215a5ab2..00000000 --- a/lib/extensions/track.dart +++ /dev/null @@ -1,98 +0,0 @@ -import 'dart:io'; -import 'dart:typed_data'; - -import 'package:metadata_god/metadata_god.dart'; -import 'package:path/path.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/services/audio_player/audio_player.dart'; - -extension TrackExtensions on Track { - Track fromFile( - File file, { - Metadata? metadata, - String? art, - }) { - album = Album() - ..name = metadata?.album ?? "Unknown" - ..images = [if (art != null) Image()..url = art] - ..genres = [if (metadata?.genre != null) metadata!.genre!] - ..artists = [ - Artist() - ..name = metadata?.albumArtist ?? "Unknown" - ..id = metadata?.albumArtist ?? "Unknown" - ..type = "artist", - ] - ..id = metadata?.album - ..releaseDate = metadata?.year?.toString(); - artists = [ - Artist() - ..name = metadata?.artist ?? "Unknown" - ..id = metadata?.artist ?? "Unknown" - ]; - - id = metadata?.title ?? basenameWithoutExtension(file.path); - name = metadata?.title ?? basenameWithoutExtension(file.path); - type = "track"; - uri = file.path; - durationMs = (metadata?.durationMs?.toInt() ?? 0); - - return this; - } - - Metadata toMetadata({ - required int fileLength, - Uint8List? imageBytes, - }) { - return Metadata( - title: name, - artist: artists?.map((a) => a.name).join(", "), - album: album?.name, - albumArtist: artists?.map((a) => a.name).join(", "), - year: album?.releaseDate != null - ? int.tryParse(album!.releaseDate!.split("-").first) ?? 1969 - : 1969, - trackNumber: trackNumber, - discNumber: discNumber, - durationMs: durationMs?.toDouble() ?? 0.0, - fileSize: BigInt.from(fileLength), - trackTotal: album?.tracks?.length ?? 0, - picture: imageBytes != null - ? Picture( - data: imageBytes, - // Spotify images are always JPEGs - mimeType: 'image/jpeg', - ) - : null, - ); - } -} - -extension TrackSimpleExtensions on TrackSimple { - Track asTrack(AlbumSimple album) { - Track track = Track(); - track.name = name; - track.album = album; - track.artists = artists; - track.availableMarkets = availableMarkets; - track.discNumber = discNumber; - track.durationMs = durationMs; - track.explicit = explicit; - track.externalUrls = externalUrls; - track.href = href; - track.id = id; - track.isPlayable = isPlayable; - track.linkedFrom = linkedFrom; - track.name = name; - track.previewUrl = previewUrl; - track.trackNumber = trackNumber; - track.type = type; - track.uri = uri; - return track; - } -} - -extension TracksToMediaExtension on Iterable { - List asMediaList() { - return map((track) => SpotubeMedia(track)).toList(); - } -} diff --git a/lib/hooks/configurators/use_check_yt_dlp_installed.dart b/lib/hooks/configurators/use_check_yt_dlp_installed.dart new file mode 100644 index 00000000..1d948258 --- /dev/null +++ b/lib/hooks/configurators/use_check_yt_dlp_installed.dart @@ -0,0 +1,40 @@ +import 'dart:io'; + +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/modules/settings/youtube_engine_not_installed_dialog.dart'; +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +import 'package:spotube/services/kv_store/kv_store.dart'; +import 'package:spotube/services/youtube_engine/yt_dlp_engine.dart'; + +void useCheckYtDlpInstalled(WidgetRef ref) { + final context = useContext(); + + useEffect(() { + WidgetsBinding.instance.addPostFrameCallback((_) async { + final youtubeEngine = ref.read( + userPreferencesProvider.select( + (value) => value.youtubeClientEngine, + ), + ); + + final customPath = + KVStoreService.getYoutubeEnginePath(YoutubeClientEngine.ytDlp); + + if (youtubeEngine == YoutubeClientEngine.ytDlp && + !await YtDlpEngine.isInstalled() && + (customPath == null || !await File(customPath).exists()) && + context.mounted) { + await showDialog( + context: context, + builder: (context) => + YouTubeEngineNotInstalledDialog(engine: youtubeEngine), + ); + } + }); + + return null; + }, []); +} diff --git a/lib/hooks/configurators/use_deep_linking.dart b/lib/hooks/configurators/use_deep_linking.dart index ec6d8516..aaa4111c 100644 --- a/lib/hooks/configurators/use_deep_linking.dart +++ b/lib/hooks/configurators/use_deep_linking.dart @@ -4,7 +4,7 @@ import 'package:app_links/app_links.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:spotube/collections/routes.dart'; -import 'package:spotube/provider/spotify_provider.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:flutter_sharing_intent/flutter_sharing_intent.dart'; import 'package:flutter_sharing_intent/model/sharing_file.dart'; import 'package:spotube/services/logger/logger.dart'; @@ -13,89 +13,95 @@ import 'package:spotube/utils/platform.dart'; final appLinks = AppLinks(); final linkStream = appLinks.stringLinkStream.asBroadcastStream(); -void useDeepLinking(WidgetRef ref) { - // single instance no worries - final spotify = ref.watch(spotifyProvider); - final router = ref.watch(routerProvider); +@Deprecated( + "Deeplinking is deprecated. Later a custom API for metadata provider will be created.") +void useDeepLinking(WidgetRef ref, AppRouter router) { + // // single instance no worries + // final spotify = ref.watch(spotifyProvider); - useEffect(() { - void uriListener(List files) async { - for (final file in files) { - if (file.type != SharedMediaType.URL) continue; - final url = Uri.parse(file.value!); - if (url.pathSegments.length != 2) continue; + // useEffect(() { + // void uriListener(List files) async { + // for (final file in files) { + // if (file.type != SharedMediaType.URL) continue; + // final url = Uri.parse(file.value!); + // if (url.pathSegments.length != 2) continue; - switch (url.pathSegments.first) { - case "album": - router.push( - "/album/${url.pathSegments.last}", - extra: await spotify.albums.get(url.pathSegments.last), - ); - break; - case "artist": - router.push("/artist/${url.pathSegments.last}"); - break; - case "playlist": - router.push( - "/playlist/${url.pathSegments.last}", - extra: await spotify.playlists.get(url.pathSegments.last), - ); - break; - case "track": - router.push( - "/track/${url.pathSegments.last}", - ); - break; - default: - break; - } - } - } + // switch (url.pathSegments.first) { + // case "album": + // final album = await spotify.invoke((api) { + // return api.albums.get(url.pathSegments.last); + // }); + // // router.navigate( + // // AlbumRoute(id: album.id!, album: album), + // // ); + // break; + // case "artist": + // router.navigate(ArtistRoute(artistId: url.pathSegments.last)); + // break; + // case "playlist": + // final playlist = await spotify.invoke((api) { + // return api.playlists.get(url.pathSegments.last); + // }); + // // router + // // .navigate(PlaylistRoute(id: playlist.id!, playlist: playlist)); + // break; + // case "track": + // router.navigate(TrackRoute(trackId: url.pathSegments.last)); + // break; + // default: + // break; + // } + // } + // } - StreamSubscription? mediaStream; + // StreamSubscription? mediaStream; - if (kIsMobile) { - FlutterSharingIntent.instance.getInitialSharing().then(uriListener); + // if (kIsMobile) { + // FlutterSharingIntent.instance.getInitialSharing().then(uriListener); - mediaStream = - FlutterSharingIntent.instance.getMediaStream().listen(uriListener); - } + // mediaStream = + // FlutterSharingIntent.instance.getMediaStream().listen(uriListener); + // } - final subscription = linkStream.listen((uri) async { - try { - final startSegment = uri.split(":").take(2).join(":"); - final endSegment = uri.split(":").last; + // final subscription = linkStream.listen((uri) async { + // try { + // final startSegment = uri.split(":").take(2).join(":"); + // final endSegment = uri.split(":").last; - switch (startSegment) { - case "spotify:album": - await router.push( - "/album/$endSegment", - extra: await spotify.albums.get(endSegment), - ); - break; - case "spotify:artist": - await router.push("/artist/$endSegment"); - break; - case "spotify:track": - await router.push("/track/$endSegment"); - break; - case "spotify:playlist": - await router.push( - "/playlist/$endSegment", - extra: await spotify.playlists.get(endSegment), - ); - break; - default: - break; - } - } catch (e, stack) { - AppLogger.reportError(e, stack); - } - }); + // switch (startSegment) { + // case "spotify:album": + // final album = await spotify.invoke((api) { + // return api.albums.get(endSegment); + // }); + // // await router.navigate( + // // AlbumRoute(id: album.id!, album: album), + // // ); + // break; + // case "spotify:artist": + // await router.navigate(ArtistRoute(artistId: endSegment)); + // break; + // case "spotify:track": + // await router.navigate(TrackRoute(trackId: endSegment)); + // break; + // case "spotify:playlist": + // final playlist = await spotify.invoke((api) { + // return api.playlists.get(endSegment); + // }); + // // await router.navigate( + // // PlaylistRoute(id: playlist.id!, playlist: playlist), + // // ); + // break; + // default: + // break; + // } + // } catch (e, stack) { + // AppLogger.reportError(e, stack); + // } + // }); - return () { - mediaStream?.cancel(); - subscription.cancel(); - }; - }, [spotify]); + // return () { + // mediaStream?.cancel(); + // subscription.cancel(); + // }; + // }, [spotify]); } diff --git a/lib/hooks/configurators/use_endless_playback.dart b/lib/hooks/configurators/use_endless_playback.dart index e2fb1e6e..9e8c191e 100644 --- a/lib/hooks/configurators/use_endless_playback.dart +++ b/lib/hooks/configurators/use_endless_playback.dart @@ -1,24 +1,21 @@ +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; import 'package:spotube/services/logger/logger.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; void useEndlessPlayback(WidgetRef ref) { - final auth = ref.watch(authenticationProvider); final playback = ref.watch(audioPlayerProvider.notifier); - final playlist = ref.watch(audioPlayerProvider.select((s) => s.playlist)); - final spotify = ref.watch(spotifyProvider); + final audioPlayerState = ref.watch(audioPlayerProvider); final endlessPlayback = ref.watch(userPreferencesProvider.select((s) => s.endlessPlayback)); + final metadataPlugin = ref.watch(metadataPluginProvider.future); useEffect( () { - if (!endlessPlayback || auth.asData?.value == null) return null; + if (!endlessPlayback) return null; void listener(int index) async { try { @@ -27,31 +24,9 @@ void useEndlessPlayback(WidgetRef ref) { final track = playlist.tracks.last; - final query = "${track.name} Radio"; - final pages = await spotify.search - .get(query, types: [SearchType.playlist]).first(); + final tracks = await (await metadataPlugin)?.track.radio(track.id); - final radios = pages - .expand((e) => e.items?.toList() ?? []) - .toList() - .cast(); - - final artists = track.artists!.map((e) => e.name); - - final radio = radios.firstWhere( - (e) { - final validPlaylists = - artists.where((a) => e.description!.contains(a!)); - return e.name == "${track.name} Radio" && - (validPlaylists.length >= 2 || - validPlaylists.length == artists.length) && - e.owner?.displayName != "Spotify"; - }, - orElse: () => radios.first, - ); - - final tracks = - await spotify.playlists.getTracksByPlaylistId(radio.id!).all(); + if (tracks == null || tracks.isEmpty) return; await playback.addTracks( tracks.toList() @@ -69,9 +44,9 @@ void useEndlessPlayback(WidgetRef ref) { // Sometimes user can change settings for which the currentIndexChanged // might not be called. So we need to check if the current track is the // last track and if it is then we need to call the listener manually. - if (playlist.index == playlist.medias.length - 1 && + if (audioPlayerState.currentIndex == audioPlayerState.tracks.length - 1 && audioPlayer.isPlaying) { - listener(playlist.index); + listener(audioPlayerState.currentIndex); } final subscription = @@ -80,11 +55,11 @@ void useEndlessPlayback(WidgetRef ref) { return subscription.cancel; }, [ - spotify, + metadataPlugin, playback, - playlist.medias, + audioPlayerState.tracks, + audioPlayerState.currentIndex, endlessPlayback, - auth, ], ); } diff --git a/lib/hooks/configurators/use_fix_window_stretching.dart b/lib/hooks/configurators/use_fix_window_stretching.dart index a6603d59..b94098ab 100644 --- a/lib/hooks/configurators/use_fix_window_stretching.dart +++ b/lib/hooks/configurators/use_fix_window_stretching.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:spotube/utils/platform.dart'; import 'package:window_manager/window_manager.dart'; diff --git a/lib/hooks/configurators/use_has_touch.dart b/lib/hooks/configurators/use_has_touch.dart index 75353f27..5ce309b8 100644 --- a/lib/hooks/configurators/use_has_touch.dart +++ b/lib/hooks/configurators/use_has_touch.dart @@ -1,5 +1,5 @@ import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:spotube/utils/platform.dart'; diff --git a/lib/hooks/controllers/use_auto_scroll_controller.dart b/lib/hooks/controllers/use_auto_scroll_controller.dart index 0c7119e4..befc4351 100644 --- a/lib/hooks/controllers/use_auto_scroll_controller.dart +++ b/lib/hooks/controllers/use_auto_scroll_controller.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:scroll_to_index/scroll_to_index.dart'; diff --git a/lib/hooks/controllers/use_package_info.dart b/lib/hooks/controllers/use_package_info.dart index b3c05665..07b53af6 100644 --- a/lib/hooks/controllers/use_package_info.dart +++ b/lib/hooks/controllers/use_package_info.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:package_info_plus/package_info_plus.dart'; diff --git a/lib/hooks/controllers/use_shadcn_text_editing_controller.dart b/lib/hooks/controllers/use_shadcn_text_editing_controller.dart new file mode 100644 index 00000000..ae33f4e4 --- /dev/null +++ b/lib/hooks/controllers/use_shadcn_text_editing_controller.dart @@ -0,0 +1,97 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +class _TextEditingControllerHookCreator { + const _TextEditingControllerHookCreator(); + + /// Creates a [TextEditingController] that will be disposed automatically. + /// + /// The [text] parameter can be used to set the initial value of the + /// controller. + TextEditingController call({String? text, List? keys}) { + return use(_TextEditingControllerHook(text, keys)); + } + + /// Creates a [TextEditingController] from the initial [value] that will + /// be disposed automatically. + TextEditingController fromValue( + TextEditingValue value, [ + List? keys, + ]) { + return use(_TextEditingControllerHook.fromValue(value, keys)); + } +} + +/// Creates a [TextEditingController], either via an initial text or an initial +/// [TextEditingValue]. +/// +/// To use a [TextEditingController] with an optional initial text, use: +/// ```dart +/// final controller = useTextEditingController(text: 'initial text'); +/// ``` +/// +/// To use a [TextEditingController] with an optional initial value, use: +/// ```dart +/// final controller = useTextEditingController +/// .fromValue(TextEditingValue.empty); +/// ``` +/// +/// Changing the text or initial value after the widget has been built has no +/// effect whatsoever. To update the value in a callback, for instance after a +/// button was pressed, use the [TextEditingController.text] or +/// [TextEditingController.value] setters. To have the [TextEditingController] +/// reflect changing values, you can use [useEffect]. This example will update +/// the [TextEditingController.text] whenever a provided [ValueListenable] +/// changes: +/// ```dart +/// final controller = useTextEditingController(); +/// final update = useValueListenable(myTextControllerUpdates); +/// +/// useEffect(() { +/// controller.text = update; +/// }, [update]); +/// ``` +/// +/// See also: +/// - [TextEditingController], which this hook creates. +const useShadcnTextEditingController = _TextEditingControllerHookCreator(); + +class _TextEditingControllerHook extends Hook { + const _TextEditingControllerHook( + this.initialText, [ + List? keys, + ]) : initialValue = null, + super(keys: keys); + + const _TextEditingControllerHook.fromValue( + TextEditingValue this.initialValue, [ + List? keys, + ]) : initialText = null, + super(keys: keys); + + final String? initialText; + final TextEditingValue? initialValue; + + @override + _TextEditingControllerHookState createState() { + return _TextEditingControllerHookState(); + } +} + +class _TextEditingControllerHookState + extends HookState { + late final _controller = hook.initialValue != null + ? TextEditingController.fromValue( + hook.initialValue ?? TextEditingValue.empty, + ) + : TextEditingController(text: hook.initialText); + + @override + TextEditingController build(BuildContext context) => _controller; + + @override + void dispose() => _controller.dispose(); + + @override + String get debugLabel => 'useTextEditingController'; +} diff --git a/lib/hooks/controllers/use_sidebarx_controller.dart b/lib/hooks/controllers/use_sidebarx_controller.dart deleted file mode 100644 index a14c3305..00000000 --- a/lib/hooks/controllers/use_sidebarx_controller.dart +++ /dev/null @@ -1,59 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:sidebarx/sidebarx.dart'; - -/// Creates [SidebarXController] that will be disposed automatically. -/// -/// See also: -/// - [SidebarXController] -SidebarXController useSidebarXController({ - required int selectedIndex, - bool? extended, - List? keys, -}) { - return use( - _SidebarXControllerHook( - selectedIndex: selectedIndex, - extended: extended, - keys: keys, - ), - ); -} - -class _SidebarXControllerHook extends Hook { - const _SidebarXControllerHook({ - required this.selectedIndex, - this.extended, - super.keys, - }); - - final int selectedIndex; - final bool? extended; - - @override - HookState> createState() => - _SidebarXControllerHookState(); -} - -class _SidebarXControllerHookState - extends HookState { - late final SidebarXController controller; - - @override - void initHook() { - super.initHook(); - controller = SidebarXController( - selectedIndex: hook.selectedIndex, - extended: hook.extended, - ); - } - - @override - SidebarXController build(BuildContext context) => controller; - - @override - void dispose() => controller.dispose(); - - @override - String get debugLabel => 'useSidebarXController'; -} diff --git a/lib/hooks/utils/use_breakpoint_value.dart b/lib/hooks/utils/use_breakpoint_value.dart index b2592124..74b2f860 100644 --- a/lib/hooks/utils/use_breakpoint_value.dart +++ b/lib/hooks/utils/use_breakpoint_value.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:spotube/extensions/constrains.dart'; diff --git a/lib/hooks/utils/use_brightness_value.dart b/lib/hooks/utils/use_brightness_value.dart index d3823b2f..64e3f27c 100644 --- a/lib/hooks/utils/use_brightness_value.dart +++ b/lib/hooks/utils/use_brightness_value.dart @@ -1,4 +1,5 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; T useBrightnessValue( diff --git a/lib/hooks/utils/use_custom_status_bar_color.dart b/lib/hooks/utils/use_custom_status_bar_color.dart index 8afc6a59..f34ae7a8 100644 --- a/lib/hooks/utils/use_custom_status_bar_color.dart +++ b/lib/hooks/utils/use_custom_status_bar_color.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter/services.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; @@ -9,7 +9,7 @@ VoidCallback useCustomStatusBarColor( bool? automaticSystemUiAdjustment, }) { final context = useContext(); - final backgroundColor = Theme.of(context).scaffoldBackgroundColor; + final backgroundColor = Theme.of(context).colorScheme.background; // ignore: invalid_use_of_visible_for_testing_member final previousState = SystemChrome.latestStyle; diff --git a/lib/hooks/utils/use_palette_color.dart b/lib/hooks/utils/use_palette_color.dart index 64994d2b..c70bcf72 100644 --- a/lib/hooks/utils/use_palette_color.dart +++ b/lib/hooks/utils/use_palette_color.dart @@ -1,4 +1,5 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:palette_generator/palette_generator.dart'; @@ -6,7 +7,7 @@ import 'package:spotube/components/image/universal_image.dart'; final _paletteColorState = StateProvider( (ref) { - return PaletteColor(Colors.grey[300]!, 0); + return PaletteColor(Colors.gray[300], 0); }, ); diff --git a/lib/l10n/app_ar.arb b/lib/l10n/app_ar.arb index 8cb52b38..f1997517 100644 --- a/lib/l10n/app_ar.arb +++ b/lib/l10n/app_ar.arb @@ -401,5 +401,94 @@ "export_cache_files": "تصدير الملفات المخزنة مؤقتًا", "found_n_files": "تم العثور على {count} ملف", "export_cache_confirmation": "هل تريد تصدير هذه الملفات إلى", - "exported_n_out_of_m_files": "تم تصدير {filesExported} من أصل {files} ملفات" + "exported_n_out_of_m_files": "تم تصدير {filesExported} من أصل {files} ملفات", + "playlist": "قائمة التشغيل", + "no_loop": "بدون تكرار", + "generate": "إنشاء", + "undo": "تراجع", + "download_all": "تنزيل الكل", + "add_all_to_playlist": "إضافة الكل إلى قائمة التشغيل", + "add_all_to_queue": "إضافة الكل إلى القائمة", + "play_all_next": "تشغيل الكل بعد ذلك", + "pause": "إيقاف مؤقت", + "view_all": "عرض الكل", + "no_tracks_added_yet": "يبدو أنك لم تضف أي مسارات بعد", + "no_tracks": "يبدو أنه لا يوجد أي مسارات هنا", + "no_tracks_listened_yet": "يبدو أنك لم تستمع إلى أي شيء بعد", + "not_following_artists": "أنت لا تتابع أي فنانين", + "no_favorite_albums_yet": "يبدو أنك لم تضف أي ألبومات إلى المفضلة بعد", + "no_logs_found": "لم يتم العثور على سجلات", + "youtube_engine": "محرك يوتيوب", + "youtube_engine_not_installed_title": "{engine} غير مثبت", + "youtube_engine_not_installed_message": "{engine} غير مثبت في نظامك.", + "youtube_engine_set_path": "تأكد من أنه متاح في متغير PATH أو\nحدد المسار الكامل للملف القابل للتنفيذ {engine} أدناه", + "youtube_engine_unix_issue_message": "في أنظمة macOS/Linux/Unix مثل الأنظمة، لن يعمل تعيين المسار في .zshrc/.bashrc/.bash_profile وما إلى ذلك.\nيجب تعيين المسار في ملف تكوين الصدفة", + "download": "تنزيل", + "file_not_found": "الملف غير موجود", + "custom": "مخصص", + "add_custom_url": "إضافة URL مخصص", + "edit_port": "تعديل المنفذ", + "port_helper_msg": "القيمة الافتراضية هي -1 والتي تشير إلى رقم عشوائي. إذا كان لديك جدار ناري مُعد، يُوصى بتعيين هذا.", + "connect_request": "السماح لـ {client} بالاتصال؟", + "connection_request_denied": "تم رفض الاتصال. المستخدم رفض الوصول.", + "hipotetical_calculation": "*تمّ الحساب بمعدّل دفعة تتراوح بين 0.003–0.005 دولار أمريكي لكل تشغيل على منصات الموسيقى عبر الإنترنت. هذا حساب افتراضي لتوضيح للمستخدم مقدار ما كان سيدفعه للفنانين لو استمع إلى أغنيتهم على منصات مختلفة.", + "an_error_occurred": "حدث خطأ", + "copy_to_clipboard": "نسخ إلى الحافظة", + "view_logs": "عرض السجلات", + "retry": "إعادة المحاولة", + "no_default_metadata_provider_selected": "لم تقُم بتعيين مزود بيانات افتراضي", + "manage_metadata_providers": "إدارة مزوّدي البيانات", + "open_link_in_browser": "فتح الرابط في المتصفح؟", + "do_you_want_to_open_the_following_link": "هل ترغب في فتح الرابط التالي؟", + "unsafe_url_warning": "قد يكون فتح الروابط من مصادر غير موثوقة غير آمن. تحرّ الحذر!\nيمكنك أيضًا نسخ الرابط إلى الحافظة.", + "copy_link": "نسخ الرابط", + "building_your_timeline": "جاري بناء المخطط الزمني استنادًا إلى استماعاتك...", + "official": "رسمي", + "author_name": "المؤلّف: {author}", + "third_party": "طرف ثالث", + "plugin_requires_authentication": "تتطلّب الإضافة تسجيل الدخول", + "update_available": "تحديث متوفر", + "supports_scrobbling": "يدعم التتبع (scrobbling)", + "plugin_scrobbling_info": "تقوم هذه الإضافة بتتبع مقاطعك الموسيقية لإنشاء سجل الاستماع الخاص بك.", + "default_plugin": "الافتراضي", + "set_default": "تعيين كافتراضي", + "support": "الدعم", + "support_plugin_development": "دعم تطوير الإضافات", + "can_access_name_api": "- يمكن الوصول إلى واجهة برمجة التطبيقات **{name}**", + "do_you_want_to_install_this_plugin": "هل ترغب في تثبيت هذه الإضافة؟", + "third_party_plugin_warning": "هذه الإضافة من مستودع طرف ثالث. تأكد من موثوقية المصدر قبل التثبيت.", + "author": "المؤلف", + "this_plugin_can_do_following": "يمكن لهذه الإضافة القيام بما يلي", + "install": "تثبيت", + "install_a_metadata_provider": "تثبيت مزوّد بيانات", + "no_tracks_playing": "لا توجد مقاطع تعمل حاليًا", + "synced_lyrics_not_available": "الكلمات المتزامنة غير متوفرة لهذه الأغنية. يُرجى استخدام", + "plain_lyrics": "الكلمات العادية", + "tab_instead": "بدلاً من ذلك، استخدم التبويب.", + "disclaimer": "إخلاء المسؤولية", + "third_party_plugin_dmca_notice": "لا تتحمّل فريق Spotube أي مسؤولية (بما في ذلك القانونية) عن أي من الإضافات “لطرف ثالث”.\nاستخدمها على مسؤوليتك الخاصّة. لأيّة أخطاء/مشكلات، يُرجى الإبلاغ عنها في مستودع الإضافة.\n\nإذا كانت أي إضافة “لطرف ثالث” تنتهك شروط الخدمة أو قانون DMCA الخاص بأي خدمة أو كيان قانوني، فيُرجى طلب اتخاذ إجراء من مؤلف الإضافة أو منصة الاستضافة مثل GitHub/Codeberg. الإضافات المدرجة كـ “لطرف ثالث” هي مفعّلة ومُدارة من المجتمع، وليس لدينا صلاحية إدارتها أو التدخل فيها.\n\n", + "input_does_not_match_format": "المدخل لا يتوافق مع التنسيق المطلوب", + "metadata_provider_plugins": "إضافات مزود البيانات", + "paste_plugin_download_url": "الصق رابط التنزيل أو GitHub/Codeberg أو رابط مباشر لملف .smplug", + "download_and_install_plugin_from_url": "تنزيل وتثبيت الإضافة من رابط", + "failed_to_add_plugin_error": "فشل في إضافة الإضافة: {error}", + "upload_plugin_from_file": "رفع الإضافة من ملف", + "installed": "تم التثبيت", + "available_plugins": "الإضافات المتوفّرة", + "configure_your_own_metadata_plugin": "تهيئة مزوّد بيانات للقائمة/الألبوم/الفنان/المصدر خاص بك", + "audio_scrobblers": "أجهزة تتبع الصوت", + "scrobbling": "التتبع", + "download_music_format": "تنسيق تنزيل الموسيقى", + "streaming_music_format": "تنسيق بث الموسيقى", + "download_music_quality": "جودة تنزيل الموسيقى", + "streaming_music_quality": "جودة بث الموسيقى", + "default_metadata_source": "مصدر البيانات الوصفية الافتراضي", + "set_default_metadata_source": "تعيين مصدر البيانات الوصفية الافتراضي", + "default_audio_source": "مصدر الصوت الافتراضي", + "set_default_audio_source": "تعيين مصدر الصوت الافتراضي", + "plugins": "الإضافات", + "configure_plugins": "قم بتكوين مزود البيانات الوصفية ومكونات مصدر الصوت الخاصة بك", + "source": "المصدر: ", + "uncompressed": "غير مضغوط", + "dab_music_source_description": "لمحبي الصوتيات. يوفر تدفقات صوتية عالية الجودة/بدون فقدان. مطابقة دقيقة للمسارات بناءً على ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_bn.arb b/lib/l10n/app_bn.arb index ff49aafd..4d001da1 100644 --- a/lib/l10n/app_bn.arb +++ b/lib/l10n/app_bn.arb @@ -401,5 +401,94 @@ "export_cache_files": "ক্যাশে ফাইল রপ্তানি", "found_n_files": "{count} টি ফাইল পাওয়া গেছে", "export_cache_confirmation": "আপনি কি এই ফাইলগুলি রপ্তানি করতে চান", - "exported_n_out_of_m_files": "{filesExported} টি ফাইল রপ্তানি করা হয়েছে {files} এর মধ্যে" + "exported_n_out_of_m_files": "{filesExported} টি ফাইল রপ্তানি করা হয়েছে {files} এর মধ্যে", + "playlist": "প্লেলিস্ট", + "no_loop": "কোনো লুপ নেই", + "generate": "উৎপন্ন করুন", + "undo": "পূর্বাবস্থায় ফিরুন", + "download_all": "সব ডাউনলোড করুন", + "add_all_to_playlist": "সব প্লেলিস্টে যোগ করুন", + "add_all_to_queue": "সব কিউতে যোগ করুন", + "play_all_next": "সব পরবর্তী খেলুন", + "pause": "বিরতি", + "view_all": "সব দেখুন", + "no_tracks_added_yet": "এখনও কোনো ট্র্যাক যোগ করা হয়নি মনে হচ্ছে", + "no_tracks": "এখানে কোনো ট্র্যাক নেই মনে হচ্ছে", + "no_tracks_listened_yet": "এখনও কিছু শোনা হয়নি মনে হচ্ছে", + "not_following_artists": "আপনি কোনো শিল্পীকে অনুসরণ করছেন না", + "no_favorite_albums_yet": "এখনও কোনো অ্যালবাম প্রিয় তালিকায় যোগ করা হয়নি মনে হচ্ছে", + "no_logs_found": "কোনো লগ পাওয়া যায়নি", + "youtube_engine": "ইউটিউব ইঞ্জিন", + "youtube_engine_not_installed_title": "{engine} ইনস্টল করা নেই", + "youtube_engine_not_installed_message": "{engine} আপনার সিস্টেমে ইনস্টল করা নেই।", + "youtube_engine_set_path": "এটি PATH ভেরিয়েবলে উপলব্ধ কিনা নিশ্চিত করুন অথবা\nনীচে {engine} এক্সিকিউটেবল এর পূর্ণপথ সেট করুন", + "youtube_engine_unix_issue_message": "macOS/Linux/Unix-এর মতো অপারেটিং সিস্টেমে, .zshrc/.bashrc/.bash_profile ইত্যাদিতে পাথ সেট করা কাজ করবে না।\nআপনাকে শেল কনফিগারেশন ফাইলে পাথ সেট করতে হবে", + "download": "ডাউনলোড", + "file_not_found": "ফাইল পাওয়া যায়নি", + "custom": "কাস্টম", + "add_custom_url": "কাস্টম URL যোগ করুন", + "edit_port": "পোর্ট সম্পাদনা করুন", + "port_helper_msg": "ডিফল্ট হল -1 যা এলোমেলো সংখ্যা নির্দেশ করে। যদি আপনার ফায়ারওয়াল কনফিগার করা থাকে, তবে এটি সেট করা সুপারিশ করা হয়।", + "connect_request": "{client} কে সংযোগ করতে অনুমতি দেবেন?", + "connection_request_denied": "সংযোগ অস্বীকৃত। ব্যবহারকারী প্রবেশাধিকার অস্বীকার করেছে।", + "hipotetical_calculation": "*এটি নিরূপণ করা হয়েছে গড় অনলাইন মিউজিক স্ট্রিমিং প্ল্যাটফর্মের প্রতি স্ট্রিম 0.003–0.005 USD পেআউটের ভিত্তিতে। এটি একটি কাল্পনিক হিসাব যা ব্যবহারকারীকে ধারণা দিতে পারে তারা অন্যান্য স্ট্রিমিং প্ল্যাটফর্মে একই গান শোনার জন্য শিল্পীদের কত টাকা দিয়েছেন হোক।", + "an_error_occurred": "একটি ত্রুটি ঘটেছে", + "copy_to_clipboard": "ক্লিপবোর্ডে কপি করুন", + "view_logs": "লগ দেখুন", + "retry": "পুনরায় চেষ্টা করুন", + "no_default_metadata_provider_selected": "আপনি কোনো ডিফল্ট মেটাডেটা প্রদানকারী সেট করেননি", + "manage_metadata_providers": "মেটাডেটা প্রদানকারীগণ পরিচালনা করুন", + "open_link_in_browser": "লিংকটি ব্রাউজারে খুলবেন?", + "do_you_want_to_open_the_following_link": "নিচের লিংকটি খুলতে চান?", + "unsafe_url_warning": "অবিশ্বাসযোগ্য উৎস থেকে লিংক খোলা নিরাপদ নাও হতে পারে। সতর্ক থাকুন!\nআপনি এটি ক্লিপবোর্ডে কপি করতে পারেন।", + "copy_link": "লিংক কপি করুন", + "building_your_timeline": "আপনার শোনার ধারা অনুযায়ী টাইমলাইন তৈরি করা হচ্ছে...", + "official": "সরকারি", + "author_name": "লেখক: {author}", + "third_party": "তৃতীয় পক্ষ", + "plugin_requires_authentication": "প্লাগইনটি প্রমাণীকরণ প্রয়োজন", + "update_available": "হালনাগাদ উপলব্ধ", + "supports_scrobbling": "স্ক্রোব্বলিং সমর্থিত", + "plugin_scrobbling_info": "এই প্লাগইনটি আপনার সঙ্গীত স্ক্রোব্বল করে আপনার শোনা ইতিহাস তৈরি করে।", + "default_plugin": "ডিফল্ট", + "set_default": "ডিফল্ট হিসাবে নির্ধারণ করুন", + "support": "সমর্থন", + "support_plugin_development": "প্লাগইন উন্নয়নকে সমর্থন করুন", + "can_access_name_api": "- **{name}** API-তে অ্যাক্সেস করতে পারে", + "do_you_want_to_install_this_plugin": "আপনি কি এই প্লাগইন ইনস্টল করতে চান?", + "third_party_plugin_warning": "এই প্লাগইন একটি তৃতীয় পক্ষের রেপোজিটরির। ইনস্টল করার আগে উৎস বিশ্বস্ত কিনা নিশ্চিত করুন।", + "author": "লেখক", + "this_plugin_can_do_following": "এই প্লাগইন নিচের কাজ করতে পারে", + "install": "ইনস্টল করুন", + "install_a_metadata_provider": "একটি মেটাডেটা প্রদানকারী ইনস্টল করুন", + "no_tracks_playing": "বর্তমানে কোনো ট্র্যাক শোনা হচ্ছে না", + "synced_lyrics_not_available": "এই গানের জন্য সিঙ্ক্রোনাইজড লিরিক্স পাওয়া যায় না। অনুগ্রহ করে ব্যবহার করুন", + "plain_lyrics": "সহজ লিরিক্স", + "tab_instead": "তার পরিবর্তে ট্যাব ব্যবহার করুন।", + "disclaimer": "অস্বীকৃতি", + "third_party_plugin_dmca_notice": "Spotube দল কোনো “তৃতীয় পক্ষ” প্লাগইনের জন্য কোনো (আইনগত সহ) দায়িত্ব নেয় না। নিজের বিপদে ব্যবহার করুন। কোনো বাগ/সমস্যা হলে প্লাগইন রেপোজিটরিতে জানাতে অনুরোধ করা হচ্ছে।\n\nযদি কোনো “তৃতীয় পক্ষ” প্লাগইন কোনো পরিষেবা/আইনগত সংস্থার ToS/DMCA ভূঙ্গ করে, অনুগ্রহ করে “তৃতীয় পক্ষ” প্লাগইনের লেখক বা হোস্টিং প্ল্যাটফর্মে (যেমন GitHub/Codeberg) পদক্ষেপ নিতে বলুন। “তৃতীয় পক্ষ” লেবেলযুক্ত যুক্তিগুলি সকলই পাবলিক/কমিউনিটি দ্বারা রক্ষণাবেক্ষণ করা হয়; আমরা সেগুলি কিউরেট করি না, তাই আমরা কোনো পদক্ষেপ নিতে পারি না।\n\n", + "input_does_not_match_format": "ইনপুট প্রয়োজনীয় ফরম্যাটের সাথে মেলে না", + "metadata_provider_plugins": "মেটাডেটা প্রদানকারী প্লাগইনসমূহ", + "paste_plugin_download_url": "ডাউনলোড URL বা GitHub/Codeberg রিপো URL বা .smplug ফাইলের সরাসরি লিঙ্ক পেস্ট করুন", + "download_and_install_plugin_from_url": "URL থেকে প্লাগইন ডাউনলোড এবং ইনস্টল করুন", + "failed_to_add_plugin_error": "প্লাগইন যোগ করতে ব্যর্থ: {error}", + "upload_plugin_from_file": "ফাইল থেকে প্লাগইন আপলোড করুন", + "installed": "ইনস্টল করা হয়েছে", + "available_plugins": "উপলব্ধ প্লাগইনগুলো", + "configure_your_own_metadata_plugin": "নিজস্ব প্লেলিস্ট/অ্যালবাম/শিল্পী/ফিড মেটাডেটা প্রদানকারী কনফিগার করুন", + "audio_scrobblers": "অডিও স্ক্রোব্বলার্স", + "scrobbling": "স্ক্রোব্বলিং", + "download_music_format": "গান ডাউনলোডের বিন্যাস", + "streaming_music_format": "গান স্ট্রিমিং এর বিন্যাস", + "download_music_quality": "গান ডাউনলোডের মান", + "streaming_music_quality": "গান স্ট্রিমিং এর মান", + "default_metadata_source": "ডিফল্ট মেটাডেটা উৎস", + "set_default_metadata_source": "ডিফল্ট মেটাডেটা উৎস সেট করুন", + "default_audio_source": "ডিফল্ট অডিও উৎস", + "set_default_audio_source": "ডিফল্ট অডিও উৎস সেট করুন", + "plugins": "প্লাগইন", + "configure_plugins": "আপনার নিজের মেটাডেটা প্রদানকারী এবং অডিও উৎস প্লাগইন কনফিগার করুন", + "source": "উৎস: ", + "uncompressed": "অ-সংকুচিত", + "dab_music_source_description": "অডিওফাইলদের জন্য। উচ্চ-মানের/লসলেস অডিও স্ট্রিম প্রদান করে। সঠিক ISRC ভিত্তিক ট্র্যাক ম্যাচিং।" } \ No newline at end of file diff --git a/lib/l10n/app_ca.arb b/lib/l10n/app_ca.arb index aee39ffd..06ed7ec6 100644 --- a/lib/l10n/app_ca.arb +++ b/lib/l10n/app_ca.arb @@ -401,5 +401,94 @@ "export_cache_files": "Exportar arxius en caché", "found_n_files": "S'han trobat {count} arxius", "export_cache_confirmation": "Voleu exportar aquests arxius a", - "exported_n_out_of_m_files": "S'han exportat {filesExported} de {files} arxius" + "exported_n_out_of_m_files": "S'han exportat {filesExported} de {files} arxius", + "playlist": "Llista de reproducció", + "no_loop": "Sense repetició", + "generate": "Generar", + "undo": "Desfer", + "download_all": "Descarregar tot", + "add_all_to_playlist": "Afegir tot a la llista de reproducció", + "add_all_to_queue": "Afegir tot a la cua", + "play_all_next": "Reproduir tot a continuació", + "pause": "Pausa", + "view_all": "Veure tot", + "no_tracks_added_yet": "Sembla que encara no has afegit cap pista", + "no_tracks": "Sembla que no hi ha pistes aquí", + "no_tracks_listened_yet": "Sembla que no has escoltat res encara", + "not_following_artists": "No estàs seguint cap artista", + "no_favorite_albums_yet": "Sembla que encara no has afegit cap àlbum als teus favorits", + "no_logs_found": "No s'han trobat registres", + "youtube_engine": "Motor de YouTube", + "youtube_engine_not_installed_title": "{engine} no està instal·lat", + "youtube_engine_not_installed_message": "{engine} no està instal·lat al teu sistema.", + "youtube_engine_set_path": "Assegura't que estigui disponible a la variable PATH o\nestableix el camí absolut a l'executable de {engine} a continuació", + "youtube_engine_unix_issue_message": "En macOS/Linux/Unix com a sistemes operatius, establir el camí a .zshrc/.bashrc/.bash_profile etc. no funcionarà.\nHas de configurar el camí al fitxer de configuració de la shell", + "download": "Descarregar", + "file_not_found": "Fitxer no trobat", + "custom": "Personalitzat", + "add_custom_url": "Afegir URL personalitzada", + "edit_port": "Editar port", + "port_helper_msg": "El valor per defecte és -1, que indica un número aleatori. Si teniu un tallafoc configurat, es recomana establir-ho.", + "connect_request": "Permetre que {client} es connecti?", + "connection_request_denied": "Connexió denegada. L'usuari ha denegat l'accés.", + "hipotetical_calculation": "*Això està calculat en funció d’un pagament mitjà per reproducció de 0,003–0,005 USD en plataformes de reproducció musical en línia. És un càlcul hipotètic per ajudar l’usuari a entendre quant hauria pagat als artistes si hagués escoltat la seva cançó en diferents plataformes.", + "an_error_occurred": "S’ha produït un error", + "copy_to_clipboard": "Copiar al porta-retalls", + "view_logs": "Veure registres", + "retry": "Tornar-ho a provar", + "no_default_metadata_provider_selected": "No has configurat cap proveïdor de metadades predeterminat", + "manage_metadata_providers": "Gestionar proveïdors de metadades", + "open_link_in_browser": "Obrir l’enllaç en el navegador?", + "do_you_want_to_open_the_following_link": "Vols obrir l’enllaç següent?", + "unsafe_url_warning": "Pot ser perillós obrir enllaços de fonts no fiables. Sigues precavís!\nTambé pots copiar l’enllaç al porta-retalls.", + "copy_link": "Copiar enllaç", + "building_your_timeline": "Construint la teva cronologia en funció de les teves escoltes...", + "official": "Oficial", + "author_name": "Autor: {author}", + "third_party": "Tercers", + "plugin_requires_authentication": "El complement requereix autenticació", + "update_available": "Actualització disponible", + "supports_scrobbling": "Admet scrobbling", + "plugin_scrobbling_info": "Aquest complement fa scrobbling de la teva música per generar l’historial d’escoltes.", + "default_plugin": "Predeterminat", + "set_default": "Establir com a predeterminat", + "support": "Suport", + "support_plugin_development": "Suportar el desenvolupament del complement", + "can_access_name_api": "- Pot accedir a l’API **{name}**", + "do_you_want_to_install_this_plugin": "Vols instal·lar aquest complement?", + "third_party_plugin_warning": "Aquest complement prové d’un repositori de tercers. Assegura’t de confiar en la font abans d’instal·lar-lo.", + "author": "Autor", + "this_plugin_can_do_following": "Aquest complement pot fer el següent", + "install": "Instal·lar", + "install_a_metadata_provider": "Instal·lar un proveïdor de metadades", + "no_tracks_playing": "No s’està reproduint cap pista actualment", + "synced_lyrics_not_available": "Les lletres sincronitzades no estan disponibles per a aquesta cançó. Si us plau, usa", + "plain_lyrics": "Lletres sense format", + "tab_instead": "en lloc d’això, utilitza la tecla Tab.", + "disclaimer": "Avís legal", + "third_party_plugin_dmca_notice": "L’equip de Spotube no accepta cap responsabilitat (inclosa legal) pels complements de “tercers”.\nFes-los servir sota la teva responsabilitat. Si detectes errors/problemes, informa’ls al repositori del complement.\n\nSi algun complement de “tercers” incompleix els ToS/DMCA d’un servei o entitat legal, contacta amb l’autor del complement o amb la plataforma d’allotjament (per exemple GitHub/Codeberg) per prendre mesures. Els complements etiquetats com a “tercers” són públics i gestionats per la comunitat; no els curatem, per la qual cosa no podem intervenir-hi.\n\n", + "input_does_not_match_format": "L’entrada no coincideix amb el format requerit", + "metadata_provider_plugins": "Complements de proveïdor de metadades", + "paste_plugin_download_url": "Enllaça l’URL de descàrrega o el repositori de GitHub/Codeberg o l’enllaç directe al fitxer .smplug", + "download_and_install_plugin_from_url": "Descarrega i instal·la el complement des d’un URL", + "failed_to_add_plugin_error": "Error en afegir el complement: {error}", + "upload_plugin_from_file": "Penja el complement des d’un fitxer", + "installed": "Instal·lat", + "available_plugins": "Complements disponibles", + "configure_your_own_metadata_plugin": "Configura el teu propi proveïdor de metadades per llistes/reproduccions àlbum/artista/flux", + "audio_scrobblers": "Scrobblers d’àudio", + "scrobbling": "Scrobbling", + "download_music_format": "Format de descàrrega de música", + "streaming_music_format": "Format de reproducció de música en temps real", + "download_music_quality": "Qualitat de descàrrega de música", + "streaming_music_quality": "Qualitat de reproducció de música en temps real", + "default_metadata_source": "Font de metadades per defecte", + "set_default_metadata_source": "Estableix la font de metadades per defecte", + "default_audio_source": "Font d'àudio per defecte", + "set_default_audio_source": "Estableix la font d'àudio per defecte", + "plugins": "Connectors", + "configure_plugins": "Configura els teus propis connectors de proveïdor de metadades i de font d'àudio", + "source": "Font: ", + "uncompressed": "Sense comprimir", + "dab_music_source_description": "Per als audiòfils. Ofereix fluxos d'àudio d'alta qualitat/sense pèrdua. Coincidència precisa de pistes basada en ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_cs.arb b/lib/l10n/app_cs.arb index a40251c0..59938004 100644 --- a/lib/l10n/app_cs.arb +++ b/lib/l10n/app_cs.arb @@ -401,5 +401,94 @@ "export_cache_files": "Exportovat soubory z mezipaměti", "found_n_files": "Nalezeno {count} souborů", "export_cache_confirmation": "Chcete exportovat tyto soubory do", - "exported_n_out_of_m_files": "Exportováno {filesExported} z {files} souborů" + "exported_n_out_of_m_files": "Exportováno {filesExported} z {files} souborů", + "playlist": "Seznam skladeb", + "no_loop": "Žádné opakování", + "generate": "Generovat", + "undo": "Zpět", + "download_all": "Stáhnout vše", + "add_all_to_playlist": "Přidat vše do seznamu skladeb", + "add_all_to_queue": "Přidat vše do fronty", + "play_all_next": "Přehrát vše následně", + "pause": "Pauza", + "view_all": "Zobrazit vše", + "no_tracks_added_yet": "Zdá se, že jste ještě nepřidali žádné skladby", + "no_tracks": "Zdá se, že zde nejsou žádné skladby", + "no_tracks_listened_yet": "Zdá se, že jste ještě nic neposlouchali", + "not_following_artists": "Nezajímáte se o žádné umělce", + "no_favorite_albums_yet": "Zdá se, že jste ještě nepřidali žádné alba mezi oblíbené", + "no_logs_found": "Žádné záznamy nenalezeny", + "youtube_engine": "YouTube Engine", + "youtube_engine_not_installed_title": "{engine} není nainstalován", + "youtube_engine_not_installed_message": "{engine} není nainstalován ve vašem systému.", + "youtube_engine_set_path": "Ujistěte se, že je k dispozici v proměnné PATH nebo\nnastavte absolutní cestu k {engine} spustitelnému souboru níže", + "youtube_engine_unix_issue_message": "V macOS/Linux/Unixových systémech nebude fungovat nastavení cesty v .zshrc/.bashrc/.bash_profile atd.\nMusíte nastavit cestu v konfiguračním souboru shellu", + "download": "Stáhnout", + "file_not_found": "Soubor nenalezen", + "custom": "Vlastní", + "add_custom_url": "Přidat vlastní URL", + "edit_port": "Upravit port", + "port_helper_msg": "Výchozí hodnota je -1, což znamená náhodné číslo. Pokud máte nakonfigurován firewall, doporučuje se to nastavit.", + "connect_request": "Povolit {client} připojení?", + "connection_request_denied": "Připojení bylo zamítnuto. Uživatel odmítl přístup.", + "hipotetical_calculation": "*Toto je vypočítáno na základě průměrného výplatu za přehrání 0,003–0,005 USD na online hudebních streamovacích platformách. Jedná se o hypotetický výpočet, který má uživateli ukázat, kolik by umělci dostali, pokud by jeho píseň poslouchal na jiné platformě.", + "an_error_occurred": "Došlo k chybě", + "copy_to_clipboard": "Kopírovat do schránky", + "view_logs": "Zobrazit protokoly", + "retry": "Zkusit znovu", + "no_default_metadata_provider_selected": "Nemáte nastaven výchozí poskytovatel metadat", + "manage_metadata_providers": "Spravovat poskytovatele metadat", + "open_link_in_browser": "Otevřít odkaz v prohlížeči?", + "do_you_want_to_open_the_following_link": "Chcete otevřít následující odkaz?", + "unsafe_url_warning": "Odkazy z nedůvěryhodných zdrojů mohou být nebezpečné. Buďte opatrní!\nOdkaz si také můžete zkopírovat do schránky.", + "copy_link": "Zkopírovat odkaz", + "building_your_timeline": "Vytváří se váš časový přehled podle poslechů...", + "official": "Oficiální", + "author_name": "Autor: {author}", + "third_party": "Třetí strana", + "plugin_requires_authentication": "Plugin vyžaduje ověření", + "update_available": "Aktualizace dostupná", + "supports_scrobbling": "Podpora scrobblování", + "plugin_scrobbling_info": "Tento plugin scrobbles vaši hudbu pro vytvoření historie poslechů.", + "default_plugin": "Výchozí", + "set_default": "Nastavit jako výchozí", + "support": "Podpora", + "support_plugin_development": "Podpořit vývoj pluginu", + "can_access_name_api": "- Může přistupovat k API **{name}**", + "do_you_want_to_install_this_plugin": "Chcete tento plugin nainstalovat?", + "third_party_plugin_warning": "Tento plugin pochází z repozitáře třetí strany. Ujistěte se, že důvěřujete zdroji, než ho nainstalujete.", + "author": "Autor", + "this_plugin_can_do_following": "Tento plugin může provádět následující úkony", + "install": "Instalovat", + "install_a_metadata_provider": "Nainstalovat poskytovatele metadat", + "no_tracks_playing": "Momentálně není přehrávána žádná skladba", + "synced_lyrics_not_available": "Synchronizované texty nejsou k dispozici k této písni. Prosím použijte", + "plain_lyrics": "Prostý text", + "tab_instead": "místo toho použijte tabulátor.", + "disclaimer": "Prohlášení", + "third_party_plugin_dmca_notice": "Tým Spotube nenese žádnou odpovědnost (včetně právní) za pluginy „třetích stran“.\nPoužívejte je na vlastní riziko. Pro chyby/problémy je nahlaste do repozitáře pluginu.\n\nPokud jakýkoli plugin „třetí strany“ porušuje podmínky služby nebo DMCA kteréhokoli poskytovatele či právního subjektu, požádejte autora pluginu nebo hostingovou platformu (např. GitHub/Codeberg), aby podnikla kroky. Pluginy označené jako „třetí strana“ jsou otevřené a spravovány komunitou; nespravujeme je, tudíž nemůžeme jednat.\n\n", + "input_does_not_match_format": "Vstup neodpovídá požadovanému formátu", + "metadata_provider_plugins": "Pluginy poskytovatelů metadat", + "paste_plugin_download_url": "Vložte URL ke stažení nebo GitHub/Codeberg repozitář či přímý odkaz na soubor .smplug", + "download_and_install_plugin_from_url": "Stáhnout a nainstalovat plugin z URL", + "failed_to_add_plugin_error": "Nepodařilo se přidat plugin: {error}", + "upload_plugin_from_file": "Nahrát plugin ze souboru", + "installed": "Nainstalováno", + "available_plugins": "Dostupné pluginy", + "configure_your_own_metadata_plugin": "Nakonfigurujte si vlastního poskytovatele metadat pro playlist/album/umělec/fid", + "audio_scrobblers": "Audio scrobblers", + "scrobbling": "Scrobbling", + "download_music_format": "Formát stahování hudby", + "streaming_music_format": "Formát streamování hudby", + "download_music_quality": "Kvalita stahování hudby", + "streaming_music_quality": "Kvalita streamování hudby", + "default_metadata_source": "Výchozí zdroj metadat", + "set_default_metadata_source": "Nastavit výchozí zdroj metadat", + "default_audio_source": "Výchozí zdroj zvuku", + "set_default_audio_source": "Nastavit výchozí zdroj zvuku", + "plugins": "Pluginy", + "configure_plugins": "Konfigurujte své vlastní pluginy poskytovatele metadat a zdroje zvuku", + "source": "Zdroj: ", + "uncompressed": "Nekomprimováno", + "dab_music_source_description": "Pro audiofily. Poskytuje vysoce kvalitní/bezztrátové zvukové toky. Přesná shoda skladeb na základě ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_de.arb b/lib/l10n/app_de.arb index 76ec2218..458e7c07 100644 --- a/lib/l10n/app_de.arb +++ b/lib/l10n/app_de.arb @@ -401,5 +401,94 @@ "export_cache_files": "Cachedateien exportieren", "found_n_files": "{count} Dateien gefunden", "export_cache_confirmation": "Möchten Sie diese Dateien exportieren nach", - "exported_n_out_of_m_files": "{filesExported} von {files} Dateien exportiert" + "exported_n_out_of_m_files": "{filesExported} von {files} Dateien exportiert", + "playlist": "Playlist", + "no_loop": "Kein Loop", + "generate": "Generieren", + "undo": "Rückgängig", + "download_all": "Alle herunterladen", + "add_all_to_playlist": "Alle zur Playlist hinzufügen", + "add_all_to_queue": "Alle zur Warteschlange hinzufügen", + "play_all_next": "Alle als Nächstes abspielen", + "pause": "Pause", + "view_all": "Alle ansehen", + "no_tracks_added_yet": "Sie haben noch keine Titel hinzugefügt.", + "no_tracks": "Es sieht so aus, als ob hier keine Titel sind.", + "no_tracks_listened_yet": "Es scheint, dass Sie noch nichts gehört haben.", + "not_following_artists": "Sie folgen noch keinem Künstler.", + "no_favorite_albums_yet": "Es sieht so aus, als ob Sie noch keine Alben zu Ihren Favoriten hinzugefügt haben.", + "no_logs_found": "Keine Protokolle gefunden", + "youtube_engine": "YouTube-Engine", + "youtube_engine_not_installed_title": "{engine} ist nicht installiert", + "youtube_engine_not_installed_message": "{engine} ist nicht auf Ihrem System installiert.", + "youtube_engine_set_path": "Stellen Sie sicher, dass es im PATH verfügbar ist oder\nsetzen Sie den absoluten Pfad zur {engine} ausführbaren Datei unten.", + "youtube_engine_unix_issue_message": "In macOS/Linux/unixähnlichen Betriebssystemen funktioniert das Setzen des Pfads in .zshrc/.bashrc/.bash_profile usw. nicht.\nSie müssen den Pfad in der Shell-Konfigurationsdatei festlegen.", + "download": "Herunterladen", + "file_not_found": "Datei nicht gefunden", + "custom": "Benutzerdefiniert", + "add_custom_url": "Benutzerdefinierte URL hinzufügen", + "edit_port": "Port bearbeiten", + "port_helper_msg": "Der Standardwert ist -1, was eine zufällige Zahl bedeutet. Wenn Sie eine Firewall konfiguriert haben, wird empfohlen, dies einzustellen.", + "connect_request": "{client} die Verbindung erlauben?", + "connection_request_denied": "Verbindung abgelehnt. Benutzer hat den Zugriff verweigert.", + "hipotetical_calculation": "*Diese Berechnung basiert auf der durchschnittlichen Auszahlung pro Stream (0,003 USD bis 0,005 USD) auf Online-Musik-Streaming-Plattformen. Sie ist hypothetisch und soll dem Nutzer veranschaulichen, wie viel er den Künstlern bezahlt hätte, wenn er ihren Song auf verschiedenen Streaming-Plattformen gehört hätte.", + "an_error_occurred": "Ein Fehler ist aufgetreten", + "copy_to_clipboard": "In die Zwischenablage kopieren", + "view_logs": "Protokolle anzeigen", + "retry": "Erneut versuchen", + "no_default_metadata_provider_selected": "Sie haben keinen Standard-Metadatenanbieter festgelegt", + "manage_metadata_providers": "Metadatenanbieter verwalten", + "open_link_in_browser": "Link im Browser öffnen?", + "do_you_want_to_open_the_following_link": "Möchten Sie folgenden Link öffnen?", + "unsafe_url_warning": "Das Öffnen von Links aus nicht vertrauenswürdigen Quellen kann unsicher sein. Seien Sie vorsichtig!\nSie können den Link auch in Ihre Zwischenablage kopieren.", + "copy_link": "Link kopieren", + "building_your_timeline": "Ihr Zeitverlauf wird basierend auf Ihren Hördaten erstellt…", + "official": "Offiziell", + "author_name": "Autor: {author}", + "third_party": "Drittanbieter", + "plugin_requires_authentication": "Plugin erfordert Authentifizierung", + "update_available": "Update verfügbar", + "supports_scrobbling": "Unterstützt Scrobbling", + "plugin_scrobbling_info": "Dieses Plugin scrobbelt Ihre Musik, um Ihre Hörhistorie zu erstellen.", + "default_plugin": "Standard", + "set_default": "Als Standard festlegen", + "support": "Unterstützung", + "support_plugin_development": "Plugin-Entwicklung unterstützen", + "can_access_name_api": "- Kann auf **{name}**-API zugreifen", + "do_you_want_to_install_this_plugin": "Möchten Sie dieses Plugin installieren?", + "third_party_plugin_warning": "Dieses Plugin stammt aus einem Drittanbieter-Repository. Bitte stellen Sie sicher, dass Sie der Quelle vertrauen, bevor Sie es installieren.", + "author": "Autor", + "this_plugin_can_do_following": "Dieses Plugin kann Folgendes:", + "install": "Installieren", + "install_a_metadata_provider": "Einen Metadatenanbieter installieren", + "no_tracks_playing": "Derzeit wird kein Titel abgespielt", + "synced_lyrics_not_available": "Synchronisierte Liedtexte sind für dieses Lied nicht verfügbar. Bitte verwenden Sie stattdessen", + "plain_lyrics": "Einfache Liedtexte", + "tab_instead": "stattdessen die Tab-Taste verwenden.", + "disclaimer": "Haftungsausschluss", + "third_party_plugin_dmca_notice": "Das Spotube-Team übernimmt keine Verantwortung (auch nicht rechtlicher Art) für Plugins \"Drittanbieter\". Nutzen Sie diese auf eigenes Risiko. Für Fehler/Probleme melden Sie sich bitte beim Plugin-Repository.\n\nWenn ein Plugin \"Drittanbieter\" gegen die ToS/DMCA eines Dienstes bzw. gesetzlicher Vorschriften verstößt, wenden Sie sich bitte an den Plugin-Autor oder die Hosting-Plattform (z. B. GitHub/Codeberg), um Maßnahmen zu ergreifen. Die genannten Plugins (mit \"Drittanbieter\"-Kennzeichnung) werden öffentlich und gemeinschaftlich gepflegt. Wir kuratieren sie nicht und können keine Maßnahmen ergreifen.\n\n", + "input_does_not_match_format": "Eingabe entspricht nicht dem geforderten Format", + "metadata_provider_plugins": "Plugins für Metadatenanbieter", + "paste_plugin_download_url": "Download-URL, GitHub/Codeberg-Repo-URL oder direkten Link zur .smplug-Datei einfügen", + "download_and_install_plugin_from_url": "Plugin per URL herunterladen und installieren", + "failed_to_add_plugin_error": "Plugin konnte nicht hinzugefügt werden: {error}", + "upload_plugin_from_file": "Plugin per Datei hochladen", + "installed": "Installiert", + "available_plugins": "Verfügbare Plugins", + "configure_your_own_metadata_plugin": "Eigenen Anbieter für Playlist-/Album-/Künstler-/Feed-Metadaten konfigurieren", + "audio_scrobblers": "Audio-Scrobbler", + "scrobbling": "Scrobbling", + "download_music_format": "Musik-Downloadformat", + "streaming_music_format": "Musik-Streamingformat", + "download_music_quality": "Musik-Downloadqualität", + "streaming_music_quality": "Musik-Streamingqualität", + "default_metadata_source": "Standard-Metadatenquelle", + "set_default_metadata_source": "Standard-Metadatenquelle festlegen", + "default_audio_source": "Standard-Audioquelle", + "set_default_audio_source": "Standard-Audioquelle festlegen", + "plugins": "Plugins", + "configure_plugins": "Richte deine eigenen Metadatenanbieter- und Audioquellen-Plugins ein", + "source": "Quelle: ", + "uncompressed": "Unkomprimiert", + "dab_music_source_description": "Für Audiophile. Bietet hochwertige/verlustfreie Audiostreams. Präzises ISRC-basiertes Track-Matching." } \ No newline at end of file diff --git a/lib/l10n/app_en.arb b/lib/l10n/app_en.arb index f949480e..111d76a8 100644 --- a/lib/l10n/app_en.arb +++ b/lib/l10n/app_en.arb @@ -22,7 +22,7 @@ "filter_playlists": "Filter your playlists...", "liked_tracks": "Liked Tracks", "liked_tracks_description": "All your liked tracks", - "create_playlist": "Create Playlist", + "playlist": "Playlist", "create_a_playlist": "Create a playlist", "update_playlist": "Update playlist", "create": "Create", @@ -97,6 +97,7 @@ "pause_playback": "Pause Playback", "resume_playback": "Resume Playback", "loop_track": "Loop track", + "no_loop": "No loop", "repeat_playlist": "Repeat playlist", "queue": "Queue", "alternative_track_sources": "Alternative track sources", @@ -111,8 +112,6 @@ "add_library_location": "Add to library", "remove_library_location": "Remove from library", "account": "Account", - "login_with_spotify": "Login with your Spotify account", - "connect_with_spotify": "Connect with Spotify", "logout": "Logout", "logout_of_this_account": "Logout of this account", "language_region": "Language & Region", @@ -153,7 +152,7 @@ "about_spotube": "About Spotube", "blacklist": "Blacklist", "please_sponsor": "Please Sponsor/Donate", - "spotube_description": "Spotube, a lightweight, cross-platform, free-for-all spotify client", + "spotube_description": "Open source extensible music streaming platform and app, based on BYOMM (Bring your own music metadata) concept", "version": "Version", "build_number": "Build Number", "founder": "Founder", @@ -163,11 +162,9 @@ "kingkor_roy_tirtho": "Kingkor Roy Tirtho", "copyright": "© 2021-{current_year} Kingkor Roy Tirtho", "license": "License", - "add_spotify_credentials": "Add your spotify credentials to get started", "credentials_will_not_be_shared_disclaimer": "Don't worry, any of your credentials won't be collected or shared with anyone", "know_how_to_login": "Don't know how to do this?", "follow_step_by_step_guide": "Follow along the Step by Step guide", - "spotify_cookie": "Spotify {name} Cookie", "cookie_name_cookie": "{name} Cookie", "fill_in_all_fields": "Please fill in all the fields", "submit": "Submit", @@ -177,15 +174,6 @@ "done": "Done", "step_1": "Step 1", "first_go_to": "First, Go to", - "login_if_not_logged_in": "and Login/Signup if you are not logged in", - "step_2": "Step 2", - "step_2_steps": "1. Once you're logged in, press F12 or Mouse Right Click > Inspect to Open the Browser devtools.\n2. Then go the \"Application\" Tab (Chrome, Edge, Brave etc..) or \"Storage\" Tab (Firefox, Palemoon etc..)\n3. Go to the \"Cookies\" section then the \"https://accounts.spotify.com\" subsection", - "step_3": "Step 3", - "step_3_steps": "Copy the value of \"sp_dc\" Cookie", - "success_emoji": "Success🥳", - "success_message": "Now you've successfully Logged in with your Spotify account. Good Job, mate!", - "step_4": "Step 4", - "step_4_steps": "Paste the copied \"sp_dc\" value", "something_went_wrong": "Something went wrong", "piped_instance": "Piped Server Instance", "piped_description": "The Piped server instance to use for track matching", @@ -193,7 +181,7 @@ "invidious_instance": "Invidious Server Instance", "invidious_description": "The Invidious server instance to use for track matching", "invidious_warning": "Some of them might not work well. So use at your own risk", - "generate_playlist": "Generate Playlist", + "generate": "Generate", "track_exists": "Track {track} already exists", "replace_downloaded_tracks": "Replace all downloaded tracks", "skip_download_tracks": "Skip downloading all downloaded tracks", @@ -276,8 +264,10 @@ "change_cover": "Change cover", "add_cover": "Add cover", "restore_defaults": "Restore defaults", - "download_music_codec": "Download music codec", - "streaming_music_codec": "Streaming music codec", + "download_music_format": "Download music format", + "streaming_music_format": "Streaming music format", + "download_music_quality": "Download music quality", + "streaming_music_quality": "Streaming music quality", "login_with_lastfm": "Login with Last.fm", "connect": "Connect", "disconnect_lastfm": "Disconnect Last.fm", @@ -375,7 +365,7 @@ "count_streams": "{count} streams", "owned_by_you": "Owned by you", "copied_shareurl_to_clipboard": "Copied {shareUrl} to clipboard", - "spotify_hipotetical_calculation": "*This is calculated based on Spotify's per stream\npayout of $0.003 to $0.005. This is a hypothetical\ncalculation to give user insight about how much they\nwould have paid to the artists if they were to listen\ntheir song in Spotify.", + "hipotetical_calculation": "*This is calculated based on average online music streaming platform's per stream\npayout of $0.003 to $0.005. This is a hypothetical\ncalculation to give user insight about how much they\nwould have paid to the artists if they were to listen\ntheir song in different music streaming platform.", "count_mins": "{minutes} mins", "summary_minutes": "minutes", "summary_listened_to_music": "Listened to music", @@ -401,5 +391,83 @@ "export_cache_files": "Export Cached Files", "found_n_files": "Found {count} files", "export_cache_confirmation": "Do you want to export these files to", - "exported_n_out_of_m_files": "Exported {filesExported} out of {files} files" -} \ No newline at end of file + "exported_n_out_of_m_files": "Exported {filesExported} out of {files} files", + "undo": "Undo", + "download_all": "Download all", + "add_all_to_playlist": "Add all to playlist", + "add_all_to_queue": "Add all to queue", + "play_all_next": "Play all next", + "pause": "Pause", + "view_all": "View all", + "no_tracks_added_yet": "Looks like you haven't added any tracks yet", + "no_tracks": "Looks like there are no tracks here", + "no_tracks_listened_yet": "Looks like you haven't listened to anything yet", + "not_following_artists": "You're not following any artists", + "no_favorite_albums_yet": "Looks like you haven't added any albums to your favorites yet", + "no_logs_found": "No logs found", + "youtube_engine": "YouTube Engine", + "youtube_engine_not_installed_title": "{engine} is not installed", + "youtube_engine_not_installed_message": "{engine} is not installed in your system.", + "youtube_engine_set_path": "Make sure it's available in the PATH variable or\nset the absolute path to the {engine} executable below", + "youtube_engine_unix_issue_message": "In macOS/Linux/unix like OS's, setting path on .zshrc/.bashrc/.bash_profile etc. won't work.\nYou need to set the path in the shell configuration file", + "download": "Download", + "file_not_found": "File not found", + "custom": "Custom", + "add_custom_url": "Add custom URL", + "edit_port": "Edit port", + "port_helper_msg": "Default is -1 which indicates random number. If you've firewall configured, setting this is recommended.", + "connect_request": "Allow {client} to connect?", + "connection_request_denied": "Connection denied. User denied access.", + "an_error_occurred": "An error occurred", + "copy_to_clipboard": "Copy to clipboard", + "view_logs": "View logs", + "retry": "Retry", + "no_default_metadata_provider_selected": "You've no default metadata provider set", + "manage_metadata_providers": "Manage metadata providers", + "open_link_in_browser": "Open Link in Browser?", + "do_you_want_to_open_the_following_link": "Do you want to open the following link", + "unsafe_url_warning": "It can be unsafe to open links from untrusted sources. Be cautious!\nYou can also copy the link to your clipboard.", + "copy_link": "Copy Link", + "building_your_timeline": "Building your timeline based on your listenings...", + "official": "Official", + "author_name": "Author: {author}", + "third_party": "Third-party", + "plugin_requires_authentication": "Plugin requires authentication", + "update_available": "Update available", + "supports_scrobbling": "Supports scrobbling", + "plugin_scrobbling_info": "This plugin scrobbles your music to generate your listening history.", + "default_metadata_source": "Default metadata source", + "set_default_metadata_source": "Set default metadata source", + "default_audio_source": "Default audio source", + "set_default_audio_source": "Set default audio source", + "set_default": "Set default", + "support": "Support", + "support_plugin_development": "Support plugin development", + "can_access_name_api": "- Can access **{name}** API", + "do_you_want_to_install_this_plugin": "Do you want to install this plugin?", + "third_party_plugin_warning": "This plugin is from a third-party repository. Please ensure you trust the source before installing.", + "author": "Author", + "this_plugin_can_do_following": "This plugin can do following", + "install": "Install", + "install_a_metadata_provider": "Install a Metadata Provider", + "no_tracks_playing": "No Track being played currently", + "synced_lyrics_not_available": "Synced lyrics are not available for this song. Please use the", + "plain_lyrics": "Plain Lyrics", + "tab_instead": "tab instead.", + "disclaimer": "Disclaimer", + "third_party_plugin_dmca_notice": "The Spotube team does not hold any responsibility (including legal) for any \"Third-party\" plugins.\nPlease use them at your own risk. For any bugs/issues, please report them to the plugin repository.\n\nIf any \"Third-party\" plugin is breaking ToS/DMCA of any service/legal entity, please ask the \"Third-party\" plugin author or the hosting platform .e.g GitHub/Codeberg to take action. Above listed (\"Third-party\" labelled) are all public/community maintained plugins. We're not curating them, so we cannot take any action on them.\n\n", + "input_does_not_match_format": "Input doesn't match the required format", + "plugins": "Plugins", + "paste_plugin_download_url": "Paste download url or GitHub/Codeberg repo url or direct link to .smplug file", + "download_and_install_plugin_from_url": "Download and install plugin from url", + "failed_to_add_plugin_error": "Failed to add plugin: {error}", + "upload_plugin_from_file": "Upload plugin from file", + "installed": "Installed", + "available_plugins": "Available plugins", + "configure_plugins": "Configure your own metadata provider and audio source plugins", + "audio_scrobblers": "Audio Scrobblers", + "scrobbling": "Scrobbling", + "source": "Source: ", + "uncompressed": "Uncompressed", + "dab_music_source_description": "For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching." +} diff --git a/lib/l10n/app_es.arb b/lib/l10n/app_es.arb index 9fc7e560..32822763 100644 --- a/lib/l10n/app_es.arb +++ b/lib/l10n/app_es.arb @@ -401,5 +401,94 @@ "export_cache_files": "Exportar archivos en caché", "found_n_files": "Se encontraron {count} archivos", "export_cache_confirmation": "¿Desea exportar estos archivos a", - "exported_n_out_of_m_files": "Se exportaron {filesExported} de {files} archivos" + "exported_n_out_of_m_files": "Se exportaron {filesExported} de {files} archivos", + "playlist": "Lista de reproducción", + "no_loop": "Sin bucle", + "generate": "Generar", + "undo": "Deshacer", + "download_all": "Descargar todo", + "add_all_to_playlist": "Agregar todo a la lista de reproducción", + "add_all_to_queue": "Agregar todo a la cola", + "play_all_next": "Reproducir todo a continuación", + "pause": "Pausa", + "view_all": "Ver todo", + "no_tracks_added_yet": "Parece que aún no has agregado ninguna canción.", + "no_tracks": "Parece que no hay canciones aquí.", + "no_tracks_listened_yet": "Parece que no has escuchado nada todavía.", + "not_following_artists": "No sigues a ningún artista.", + "no_favorite_albums_yet": "Parece que aún no has agregado ningún álbum a tus favoritos.", + "no_logs_found": "No se encontraron registros", + "youtube_engine": "Motor de YouTube", + "youtube_engine_not_installed_title": "{engine} no está instalado", + "youtube_engine_not_installed_message": "{engine} no está instalado en tu sistema.", + "youtube_engine_set_path": "Asegúrate de que esté disponible en la variable PATH o\nestablece la ruta absoluta del ejecutable de {engine} a continuación.", + "youtube_engine_unix_issue_message": "En macOS/Linux/sistemas operativos similares a Unix, establecer la ruta en .zshrc/.bashrc/.bash_profile etc. no funcionará.\nNecesitas establecer la ruta en el archivo de configuración del shell.", + "download": "Descargar", + "file_not_found": "Archivo no encontrado", + "custom": "Personalizado", + "add_custom_url": "Agregar URL personalizada", + "edit_port": "Editar puerto", + "port_helper_msg": "El valor predeterminado es -1, lo que indica un número aleatorio. Si tienes un firewall configurado, se recomienda establecer esto.", + "connect_request": "¿Permitir que {client} se conecte?", + "connection_request_denied": "Conexión denegada. El usuario denegó el acceso.", + "hipotetical_calculation": "*Este cálculo se basa en el pago promedio por reproducción en plataformas de música en línea (de 0,003 a 0,005 USD). Es hipotético y sirve para dar al usuario una idea de cuánto habría pagado a los artistas si hubiera escuchado su canción en distintas plataformas.", + "an_error_occurred": "Ocurrió un error", + "copy_to_clipboard": "Copiar al portapapeles", + "view_logs": "Ver registros", + "retry": "Reintentar", + "no_default_metadata_provider_selected": "No has configurado un proveedor de metadatos predeterminado", + "manage_metadata_providers": "Gestionar proveedores de metadatos", + "open_link_in_browser": "¿Abrir enlace en el navegador?", + "do_you_want_to_open_the_following_link": "¿Quieres abrir el siguiente enlace?", + "unsafe_url_warning": "Abrir enlaces de fuentes no confiables puede ser inseguro. ¡Ten cuidado!\nTambién puedes copiar el enlace al portapapeles.", + "copy_link": "Copiar enlace", + "building_your_timeline": "Construyendo tu línea de tiempo según tus escuchas…", + "official": "Oficial", + "author_name": "Autor: {author}", + "third_party": "Terceros", + "plugin_requires_authentication": "El complemento requiere autenticación", + "update_available": "Actualización disponible", + "supports_scrobbling": "Admite scrobbling", + "plugin_scrobbling_info": "Este complemento scrobblea tu música para generar tu historial de reproducción.", + "default_plugin": "Predeterminado", + "set_default": "Establecer como predeterminado", + "support": "Soporte", + "support_plugin_development": "Apoyar el desarrollo del complemento", + "can_access_name_api": "- Puede acceder a la API de **{name}**", + "do_you_want_to_install_this_plugin": "¿Deseas instalar este complemento?", + "third_party_plugin_warning": "Este complemento proviene de un repositorio de terceros. Asegúrate de confiar en la fuente antes de instalarlo.", + "author": "Autor", + "this_plugin_can_do_following": "Este complemento puede hacer lo siguiente", + "install": "Instalar", + "install_a_metadata_provider": "Instalar un proveedor de metadatos", + "no_tracks_playing": "No hay ninguna pista reproduciéndose actualmente", + "synced_lyrics_not_available": "Las letras sincronizadas no están disponibles para esta canción. Por favor, utiliza", + "plain_lyrics": "Letras sin formato", + "tab_instead": "en su lugar, usa la tecla Tab.", + "disclaimer": "Descargo de responsabilidad", + "third_party_plugin_dmca_notice": "El equipo de Spotube no asume ninguna responsabilidad (incluida la legal) por complementos de \"terceros\". Úsalos bajo tu propio riesgo. Para errores o problemas, repórtalos en el repositorio del complemento.\n\nSi algún complemento de “terceros” infringe los ToS/DMCA de algún servicio o entidad legal, por favor, solicita al autor del complemento o a la plataforma de alojamiento (p. ej., GitHub/Codeberg) que tome medidas. Los complementos etiquetados como “de terceros” son mantenidos públicamente por la comunidad; no los gestionamos y no podemos intervenir.\n\n", + "input_does_not_match_format": "La entrada no coincide con el formato requerido", + "metadata_provider_plugins": "Complementos de proveedor de metadatos", + "paste_plugin_download_url": "Pega la URL de descarga, el repositorio de GitHub/Codeberg o el enlace directo al archivo .smplug", + "download_and_install_plugin_from_url": "Descargar e instalar el complemento desde una URL", + "failed_to_add_plugin_error": "Error al añadir el complemento: {error}", + "upload_plugin_from_file": "Subir complemento desde archivo", + "installed": "Instalado", + "available_plugins": "Complementos disponibles", + "configure_your_own_metadata_plugin": "Configura tu propio proveedor de metadatos para listas/álbum/artista/feeds", + "audio_scrobblers": "Scrobblers de audio", + "scrobbling": "Scrobbling", + "download_music_format": "Formato de descarga de música", + "streaming_music_format": "Formato de transmisión de música", + "download_music_quality": "Calidad de descarga de música", + "streaming_music_quality": "Calidad de transmisión de música", + "default_metadata_source": "Fuente de metadatos predeterminada", + "set_default_metadata_source": "Establecer fuente de metadatos predeterminada", + "default_audio_source": "Fuente de audio predeterminada", + "set_default_audio_source": "Establecer fuente de audio predeterminada", + "plugins": "Plugins", + "configure_plugins": "Configura tus propios plugins de proveedor de metadatos y fuente de audio", + "source": "Fuente: ", + "uncompressed": "Sin comprimir", + "dab_music_source_description": "Para audiófilos. Proporciona transmisiones de audio de alta calidad/sin pérdida. Coincidencia precisa de pistas basada en ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_eu.arb b/lib/l10n/app_eu.arb index 98596725..8c87fd2c 100644 --- a/lib/l10n/app_eu.arb +++ b/lib/l10n/app_eu.arb @@ -401,5 +401,94 @@ "export_cache_files": "Esportatu cache fitxategiak", "found_n_files": "{count} fitxategi aurkitu dira", "export_cache_confirmation": "Fitxategi hauek esportatu nahi al dituzu", - "exported_n_out_of_m_files": "{filesExported} fitxategi esportatu dira {files} -tik" + "exported_n_out_of_m_files": "{filesExported} fitxategi esportatu dira {files} -tik", + "playlist": "Playlist", + "no_loop": "Ez dago loop-ik", + "generate": "Sortu", + "undo": "Desegondu", + "download_all": "Guztia deskargatu", + "add_all_to_playlist": "Guztia playlist-era gehitu", + "add_all_to_queue": "Guztia zerrendara gehitu", + "play_all_next": "Guztia hurrengoan jolastu", + "pause": "Pausatu", + "view_all": "Ikusi guztia", + "no_tracks_added_yet": "Dirudienez, oraindik ez duzu abestirik gehitu.", + "no_tracks": "Ez dirudi hemen abestirik dagoenik.", + "no_tracks_listened_yet": "Dirudienez, oraindik ez duzu ezer entzun.", + "not_following_artists": "Ez zaude artisten atzetik.", + "no_favorite_albums_yet": "Dirudienez, oraindik ez duzu albumik gehitu zure gogokoen artean.", + "no_logs_found": "Ez dira log-ak aurkitu", + "youtube_engine": "YouTube Motorra", + "youtube_engine_not_installed_title": "{engine} ez dago instalatuta", + "youtube_engine_not_installed_message": "{engine} ez dago zure sisteman instalatuta.", + "youtube_engine_set_path": "Ziurtatu PATH aldagaiaren barruan dagoela edo\nezarri {engine} exekutagarriaren helbide absolutua behean.", + "youtube_engine_unix_issue_message": "macOS/Linux/Unix bezalako sistemetan, .zshrc/.bashrc/.bash_profile bezalako fitxategietan bidearen ezarpenak ez dira funtzionatuko.\nBidearen ezarpena shell konfigurazio fitxategian egin behar duzu.", + "download": "Deskargatu", + "file_not_found": "Fitxategia ez da aurkitu", + "custom": "Pertsonalizatua", + "add_custom_url": "Gehitu URL pertsonalizatua", + "edit_port": "Editatu portua", + "port_helper_msg": "Lehenetsitako balioa -1 da, zenbaki aleatorioa adierazten duena. Su firewall konfiguratu baduzu, gomendatzen da hau ezartzea.", + "connect_request": "{client} konektatzea baimendu?", + "connection_request_denied": "Konektatzea ukatu da. Erabiltzaileak sarbidea ukatu du.", + "hipotetical_calculation": "*Kalkulu hau online musika-streaming plataformetako batez besteko irteerako ordainari (0,003–0,005 USD) oinarrituta dago. Hipotetikoa da eta erabiltzaileari ideia bat ematen laguntzen dio artista nork zenbat kobratu zuen jakiteko, bere abestia plataform desberdinetan entzungo balu.", + "an_error_occurred": "Errore bat gertatu da", + "copy_to_clipboard": "Hiztegiraino kopiatzea", + "view_logs": "Erregistroak ikusi", + "retry": "Berriro saiatu", + "no_default_metadata_provider_selected": "Ezarri ez duzu metadaten hornitzaile lehenetsirik", + "manage_metadata_providers": "Metadaten hornitzaileak kudeatu", + "open_link_in_browser": "Esteka nabigatzailean irekiko duzu?", + "do_you_want_to_open_the_following_link": "Hurrengo esteka irekiko duzu?", + "unsafe_url_warning": "Iturri seguru gabeko estekak irekiz gero, ez da seguru suerta daiteke. Arduratu zaitez!\nEsteka ere hiztegirainokoan kopiatu dezakezu.", + "copy_link": "Esteka kopiatu", + "building_your_timeline": "Zure entzuteen arabera zure kronologia eraikitzen…", + "official": "Ofiziala", + "author_name": "Egilea: {author}", + "third_party": "Hirugarrena", + "plugin_requires_authentication": "Pluginak autentifikazioa eskatzen du", + "update_available": "Eguneratze bat dago eskuragarri", + "supports_scrobbling": "Scrobbling-a onartzen du", + "plugin_scrobbling_info": "Plugin honek zure musika scrobbled egiten du zure entzuteen historia sortzeko.", + "default_plugin": "Lehenetsia", + "set_default": "Lehenetsi gisa ezarri", + "support": "Laguntza", + "support_plugin_development": "Pluginaren garapena lagundu", + "can_access_name_api": "- **{name}** API-ra sar daiteke", + "do_you_want_to_install_this_plugin": "Plugin hau instalatu nahiko zenuke?", + "third_party_plugin_warning": "Plugin hau hirugarrenen biltegi batetik dator. Instalatu aurretik iturriari konfiantza behar diozu.", + "author": "Egilea", + "this_plugin_can_do_following": "Plugin honek honako hau egin dezake:", + "install": "Instalatu", + "install_a_metadata_provider": "Metadaten hornitzaile bat instalatu", + "no_tracks_playing": "Une honetan ez dago abestirik erreproduzitzen", + "synced_lyrics_not_available": "Abestiarentzako letra sinkronizatua ez dago erabilgarri. Mesedez, erabili", + "plain_lyrics": "Letra arrunta", + "tab_instead": "horren ordez, Tab teklatxaza erabili.", + "disclaimer": "Aldez aurreko oharra", + "third_party_plugin_dmca_notice": "Spotube taldea ezin da arduratu (“hirugarrenen”) plugin-en>gatik (barne legala). Erabili zure arriskuarekin. Erroreak/ arazoak dituzu, jakinarazi pluginaren biltegiari.\n\nPlugin batek edozein zerbitzu/legalki entitate baten ToS/DMCA hautsi baditu, eska iezaiozu pluginaren egileari edo hosting plataformari (adibidez GitHub/Codeberg) neurriak har ditzaten. “Hirugarrena” etiketatutako plugin guztiak komunitate publikoaren bidez mantentzen dira; ez ditugu kuratoriatu, beraz ezin dugu inplikatu.\n\n", + "input_does_not_match_format": "Sarrera ezin da beharrezko formatutik desberdina izan", + "metadata_provider_plugins": "Metadaten hornitzailearen pluginak", + "paste_plugin_download_url": "Kopiatu deskarga-URLa, GitHub/Codeberg biltegi-URLa edo .smplug fitxategiaren esteka zuzena", + "download_and_install_plugin_from_url": "Download eta instalatu plugin-a URL batetik", + "failed_to_add_plugin_error": "Plugin gehitu ezin izan da: {error}", + "upload_plugin_from_file": "Plugin fitxategi batetik igo", + "installed": "Instalatuta", + "available_plugins": "Eskaintzen diren pluginak", + "configure_your_own_metadata_plugin": "Konfiguratu zureko playlists-/album-/artista-/feed-metadaten hornitzailea", + "audio_scrobblers": "Audio scrobbler-ak", + "scrobbling": "Scrobbling", + "download_music_format": "Musika deskargatzeko formatua", + "streaming_music_format": "Musika streaming bidezko formatua", + "download_music_quality": "Musika deskargaren kalitatea", + "streaming_music_quality": "Streaming bidezko musika kalitatea", + "default_metadata_source": "Metadatu-iturburu lehenetsia", + "set_default_metadata_source": "Ezarri metadatu-iturburu lehenetsia", + "default_audio_source": "Audio-iturburu lehenetsia", + "set_default_audio_source": "Ezarri audio-iturburu lehenetsia", + "plugins": "Pluginak", + "configure_plugins": "Konfiguratu zure metadatu-hornitzaile eta audio-iturburu pluginak", + "source": "Iturburua: ", + "uncompressed": "Konprimitu gabea", + "dab_music_source_description": "Audiozaleentzat. Kalitate handiko/galerarik gabeko audio-streamak eskaintzen ditu. ISRC oinarritutako pistaren parekatze zehatza." } \ No newline at end of file diff --git a/lib/l10n/app_fa.arb b/lib/l10n/app_fa.arb index 4d11dd81..72f775fc 100644 --- a/lib/l10n/app_fa.arb +++ b/lib/l10n/app_fa.arb @@ -401,5 +401,94 @@ "export_cache_files": "صادر کردن فایل‌های حافظه موقت", "found_n_files": "{count} فایل یافت شد", "export_cache_confirmation": "آیا می‌خواهید این فایل‌ها را صادر کنید به", - "exported_n_out_of_m_files": "{filesExported} از {files} فایل صادر شد" + "exported_n_out_of_m_files": "{filesExported} از {files} فایل صادر شد", + "playlist": "لیست پخش", + "no_loop": "بدون حلقه", + "generate": "ایجاد", + "undo": "بازگشت", + "download_all": "دانلود همه", + "add_all_to_playlist": "افزودن همه به لیست پخش", + "add_all_to_queue": "افزودن همه به صف", + "play_all_next": "پخش همه بعدی", + "pause": "مکث", + "view_all": "مشاهده همه", + "no_tracks_added_yet": "به نظر می‌رسد هنوز هیچ آهنگی اضافه نکرده‌اید.", + "no_tracks": "به نظر می‌رسد هیچ آهنگی در اینجا وجود ندارد.", + "no_tracks_listened_yet": "به نظر می‌رسد هنوز چیزی نشنیده‌اید.", + "not_following_artists": "شما هیچ هنرمندی را دنبال نمی‌کنید.", + "no_favorite_albums_yet": "به نظر می‌رسد هنوز هیچ آلبومی را به علاقه‌مندی‌هایتان اضافه نکرده‌اید.", + "no_logs_found": "هیچ لاگی پیدا نشد", + "youtube_engine": "موتور YouTube", + "youtube_engine_not_installed_title": "{engine} نصب نشده است", + "youtube_engine_not_installed_message": "{engine} در سیستم شما نصب نشده است.", + "youtube_engine_set_path": "اطمینان حاصل کنید که در متغیر PATH موجود است یا\nآدرس مطلق فایل اجرایی {engine} را در زیر تنظیم کنید.", + "youtube_engine_unix_issue_message": "در macOS/Linux/سیستم‌عامل‌های مشابه Unix، تنظیم مسیر در .zshrc/.bashrc/.bash_profile و غیره کار نمی‌کند.\nباید مسیر را در فایل پیکربندی شل تنظیم کنید.", + "download": "دانلود", + "file_not_found": "فایل پیدا نشد", + "custom": "شخصی‌سازی شده", + "add_custom_url": "اضافه کردن URL سفارشی", + "edit_port": "ویرایش پورت", + "port_helper_msg": "پیش‌فرض -1 است که نشان‌دهنده یک عدد تصادفی است. اگر فایروال شما پیکربندی شده است، توصیه می‌شود این را تنظیم کنید.", + "connect_request": "آیا اجازه می‌دهید {client} متصل شود؟", + "connection_request_denied": "اتصال رد شد. کاربر دسترسی را رد کرد.", + "hipotetical_calculation": "*این محاسبه بر اساس میانگین پرداخت به ازای هر پخش (0.003 تا 0.005 دلار) در پلتفرم‌های استریم موزیک آنلاین انجام شده است. این یک محاسبه فرضی است که به کاربر دیدی از مقدار پرداختی به هنرمندان در صورت گوش دادن به آهنگ آن‌ها در پلتفرم‌های مختلف ارائه می‌دهد.", + "an_error_occurred": "خطایی رخ داد", + "copy_to_clipboard": "کپی به کلیپ‌بورد", + "view_logs": "مشاهده لاگ‌ها", + "retry": "دوباره تلاش کن", + "no_default_metadata_provider_selected": "هیچ ارائه‌دهندهٔ پیش‌فرض متادیتا تعیین نکرده‌اید", + "manage_metadata_providers": "مدیریت ارائه‌دهندگان متادیتا", + "open_link_in_browser": "باز کردن لینک در مرورگر؟", + "do_you_want_to_open_the_following_link": "آیا می‌خواهید لینک زیر را باز کنید؟", + "unsafe_url_warning": "باز کردن لینک از منابع نامطمئن می‌تواند ناامن باشد. مراقب باشید!\nهمچنین می‌توانید لینک را در کلیپ‌بورد خود کپی کنید.", + "copy_link": "کپی لینک", + "building_your_timeline": "در حال ساخت جدول زمانی بر اساس شنیده‌هایتان…", + "official": "رسمی", + "author_name": "نویسنده: {author}", + "third_party": "سوم‌شخص", + "plugin_requires_authentication": "افزونه نیاز به احراز هویت دارد", + "update_available": "به‌روزرسانی در دسترس است", + "supports_scrobbling": "پشتیبانی از اسکراب‌بلینگ", + "plugin_scrobbling_info": "این افزونه موسیقی شما را اسکراب می‌کند تا تاریخچهٔ شنیداری‌تان را تولید کند.", + "default_plugin": "پیش‌فرض", + "set_default": "تنظیم به عنوان پیش‌فرض", + "support": "پشتیبانی", + "support_plugin_development": "حمایت از توسعهٔ افزونه", + "can_access_name_api": "- می‌تواند به API **{name}** دسترسی پیدا کند", + "do_you_want_to_install_this_plugin": "می‌خواهید این افزونه را نصب کنید؟", + "third_party_plugin_warning": "این افزونه از مخزن شخص ثالث آمده است. لطفاً قبل از نصب از منابع آن مطمئن شوید.", + "author": "نویسنده", + "this_plugin_can_do_following": "این افزونه می‌تواند موارد زیر را انجام دهد", + "install": "نصب", + "install_a_metadata_provider": "نصب یک ارائه‌دهندهٔ متادیتا", + "no_tracks_playing": "در حال‌ حاضر هیچ تراکی در حال پخش نیست", + "synced_lyrics_not_available": "متن هم‌زمان‌شده برای این آهنگ در دسترس نیست. لطفاً از", + "plain_lyrics": "متن ساده", + "tab_instead": "به‌جای آن از کلید Tab استفاده کنید.", + "disclaimer": "سلب مسئولیت", + "third_party_plugin_dmca_notice": "تیم Spotube هیچ مسئولیتی (حتی قانونی) در قبال افزونه‌های \"شخص ثالث\" ندارد. از آن‌ها به‌خاطر خود استفاده کنید. برای خطاها/مشکلات، لطفاً در مخزن افزونه گزارش دهید.\n\nاگر هر افزونهٔ \"شخص ثالث\" قوانین ToS/DMCA سرویس یا نهاد قانونی را نقض کند، لطفاً از نویسندهٔ افزونه یا پلتفرم میزبانی (مثل GitHub/Codeberg) درخواست اقدام کنید. افزونه‌هایی که با برچسب \"شخص ثالث\" مشخص شده‌اند، عمومی هستند و توسط جامعه نگهداری می‌شوند؛ ما آن‌ها را تغییر یا مدیریت نمی‌کنیم و نمی‌توانیم دخالت کنیم.\n\n", + "input_does_not_match_format": "ورودی با قالب مورد نیاز تطابق ندارد", + "metadata_provider_plugins": "افزونه‌های ارائه‌دهندهٔ متادیتا", + "paste_plugin_download_url": "URL دانلود یا مخزن GitHub/Codeberg یا لینک مستقیم فایل .smplug را الصاق کنید", + "download_and_install_plugin_from_url": "دانلود و نصب افزونه از طریق لینک", + "failed_to_add_plugin_error": "افزونه اضافه نشد: {error}", + "upload_plugin_from_file": "بارگذاری افزونه از فایل", + "installed": "نصب شد", + "available_plugins": "افزونه‌های موجود", + "configure_your_own_metadata_plugin": "پیکربندی ارائه‌دهندهٔ متادیتا برای پلی‌لیست/آلبوم/هنرمند/فید به‌صورت سفارشی", + "audio_scrobblers": "اسکراب‌بلرهای صوتی", + "scrobbling": "اسکراب‌بلینگ", + "download_music_format": "فرمت دانلود موسیقی", + "streaming_music_format": "فرمت پخش آنلاین موسیقی", + "download_music_quality": "کیفیت دانلود موسیقی", + "streaming_music_quality": "کیفیت پخش آنلاین موسیقی", + "default_metadata_source": "منبع پیش‌فرض فراداده", + "set_default_metadata_source": "تنظیم منبع پیش‌فرض فراداده", + "default_audio_source": "منبع پیش‌فرض صوت", + "set_default_audio_source": "تنظیم منبع پیش‌فرض صوت", + "plugins": "افزونه‌ها", + "configure_plugins": "افزونه‌های منبع صوت و ارائه‌دهنده فراداده خود را پیکربندی کنید", + "source": "منبع: ", + "uncompressed": "بدون فشرده‌سازی", + "dab_music_source_description": "مخصوص علاقه‌مندان صدا. ارائه‌دهنده استریم‌های باکیفیت/بدون افت. تطبیق دقیق آهنگ بر اساس ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_fi.arb b/lib/l10n/app_fi.arb index f6794043..d92e5acf 100644 --- a/lib/l10n/app_fi.arb +++ b/lib/l10n/app_fi.arb @@ -401,5 +401,94 @@ "export_cache_files": "Vie välimuistitiedostot", "found_n_files": "Löydettiin {count} tiedostoa", "export_cache_confirmation": "Haluatko viedä nämä tiedostot", - "exported_n_out_of_m_files": "Vietiin {filesExported}/{files} tiedostoa" + "exported_n_out_of_m_files": "Vietiin {filesExported}/{files} tiedostoa", + "playlist": "Soittolista", + "no_loop": "Ei silmukkaa", + "generate": "Luo", + "undo": "Peruuta", + "download_all": "Lataa kaikki", + "add_all_to_playlist": "Lisää kaikki soittolistalle", + "add_all_to_queue": "Lisää kaikki jonoon", + "play_all_next": "Toista kaikki seuraavaksi", + "pause": "Pysäytä", + "view_all": "Näytä kaikki", + "no_tracks_added_yet": "Näyttää siltä, että et ole lisännyt vielä mitään kappaleita.", + "no_tracks": "Näyttää siltä, että täällä ei ole kappaleita.", + "no_tracks_listened_yet": "Näyttää siltä, että et ole kuunnellut mitään vielä.", + "not_following_artists": "Et seuraa yhtään artistia.", + "no_favorite_albums_yet": "Näyttää siltä, että et ole lisännyt yhtään albumia suosikkeihisi.", + "no_logs_found": "Ei lokitietoja löydetty", + "youtube_engine": "YouTube-moottori", + "youtube_engine_not_installed_title": "{engine} ei ole asennettu", + "youtube_engine_not_installed_message": "{engine} ei ole asennettu järjestelmääsi.", + "youtube_engine_set_path": "Varmista, että se on saatavilla PATH-muuttujassa tai\nasetetaan {engine} suoritettavan tiedoston absoluuttinen polku alla.", + "youtube_engine_unix_issue_message": "macOS/Linux/unix-tyyppisissä käyttöjärjestelmissä polun asettaminen .zshrc/.bashrc/.bash_profile jne. ei toimi.\nSinun täytyy asettaa polku shellin asetustiedostoon.", + "download": "Lataa", + "file_not_found": "Tiedostoa ei löydy", + "custom": "Mukautettu", + "add_custom_url": "Lisää mukautettu URL", + "edit_port": "Muokkaa porttia", + "port_helper_msg": "Oletusarvo on -1, mikä tarkoittaa satunnaista numeroa. Jos sinulla on palomuuri määritetty, tämän asettamista suositellaan.", + "connect_request": "Salli {client} yhdistää?", + "connection_request_denied": "Yhteys evätty. Käyttäjä eväsi pääsyn.", + "hipotetical_calculation": "*Tämä on laskettu keskimääräisen musiikin suoratoistopalvelun 0,003–0,005 dollarin kappalekohtaisen maksun perusteella. Tämä on hypoteettinen laskelma, joka antaa käyttäjälle käsityksen siitä, kuinka paljon he olisivat maksaneet artisteille, jos he kuuntelisivat heidän kappaleitaan eri musiikin suoratoistopalveluissa.", + "an_error_occurred": "Tapahtui virhe", + "copy_to_clipboard": "Kopioi leikepöydälle", + "view_logs": "Näytä lokit", + "retry": "Yritä uudelleen", + "no_default_metadata_provider_selected": "Et ole asettanut oletusmetatietojen tarjoajaa", + "manage_metadata_providers": "Hallinnoi metatietojen tarjoajia", + "open_link_in_browser": "Avaa linkki selaimessa?", + "do_you_want_to_open_the_following_link": "Haluatko avata seuraavan linkin", + "unsafe_url_warning": "Linkkien avaaminen epäluotettavista lähteistä voi olla vaarallista. Ole varovainen!\nVoit myös kopioida linkin leikepöydälle.", + "copy_link": "Kopioi linkki", + "building_your_timeline": "Rakennetaan aikajanaasi kuuntelujesi perusteella...", + "official": "Virallinen", + "author_name": "Tekijä: {author}", + "third_party": "Kolmannen osapuolen", + "plugin_requires_authentication": "Lisäosa vaatii todentamisen", + "update_available": "Päivitys saatavilla", + "supports_scrobbling": "Tukee scrobblingia", + "plugin_scrobbling_info": "Tämä lisäosa scrobblaa musiikkisi luodakseen kuunteluhistoriasi.", + "default_plugin": "Oletus", + "set_default": "Aseta oletukseksi", + "support": "Tuki", + "support_plugin_development": "Tue lisäosan kehitystä", + "can_access_name_api": "- Voi käyttää **{name}** APIa", + "do_you_want_to_install_this_plugin": "Haluatko asentaa tämän lisäosan?", + "third_party_plugin_warning": "Tämä lisäosa on kolmannen osapuolen arkistosta. Varmista, että luotat lähteeseen ennen asennusta.", + "author": "Tekijä", + "this_plugin_can_do_following": "Tämä lisäosa voi tehdä seuraavaa", + "install": "Asenna", + "install_a_metadata_provider": "Asenna metatietojen tarjoaja", + "no_tracks_playing": "Ei kappaletta toistossa tällä hetkellä", + "synced_lyrics_not_available": "Synkronoidut sanoitukset eivät ole saatavilla tälle kappaleelle. Käytä sen sijaan", + "plain_lyrics": "Pelkät sanoitukset", + "tab_instead": "välilehteä.", + "disclaimer": "Vastuuvapauslauseke", + "third_party_plugin_dmca_notice": "Spotube-tiimi ei ota mitään vastuuta (mukaan lukien oikeudellinen) mistään \"kolmannen osapuolen\" lisäosista.\nKäytä niitä omalla vastuullasi. Ilmoita kaikista virheistä/ongelmista lisäosan arkistoon.\n\nJos jokin \"kolmannen osapuolen\" lisäosa rikkoo jonkin palvelun/oikeushenkilön käyttöehtoja/DMCA:ta, pyydä \"kolmannen osapuolen\" lisäosan tekijää tai isännöintialustaa, esim. GitHubia/Codebergiä, ryhtymään toimiin. Yllä luetellut (\"kolmannen osapuolen\" merkityt) ovat kaikki julkisia/yhteisön ylläpitämiä lisäosia. Emme kuratoi niitä, joten emme voi ryhtyä niihin toimiin.\n\n", + "input_does_not_match_format": "Syöte ei vastaa vaadittua muotoa", + "metadata_provider_plugins": "Metatietojen tarjoajan lisäosat", + "paste_plugin_download_url": "Liitä lataus-URL-osoite tai GitHub/Codeberg-arkiston URL-osoite tai suora linkki .smplug-tiedostoon", + "download_and_install_plugin_from_url": "Lataa ja asenna lisäosa URL-osoitteesta", + "failed_to_add_plugin_error": "Lisäosan lisääminen epäonnistui: {error}", + "upload_plugin_from_file": "Lataa lisäosa tiedostosta", + "installed": "Asennettu", + "available_plugins": "Saatavilla olevat lisäosat", + "configure_your_own_metadata_plugin": "Määritä oma soittolistan/albumin/artistin/syötteen metatietojen tarjoaja", + "audio_scrobblers": "Äänen scrobblerit", + "scrobbling": "Scrobbling", + "download_music_format": "Musiikin latausmuoto", + "streaming_music_format": "Musiikin suoratoistomuoto", + "download_music_quality": "Musiikin latauslaatu", + "streaming_music_quality": "Musiikin suoratoistolaadun", + "default_metadata_source": "Oletusarvoinen metatietolähde", + "set_default_metadata_source": "Aseta oletusmetatietolähde", + "default_audio_source": "Oletusarvoinen äänilähde", + "set_default_audio_source": "Aseta oletusäänilähde", + "plugins": "Laajennukset", + "configure_plugins": "Määritä omat metatietojen tarjoaja- ja äänilähdelaajennukset", + "source": "Lähde: ", + "uncompressed": "Pakkaamaton", + "dab_music_source_description": "Audiofiileille. Tarjoaa korkealaatuisia/häviöttömiä äänivirtoja. Tarkka ISRC-pohjainen kappaleiden tunnistus." } \ No newline at end of file diff --git a/lib/l10n/app_fr.arb b/lib/l10n/app_fr.arb index 9062ada7..e73c2eb2 100644 --- a/lib/l10n/app_fr.arb +++ b/lib/l10n/app_fr.arb @@ -401,5 +401,95 @@ "export_cache_files": "Exporter les fichiers en cache", "found_n_files": "{count} fichiers trouvés", "export_cache_confirmation": "Voulez-vous exporter ces fichiers vers", - "exported_n_out_of_m_files": "{filesExported} fichiers exportés sur {files}" + "exported_n_out_of_m_files": "{filesExported} fichiers exportés sur {files}", + "playlist": "Playlist", + "no_loop": "Pas de boucle", + "generate": "Générer", + "undo": "Annuler", + "download_all": "Télécharger tout", + "add_all_to_playlist": "Ajouter tout à la playlist", + "add_all_to_queue": "Ajouter tout à la file d'attente", + "play_all_next": "Lire tout suivant", + "pause": "Pause", + "view_all": "Voir tout", + "no_tracks_added_yet": "Il semble que vous n'avez encore ajouté aucun morceau.", + "no_tracks": "Il semble qu'il n'y ait pas de morceaux ici.", + "no_tracks_listened_yet": "Il semble que vous n'avez encore rien écouté.", + "not_following_artists": "Vous ne suivez aucun artiste.", + "no_favorite_albums_yet": "Il semble que vous n'ayez encore ajouté aucun album à vos favoris.", + "no_logs_found": "Aucun log trouvé", + "youtube_engine": "Moteur YouTube", + "youtube_engine_not_installed_title": "{engine} n'est pas installé", + "youtube_engine_not_installed_message": "{engine} n'est pas installé sur votre système.", + "youtube_engine_set_path": "Assurez-vous qu'il est disponible dans la variable PATH ou\nfixez le chemin absolu du fichier exécutable {engine} ci-dessous.", + "youtube_engine_unix_issue_message": "Dans macOS/Linux/les systèmes d'exploitation similaires à Unix, définir le chemin dans .zshrc/.bashrc/.bash_profile etc. ne fonctionnera pas.\nVous devez définir le chemin dans le fichier de configuration du shell.", + "download": "Télécharger", + "file_not_found": "Fichier non trouvé", + "custom": "Personnalisé", + "add_custom_url": "Ajouter une URL personnalisée", + "edit_port": "Modifier le port", + "port_helper_msg": "La valeur par défaut est -1, ce qui indique un nombre aléatoire. Si vous avez configuré un pare-feu, il est recommandé de le définir.", + "connect_request": "Autoriser {client} à se connecter ?", + "connection_request_denied ": "Connexion refusée. L'utilisateur a refusé l'accès.", + "hipotetical_calculation": "*Ce calcul est basé sur le paiement moyen par lecture des plateformes de streaming musical en ligne, de 0,003 $ à 0,005 $. Il s'agit d'un calcul hypothétique pour donner à l'utilisateur un aperçu de ce qu'il aurait payé aux artistes s'il écoutait leur chanson sur différentes plateformes de streaming musical.", + "connection_request_denied": "Connexion refusée. L'utilisateur a refusé l'accès.", + "an_error_occurred": "Une erreur est survenue", + "copy_to_clipboard": "Copier dans le presse-papiers", + "view_logs": "Afficher les journaux", + "retry": "Réessayer", + "no_default_metadata_provider_selected": "Vous n'avez pas de fournisseur de métadonnées par défaut", + "manage_metadata_providers": "Gérer les fournisseurs de métadonnées", + "open_link_in_browser": "Ouvrir le lien dans le navigateur ?", + "do_you_want_to_open_the_following_link": "Voulez-vous ouvrir le lien suivant", + "unsafe_url_warning": "L'ouverture de liens provenant de sources non fiables peut être dangereuse. Soyez prudent !\nVous pouvez également copier le lien dans votre presse-papiers.", + "copy_link": "Copier le lien", + "building_your_timeline": "Construction de votre chronologie en fonction de vos écoutes...", + "official": "Officiel", + "author_name": "Auteur : {author}", + "third_party": "Tiers", + "plugin_requires_authentication": "Le plugin nécessite une authentification", + "update_available": "Mise à jour disponible", + "supports_scrobbling": "Supporte le scrobbling", + "plugin_scrobbling_info": "Ce plugin scrobble votre musique pour générer votre historique d'écoute.", + "default_plugin": "Par défaut", + "set_default": "Définir par défaut", + "support": "Soutien", + "support_plugin_development": "Soutenir le développement de plugins", + "can_access_name_api": "- Peut accéder à l'API **{name}**", + "do_you_want_to_install_this_plugin": "Voulez-vous installer ce plugin ?", + "third_party_plugin_warning": "Ce plugin provient d'un dépôt tiers. Assurez-vous de faire confiance à la source avant de l'installer.", + "author": "Auteur", + "this_plugin_can_do_following": "Ce plugin peut faire ce qui suit", + "install": "Installer", + "install_a_metadata_provider": "Installer un fournisseur de métadonnées", + "no_tracks_playing": "Aucune piste n'est en cours de lecture actuellement", + "synced_lyrics_not_available": "Les paroles synchronisées ne sont pas disponibles pour cette chanson. Veuillez utiliser l'onglet", + "plain_lyrics": "Paroles simples", + "tab_instead": "à la place.", + "disclaimer": "Avertissement", + "third_party_plugin_dmca_notice": "L'équipe de Spotube n'assume aucune responsabilité (y compris juridique) pour les plugins \"tiers\".\nVeuillez les utiliser à vos propres risques. Pour tout bug/problème, veuillez le signaler au dépôt du plugin.\n\nSi un plugin \"tiers\" enfreint les conditions d'utilisation/DMCA d'un service/entité juridique, veuillez demander à l'auteur du plugin \"tiers\" ou à la plateforme d'hébergement (par exemple GitHub/Codeberg) de prendre des mesures. Les plugins listés ci-dessus (étiquetés \"tiers\") sont tous des plugins publics/maintenus par la communauté. Nous ne les gérons pas, nous ne pouvons donc prendre aucune mesure à leur sujet.\n\n", + "input_does_not_match_format": "L'entrée ne correspond pas au format requis", + "metadata_provider_plugins": "Plugins de fournisseur de métadonnées", + "paste_plugin_download_url": "Collez l'URL de téléchargement ou l'URL du dépôt GitHub/Codeberg ou un lien direct vers le fichier .smplug", + "download_and_install_plugin_from_url": "Télécharger et installer le plugin à partir de l'URL", + "failed_to_add_plugin_error": "Échec de l'ajout du plugin : {error}", + "upload_plugin_from_file": "Télécharger le plugin à partir d'un fichier", + "installed": "Installé", + "available_plugins": "Plugins disponibles", + "configure_your_own_metadata_plugin": "Configurer votre propre fournisseur de métadonnées de playlist/album/artiste/flux", + "audio_scrobblers": "Scrobblers audio", + "scrobbling": "Scrobbling", + "download_music_format": "Format de téléchargement de musique", + "streaming_music_format": "Format de streaming de musique", + "download_music_quality": "Qualité de téléchargement de musique", + "streaming_music_quality": "Qualité de streaming de musique", + "default_metadata_source": "Source de métadonnées par défaut", + "set_default_metadata_source": "Définir la source de métadonnées par défaut", + "default_audio_source": "Source audio par défaut", + "set_default_audio_source": "Définir la source audio par défaut", + "plugins": "Plugins", + "configure_plugins": "Configurez vos propres plugins de fournisseur de métadonnées et de source audio", + "source": "Source : ", + "uncompressed": "Non compressé", + "dab_music_source_description": "Pour les audiophiles. Fournit des flux audio de haute qualité/sans perte. Correspondance précise des pistes basée sur ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_hi.arb b/lib/l10n/app_hi.arb index 7a1eae4e..8e8087bb 100644 --- a/lib/l10n/app_hi.arb +++ b/lib/l10n/app_hi.arb @@ -401,5 +401,94 @@ "export_cache_files": "कैश फ़ाइलें निर्यात करें", "found_n_files": "{count} फ़ाइलें मिलीं", "export_cache_confirmation": "क्या आप इन फ़ाइलों को निर्यात करना चाहते हैं", - "exported_n_out_of_m_files": "{filesExported} फ़ाइलें निर्यात की गईं {files} में से" + "exported_n_out_of_m_files": "{filesExported} फ़ाइलें निर्यात की गईं {files} में से", + "playlist": "प्लेलिस्ट", + "no_loop": "कोई लूप नहीं", + "generate": "उत्पन्न करें", + "undo": "पूर्ववत करें", + "download_all": "सभी डाउनलोड करें", + "add_all_to_playlist": "सभी को प्लेलिस्ट में जोड़ें", + "add_all_to_queue": "सभी को कतार में जोड़ें", + "play_all_next": "सभी को अगले खेलने के लिए", + "pause": "रोकें", + "view_all": "सभी देखें", + "no_tracks_added_yet": "लगता है आपने अभी तक कोई ट्रैक नहीं जोड़ा है।", + "no_tracks": "लगता है यहाँ कोई ट्रैक नहीं है।", + "no_tracks_listened_yet": "लगता है आपने अभी तक कुछ नहीं सुना है।", + "not_following_artists": "आप किसी भी कलाकार को फॉलो नहीं कर रहे हैं।", + "no_favorite_albums_yet": "लगता है आपने अभी तक कोई एल्बम अपनी पसंदीदा सूची में नहीं जोड़ा है।", + "no_logs_found": "कोई लॉग नहीं मिला", + "youtube_engine": "YouTube इंजन", + "youtube_engine_not_installed_title": "{engine} स्थापित नहीं है", + "youtube_engine_not_installed_message": "{engine} आपके सिस्टम में स्थापित नहीं है।", + "youtube_engine_set_path": "यह सुनिश्चित करें कि यह PATH वेरिएबल में उपलब्ध हो या\nनीचे {engine} निष्पादन योग्य फ़ाइल का पूर्ण पथ सेट करें।", + "youtube_engine_unix_issue_message": "macOS/Linux/यूनिक्स जैसे OS में, .zshrc/.bashrc/.bash_profile आदि में पथ सेट करना काम नहीं करेगा।\nआपको पथ को शेल कॉन्फ़िगरेशन फ़ाइल में सेट करना होगा।", + "download": "डाउनलोड करें", + "file_not_found": "फाइल नहीं मिली", + "custom": "कस्टम", + "add_custom_url": "कस्टम URL जोड़ें", + "edit_port": "पोर्ट संपादित करें", + "port_helper_msg": "डिफ़ॉल्ट -1 है जो यादृच्छिक संख्या को दर्शाता है। यदि आपने फ़ायरवॉल कॉन्फ़िगर किया है, तो इसे सेट करना अनुशंसित है।", + "connect_request": "{client} को कनेक्ट करने की अनुमति दें?", + "connection_request_denied": "कनेक्शन अस्वीकृत। उपयोगकर्ता ने पहुंच अस्वीकृत कर दी।", + "hipotetical_calculation": "*यह औसत ऑनलाइन संगीत स्ट्रीमिंग प्लेटफ़ॉर्म के प्रति स्ट्रीम भुगतान ($0.003 से $0.005) के आधार पर गणना की गई है। यह एक काल्पनिक गणना है जो उपयोगकर्ता को यह जानकारी देने के लिए है कि यदि वे विभिन्न संगीत स्ट्रीमिंग प्लेटफ़ॉर्म पर अपने गाने सुनते हैं तो उन्होंने कलाकारों को कितना भुगतान किया होगा।", + "an_error_occurred": "एक त्रुटि हुई", + "copy_to_clipboard": "क्लिपबोर्ड पर कॉपी करें", + "view_logs": "लॉग देखें", + "retry": "पुनः प्रयास करें", + "no_default_metadata_provider_selected": "आपने कोई डिफ़ॉल्ट मेटाडेटा प्रदाता सेट नहीं किया है", + "manage_metadata_providers": "मेटाडेटा प्रदाताओं को प्रबंधित करें", + "open_link_in_browser": "ब्राउज़र में लिंक खोलें?", + "do_you_want_to_open_the_following_link": "क्या आप निम्नलिखित लिंक खोलना चाहते हैं", + "unsafe_url_warning": "अविश्वसनीय स्रोतों से लिंक खोलना असुरक्षित हो सकता है। सावधान रहें!\nआप लिंक को अपने क्लिपबोर्ड पर भी कॉपी कर सकते हैं।", + "copy_link": "लिंक कॉपी करें", + "building_your_timeline": "आपकी सुनने की आदतों के आधार पर आपकी टाइमलाइन बनाई जा रही है...", + "official": "आधिकारिक", + "author_name": "लेखक: {author}", + "third_party": "तृतीय-पक्ष", + "plugin_requires_authentication": "प्लगइन को प्रमाणीकरण की आवश्यकता है", + "update_available": "अपडेट उपलब्ध है", + "supports_scrobbling": "स्क्रॉबलिंग का समर्थन करता है", + "plugin_scrobbling_info": "यह प्लगइन आपके सुनने के इतिहास को उत्पन्न करने के लिए आपके संगीत को स्क्रॉबल करता है।", + "default_plugin": "डिफ़ॉल्ट", + "set_default": "डिफ़ॉल्ट सेट करें", + "support": "समर्थन", + "support_plugin_development": "प्लगइन विकास का समर्थन करें", + "can_access_name_api": "- **{name}** API तक पहुंच सकता है", + "do_you_want_to_install_this_plugin": "क्या आप इस प्लगइन को स्थापित करना चाहते हैं?", + "third_party_plugin_warning": "यह प्लगइन एक तृतीय-पक्ष रिपॉजिटरी से है। कृपया सुनिश्चित करें कि आप इसे स्थापित करने से पहले स्रोत पर भरोसा करते हैं।", + "author": "लेखक", + "this_plugin_can_do_following": "यह प्लगइन निम्नलिखित कर सकता है", + "install": "स्थापित करें", + "install_a_metadata_provider": "एक मेटाडेटा प्रदाता स्थापित करें", + "no_tracks_playing": "वर्तमान में कोई ट्रैक नहीं चल रहा है", + "synced_lyrics_not_available": "इस गाने के लिए सिंक्रनाइज़ किए गए बोल उपलब्ध नहीं हैं। कृपया", + "plain_lyrics": "सादे बोल", + "tab_instead": "टैब का उपयोग करें।", + "disclaimer": "अस्वीकरण", + "third_party_plugin_dmca_notice": "स्पॉट्यूब टीम किसी भी \"तृतीय-पक्ष\" प्लगइन के लिए कोई जिम्मेदारी (कानूनी सहित) नहीं लेती है।\nकृपया उन्हें अपने जोखिम पर उपयोग करें। किसी भी बग/समस्या के लिए, कृपया उन्हें प्लगइन रिपॉजिटरी को रिपोर्ट करें।\n\nयदि कोई \"तृतीय-पक्ष\" प्लगइन किसी सेवा/कानूनी इकाई के ToS/DMCA को तोड़ रहा है, तो कृपया \"तृतीय-पक्ष\" प्लगइन लेखक या होस्टिंग प्लेटफ़ॉर्म जैसे GitHub/Codeberg से कार्रवाई करने के लिए कहें। ऊपर सूचीबद्ध (\"तृतीय-पक्ष\" लेबल वाले) सभी सार्वजनिक/समुदाय-द्वारा-रखरखाव किए गए प्लगइन हैं। हम उन्हें क्यूरेट नहीं कर रहे हैं, इसलिए हम उन पर कोई कार्रवाई नहीं कर सकते हैं।\n\n", + "input_does_not_match_format": "इनपुट आवश्यक प्रारूप से मेल नहीं खाता है", + "metadata_provider_plugins": "मेटाडेटा प्रदाता प्लगइन", + "paste_plugin_download_url": "डाउनलोड यूआरएल या गिटहब/कोडबर्ग रेपो यूआरएल या .smplug फ़ाइल का सीधा लिंक पेस्ट करें", + "download_and_install_plugin_from_url": "यूआरएल से प्लगइन डाउनलोड और स्थापित करें", + "failed_to_add_plugin_error": "प्लगइन जोड़ने में विफल: {error}", + "upload_plugin_from_file": "फ़ाइल से प्लगइन अपलोड करें", + "installed": "स्थापित", + "available_plugins": "उपलब्ध प्लगइन", + "configure_your_own_metadata_plugin": "अपनी खुद की प्लेलिस्ट/एल्बम/कलाकार/फ़ीड मेटाडेटा प्रदाता कॉन्फ़िगर करें", + "audio_scrobblers": "ऑडियो स्क्रॉबलर्स", + "scrobbling": "स्क्रॉबलिंग", + "download_music_format": "संगीत डाउनलोड प्रारूप", + "streaming_music_format": "संगीत स्ट्रीमिंग प्रारूप", + "download_music_quality": "संगीत डाउनलोड गुणवत्ता", + "streaming_music_quality": "संगीत स्ट्रीमिंग गुणवत्ता", + "default_metadata_source": "डिफ़ॉल्ट मेटाडेटा स्रोत", + "set_default_metadata_source": "डिफ़ॉल्ट मेटाडेटा स्रोत सेट करें", + "default_audio_source": "डिफ़ॉल्ट ऑडियो स्रोत", + "set_default_audio_source": "डिफ़ॉल्ट ऑडियो स्रोत सेट करें", + "plugins": "प्लगइन्स", + "configure_plugins": "अपने स्वयं के मेटाडेटा प्रदाता और ऑडियो स्रोत प्लगइन्स कॉन्फ़िगर करें", + "source": "स्रोत: ", + "uncompressed": "असंपीड़ित", + "dab_music_source_description": "ऑडियोफाइलों के लिए। उच्च-गुणवत्ता/बिना हानि वाले ऑडियो स्ट्रीम प्रदान करता है। सटीक ISRC आधारित ट्रैक मिलान।" } \ No newline at end of file diff --git a/lib/l10n/app_id.arb b/lib/l10n/app_id.arb index 5e041dc0..3405fd2f 100644 --- a/lib/l10n/app_id.arb +++ b/lib/l10n/app_id.arb @@ -401,5 +401,94 @@ "export_cache_files": "Export Cached Files", "found_n_files": "Found {count} files", "export_cache_confirmation": "Do you want to export these files to", - "exported_n_out_of_m_files": "Exported {filesExported} out of {files} files" + "exported_n_out_of_m_files": "Exported {filesExported} out of {files} files", + "playlist": "Playlist", + "no_loop": "No loop", + "generate": "Generate", + "undo": "Undo", + "download_all": "Download all", + "add_all_to_playlist": "Add all to playlist", + "add_all_to_queue": "Add all to queue", + "play_all_next": "Play all next", + "pause": "Pause", + "view_all": "View all", + "no_tracks_added_yet": "Looks like you haven't added any tracks yet", + "no_tracks": "Looks like there are no tracks here", + "no_tracks_listened_yet": "Looks like you haven't listened to anything yet", + "not_following_artists": "You're not following any artists", + "no_favorite_albums_yet": "Looks like you haven't added any albums to your favorites yet", + "no_logs_found": "No logs found", + "youtube_engine": "YouTube Engine", + "youtube_engine_not_installed_title": "{engine} is not installed", + "youtube_engine_not_installed_message": "{engine} is not installed in your system.", + "youtube_engine_set_path": "Make sure it's available in the PATH variable or\nset the absolute path to the {engine} executable below", + "youtube_engine_unix_issue_message": "In macOS/Linux/unix like OS's, setting path on .zshrc/.bashrc/.bash_profile etc. won't work.\nYou need to set the path in the shell configuration file", + "download": "Download", + "file_not_found": "File not found", + "custom": "Custom", + "add_custom_url": "Add custom URL", + "edit_port": "Edit port", + "port_helper_msg": "Default adalah -1 yang menunjukkan angka acak. Jika Anda telah mengonfigurasi firewall, disarankan untuk mengatur ini.", + "connect_request": "Izinkan {client} untuk terhubung?", + "connection_request_denied": "Koneksi ditolak. Pengguna menolak akses.", + "hipotetical_calculation": "*Ini dihitung berdasarkan pembayaran rata-rata per streaming dari platform streaming musik online sebesar $0,003 hingga $0,005. Ini adalah perhitungan hipotetis untuk memberikan wawasan kepada pengguna tentang seberapa banyak yang akan mereka bayarkan kepada artis jika mereka mendengarkan lagu mereka di platform streaming musik yang berbeda.", + "an_error_occurred": "Terjadi kesalahan", + "copy_to_clipboard": "Salin ke papan klip", + "view_logs": "Lihat log", + "retry": "Coba lagi", + "no_default_metadata_provider_selected": "Anda belum mengatur penyedia metadata default", + "manage_metadata_providers": "Kelola penyedia metadata", + "open_link_in_browser": "Buka Tautan di Peramban?", + "do_you_want_to_open_the_following_link": "Apakah Anda ingin membuka tautan berikut", + "unsafe_url_warning": "Tidak aman untuk membuka tautan dari sumber yang tidak tepercaya. Berhati-hatilah!\nAnda juga dapat menyalin tautan ke papan klip Anda.", + "copy_link": "Salin Tautan", + "building_your_timeline": "Membangun garis waktu Anda berdasarkan riwayat mendengarkan Anda...", + "official": "Resmi", + "author_name": "Penulis: {author}", + "third_party": "Pihak ketiga", + "plugin_requires_authentication": "Plugin memerlukan otentikasi", + "update_available": "Pembaruan tersedia", + "supports_scrobbling": "Mendukung scrobbling", + "plugin_scrobbling_info": "Plugin ini scrobble musik Anda untuk menghasilkan riwayat mendengarkan Anda.", + "default_plugin": "Bawaan", + "set_default": "Atur sebagai bawaan", + "support": "Dukungan", + "support_plugin_development": "Dukung pengembangan plugin", + "can_access_name_api": "- Dapat mengakses API **{name}**", + "do_you_want_to_install_this_plugin": "Apakah Anda ingin menginstal plugin ini?", + "third_party_plugin_warning": "Plugin ini berasal dari repositori pihak ketiga. Pastikan Anda memercayai sumbernya sebelum menginstal.", + "author": "Penulis", + "this_plugin_can_do_following": "Plugin ini dapat melakukan hal berikut", + "install": "Instal", + "install_a_metadata_provider": "Instal Penyedia Metadata", + "no_tracks_playing": "Tidak ada Lagu yang sedang diputar saat ini", + "synced_lyrics_not_available": "Lirik tersinkronisasi tidak tersedia untuk lagu ini. Silakan gunakan tab", + "plain_lyrics": "Lirik Polos", + "tab_instead": "sebagai gantinya.", + "disclaimer": "Penafian", + "third_party_plugin_dmca_notice": "Tim Spotube tidak bertanggung jawab (termasuk hukum) atas plugin \"Pihak ketiga\" mana pun.\nSilakan gunakan dengan risiko Anda sendiri. Untuk bug/masalah apa pun, silakan laporkan ke repositori plugin.\n\nJika ada plugin \"Pihak ketiga\" yang melanggar ToS/DMCA dari layanan/entitas hukum mana pun, silakan minta penulis plugin \"Pihak ketiga\" atau platform hosting, mis. GitHub/Codeberg, untuk mengambil tindakan. Yang tercantum di atas (berlabel \"Pihak ketiga\") adalah semua plugin publik/yang dikelola oleh komunitas. Kami tidak mengkurasi mereka, jadi kami tidak dapat mengambil tindakan apa pun terhadap mereka.\n\n", + "input_does_not_match_format": "Masukan tidak cocok dengan format yang diperlukan", + "metadata_provider_plugins": "Plugin Penyedia Metadata", + "paste_plugin_download_url": "Tempel url unduhan atau url repo GitHub/Codeberg atau tautan langsung ke file .smplug", + "download_and_install_plugin_from_url": "Unduh dan instal plugin dari url", + "failed_to_add_plugin_error": "Gagal menambahkan plugin: {error}", + "upload_plugin_from_file": "Unggah plugin dari file", + "installed": "Terinstal", + "available_plugins": "Plugin yang tersedia", + "configure_your_own_metadata_plugin": "Konfigurasi penyedia metadata playlist/album/artis/feed Anda sendiri", + "audio_scrobblers": "Scrobblers Audio", + "scrobbling": "Scrobbling", + "download_music_format": "Format unduh musik", + "streaming_music_format": "Format streaming musik", + "download_music_quality": "Kualitas unduh musik", + "streaming_music_quality": "Kualitas streaming musik", + "default_metadata_source": "Sumber metadata default", + "set_default_metadata_source": "Atur sumber metadata default", + "default_audio_source": "Sumber audio default", + "set_default_audio_source": "Atur sumber audio default", + "plugins": "Plugin", + "configure_plugins": "Konfigurasi plugin penyedia metadata dan sumber audio Anda sendiri", + "source": "Sumber: ", + "uncompressed": "Tidak terkompresi", + "dab_music_source_description": "Untuk audiophile. Menyediakan aliran audio berkualitas tinggi/tanpa kehilangan. Pencocokkan trek yang akurat berdasarkan ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_it.arb b/lib/l10n/app_it.arb index c4954dd1..c544dbf3 100644 --- a/lib/l10n/app_it.arb +++ b/lib/l10n/app_it.arb @@ -402,5 +402,94 @@ "export_cache_files": "Esporta file nella cache", "found_n_files": "Trovati {count} file", "export_cache_confirmation": "Vuoi esportare questi file su", - "exported_n_out_of_m_files": "Esportati {filesExported} su {files} file" + "exported_n_out_of_m_files": "Esportati {filesExported} su {files} file", + "playlist": "Playlist", + "no_loop": "Nessun ciclo", + "generate": "Genera", + "undo": "Annulla", + "download_all": "Scarica tutto", + "add_all_to_playlist": "Aggiungi tutto alla playlist", + "add_all_to_queue": "Aggiungi tutto alla coda", + "play_all_next": "Riproduci tutto dopo", + "pause": "Pausa", + "view_all": "Vedi tutto", + "no_tracks_added_yet": "Sembra che non hai ancora aggiunto nessun brano", + "no_tracks": "Sembra che non ci siano brani qui", + "no_tracks_listened_yet": "Sembra che non hai ascoltato nulla ancora", + "not_following_artists": "Non stai seguendo alcun artista", + "no_favorite_albums_yet": "Sembra che non hai ancora aggiunto album ai tuoi preferiti", + "no_logs_found": "Nessun registro trovato", + "youtube_engine": "Motore YouTube", + "youtube_engine_not_installed_title": "{engine} non è installato", + "youtube_engine_not_installed_message": "{engine} non è installato nel tuo sistema.", + "youtube_engine_set_path": "Assicurati che sia disponibile nella variabile PATH o\nimposta il percorso assoluto all'eseguibile {engine} qui sotto", + "youtube_engine_unix_issue_message": "In macOS/Linux/os simili a unix, impostare il percorso su .zshrc/.bashrc/.bash_profile ecc. non funzionerà.\nDevi impostare il percorso nel file di configurazione della shell", + "download": "Scarica", + "file_not_found": "File non trovato", + "custom": "Personalizzato", + "add_custom_url": "Aggiungi URL personalizzato", + "edit_port": "Modifica porta", + "port_helper_msg": "Il valore predefinito è -1, che indica un numero casuale. Se hai configurato un firewall, si consiglia di impostarlo.", + "connect_request": "Consentire a {client} di connettersi?", + "connection_request_denied": "Connessione negata. L'utente ha negato l'accesso.", + "hipotetical_calculation": "*Questo è calcolato in base al pagamento medio per stream delle piattaforme di streaming musicale online, che va da $0.003 a $0.005. Si tratta di un calcolo ipotetico per dare all'utente un'idea di quanto avrebbe pagato agli artisti se avesse ascoltato la loro canzone su diverse piattaforme di streaming musicale.", + "an_error_occurred": "Si è verificato un errore", + "copy_to_clipboard": "Copia negli appunti", + "view_logs": "Visualizza log", + "retry": "Riprova", + "no_default_metadata_provider_selected": "Non hai impostato alcun provider di metadati predefinito", + "manage_metadata_providers": "Gestisci provider di metadati", + "open_link_in_browser": "Aprire il link nel browser?", + "do_you_want_to_open_the_following_link": "Vuoi aprire il seguente link", + "unsafe_url_warning": "Potrebbe essere pericoloso aprire link da fonti non attendibili. Sii cauto!\nPuoi anche copiare il link negli appunti.", + "copy_link": "Copia link", + "building_your_timeline": "Creazione della tua cronologia in base ai tuoi ascolti...", + "official": "Ufficiale", + "author_name": "Autore: {author}", + "third_party": "Terze parti", + "plugin_requires_authentication": "Il plugin richiede l'autenticazione", + "update_available": "Aggiornamento disponibile", + "supports_scrobbling": "Supporta lo scrobbling", + "plugin_scrobbling_info": "Questo plugin scrobbla la tua musica per generare la tua cronologia di ascolti.", + "default_plugin": "Predefinito", + "set_default": "Imposta come predefinito", + "support": "Supporto", + "support_plugin_development": "Sostieni lo sviluppo del plugin", + "can_access_name_api": "- Può accedere all'API **{name}**", + "do_you_want_to_install_this_plugin": "Vuoi installare questo plugin?", + "third_party_plugin_warning": "Questo plugin proviene da un repository di terze parti. Assicurati di fidarti della fonte prima di installarlo.", + "author": "Autore", + "this_plugin_can_do_following": "Questo plugin può fare quanto segue", + "install": "Installa", + "install_a_metadata_provider": "Installa un provider di metadati", + "no_tracks_playing": "Nessun brano in riproduzione al momento", + "synced_lyrics_not_available": "Testi sincronizzati non disponibili per questa canzone. Si prega di utilizzare la scheda", + "plain_lyrics": "Testi semplici", + "tab_instead": "invece.", + "disclaimer": "Disclaimer", + "third_party_plugin_dmca_notice": "Il team di Spotube non si assume alcuna responsabilità (anche legale) per i plugin di \"terze parti\".\nUsali a tuo rischio e pericolo. Per eventuali bug/problemi, segnalali al repository del plugin.\n\nSe un plugin di \"terze parti\" sta violando i ToS/DMCA di un servizio/entità legale, per favore chiedi all'autore del plugin \"terzo\" o alla piattaforma di hosting, ad esempio GitHub/Codeberg, di agire. Quelli elencati sopra (etichettati come \"terze parti\") sono tutti plugin pubblici/mantenuti dalla comunità. Non li curiamo, quindi non possiamo intraprendere alcuna azione su di essi.\n\n", + "input_does_not_match_format": "L'input non corrisponde al formato richiesto", + "metadata_provider_plugins": "Plugin del provider di metadati", + "paste_plugin_download_url": "Incolla l'URL di download o l'URL del repository GitHub/Codeberg o il link diretto al file .smplug", + "download_and_install_plugin_from_url": "Scarica e installa il plugin da URL", + "failed_to_add_plugin_error": "Impossibile aggiungere il plugin: {error}", + "upload_plugin_from_file": "Carica plugin da file", + "installed": "Installato", + "available_plugins": "Plugin disponibili", + "configure_your_own_metadata_plugin": "Configura il tuo provider di metadati per playlist/album/artista/feed", + "audio_scrobblers": "Scrobbler audio", + "scrobbling": "Scrobbling", + "download_music_format": "Formato download musica", + "streaming_music_format": "Formato streaming musica", + "download_music_quality": "Qualità download musica", + "streaming_music_quality": "Qualità streaming musica", + "default_metadata_source": "Fonte metadati predefinita", + "set_default_metadata_source": "Imposta fonte metadati predefinita", + "default_audio_source": "Fonte audio predefinita", + "set_default_audio_source": "Imposta fonte audio predefinita", + "plugins": "Plugin", + "configure_plugins": "Configura i tuoi plugin per fornitore metadati e fonte audio", + "source": "Fonte: ", + "uncompressed": "Non compresso", + "dab_music_source_description": "Per audiophile. Fornisce flussi audio di alta qualità/senza perdita. Abbinamento traccia accurato basato su ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_ja.arb b/lib/l10n/app_ja.arb index 4f299025..991f56be 100644 --- a/lib/l10n/app_ja.arb +++ b/lib/l10n/app_ja.arb @@ -22,7 +22,7 @@ "filter_playlists": "あなたの再生リストを絞り込み...", "liked_tracks": "いいねした曲", "liked_tracks_description": "いいねしたすべての曲", - "create_playlist": "再生リストの作成", + "playlist": "再生リスト", "create_a_playlist": "再生リストの作成", "create": "作成", "cancel": "キャンセル", @@ -39,8 +39,9 @@ "sort_z_a": "Z-A 順に並び替え", "sort_artist": "アーティスト順に並び替え", "sort_album": "アルバム順に並び替え", + "sort_duration": "長さ順に並べ替え", "sort_tracks": "曲の並び替え", - "currently_downloading": "いまダウンロード中 ({tracks_length}) 曲", + "currently_downloading": "ダウンロード中 ({tracks_length}) 曲", "cancel_all": "すべてキャンセル", "filter_artist": "アーティストを絞り込み...", "followers": "{followers} フォロワー", @@ -94,6 +95,7 @@ "pause_playback": "再生を停止", "resume_playback": "再生を再開", "loop_track": "曲をループ", + "no_loop": "ループなし", "repeat_playlist": "再生リストをリピート", "queue": "再生キュー", "alternative_track_sources": "この曲の別の音源を選ぶ", @@ -112,8 +114,8 @@ "language_region": "言語 & 地域", "language": "言語", "system_default": "システムの既定値", - "market_place_region": "市場の地域", - "recommendation_country": "推薦先の国", + "market_place_region": "音楽市場の地域", + "recommendation_country": "おすすめの国", "appearance": "外観", "layout_mode": "レイアウトの種類", "override_layout_settings": "レスポンシブなレイアウトの種類の設定を上書きする", @@ -175,13 +177,18 @@ "step_2": "ステップ 2", "step_2_steps": "1. ログインしたら、F12を押すか、マウス右クリック > 調査(検証)でブラウザの開発者ツール (devtools) を開きます。\n2. アプリケーション (Application) タブ (Chrome, Edge, Brave など) またはストレージタブ (Firefox, Palemoon など)\n3. Cookies 欄を選択し、https://accounts.spotify.com の枝を選びます", "step_3": "ステップ 3", + "step_3_steps": "\"sp_dc\" Cookieの値をコピー", "success_emoji": "成功🥳", "success_message": "アカウントへのログインに成功しました。よくできました!", "step_4": "ステップ 4", + "step_4_steps": "コピーした\"sp_dc\"の値を貼り付け", "something_went_wrong": "何か誤りがあります", "piped_instance": "Piped サーバーのインスタンス", "piped_description": "曲の一致に使う Piped サーバーのインスタンス", "piped_warning": "それらの一部ではうまく動作しないこともあります。自己責任で使用してください", + "invidious_instance": "Invidiousサーバーインスタンス", + "invidious_description": "曲の一致に使用するInvidiousサーバーインスタンス", + "invidious_warning": "一部はうまく機能しない可能性があります。自己責任で使用してください", "generate_playlist": "再生リストの生成", "track_exists": "曲 {track} は既に存在します", "replace_downloaded_tracks": "すべてのダウンロード済みの曲を置換", @@ -247,102 +254,100 @@ "developers": "開発", "not_logged_in": "ログインしていません", "search_mode": "検索モード", - "audio_source": "音声ソース", - "ok": "分かりました", + "audio_source": "音声の提供元", + "ok": "OK", "failed_to_encrypt": "暗号化に失敗しました", - "encryption_failed_warning": "Spotubeはデータを安全に保存するために暗号化を使用しています。しかし、失敗しました。したがって、安全でないストレージにフォールバックします\nLinuxを使用している場合は、gnome-keyring、kde-wallet、keepassxcなどのシークレットサービスがインストールされていることを確認してください", + "encryption_failed_warning": "SpoTubeはデータを安全に保存するために暗号化を用いますが、暗号化に失敗しました。このため、安全でない保存領域への保存に切り替えます\nOSがLinuxなら、gnome-keyring、kde-wallet、keepassxcなどの管理ツールがインストールされていることを確認してください", "querying_info": "情報を取得中...", "piped_api_down": "Piped APIがダウンしています", - "piped_down_error_instructions": "Pipedインスタンス{pipedInstance}は現在ダウンしています\n\nインスタンスを変更するか、'APIタイプ'を公式のYouTube APIに変更してください\n\n変更後にアプリを再起動してください", + "piped_down_error_instructions": "Pipedインスタンス {pipedInstance} は現在ダウンしています\n\nインスタンスを変更するか、「APIの種類」を公式のYouTube APIに変更してください\n\n変更後にアプリを再起動してください", "you_are_offline": "現在、オフラインです", "connection_restored": "インターネット接続が復旧しました", - "use_system_title_bar": "システムタイトルバーを使用する", - "update_playlist": "プレイリストを更新", + "use_system_title_bar": "システムのタイトルバーを使う", + "update_playlist": "再生リストを更新", "update": "更新", + "local_library": "端末内ライブラリ", + "add_library_location": "ライブラリに追加", + "remove_library_location": "ライブラリから削除", "crunching_results": "結果を処理中...", "search_to_get_results": "結果を取得するために検索", - "use_amoled_mode": "AMOLEDモードを使用する", - "pitch_dark_theme": "ピッチブラックダートテーマ", - "normalize_audio": "オーディオを正規化する", - "change_cover": "カバーを変更する", - "add_cover": "カバーを追加する", - "restore_defaults": "デフォルト値に戻す", - "download_music_codec": "音楽コーデックをダウンロードする", - "streaming_music_codec": "ストリーミング音楽コーデック", - "login_with_lastfm": "Last.fmでログインする", - "connect": "接続する", - "disconnect_lastfm": "Last.fmから切断する", - "disconnect": "切断する", + "use_amoled_mode": "AMOLEDモードを使用", + "pitch_dark_theme": "ピッチブラック ダークテーマ", + "normalize_audio": "音声を正規化", + "change_cover": "カバーを変更", + "add_cover": "カバーを追加", + "restore_defaults": "設定を初期化", + "download_music_codec": "ダウンロード用の音声コーデック", + "streaming_music_codec": "ストリーミング用の音声コーデック", + "login_with_lastfm": "Last.fmでログイン", + "connect": "接続", + "disconnect_lastfm": "Last.fmから切断", + "disconnect": "切断", "username": "ユーザー名", "password": "パスワード", - "login": "ログインする", - "login_with_your_lastfm": "あなたのLast.fmアカウントでログインする", + "login": "ログイン", + "login_with_your_lastfm": "Last.fmアカウントでログイン", "scrobble_to_lastfm": "Last.fmにスクロブルする", "go_to_album": "アルバムに移動", - "discord_rich_presence": "ディスコードリッチプレゼンス", + "discord_rich_presence": "Discord リッチプレゼンス", "browse_all": "すべてを閲覧", "genres": "ジャンル", "explore_genres": "ジャンルを探索", - "step_3_steps": "\"sp_dc\" Cookieの値をコピー", - "step_4_steps": "コピーした\"sp_dc\"の値を貼り付け", "friends": "友達", - "no_lyrics_available": "申し訳ありませんが、このトラックの歌詞を見つけることができません", - "sort_duration": "時間で並べ替え", + "no_lyrics_available": "すみません、この曲の歌詞が見つかりません", "start_a_radio": "ラジオを開始", "how_to_start_radio": "ラジオをどのように開始しますか?", "replace_queue_question": "現在のキューを置き換えるか、追加しますか?", "endless_playback": "エンドレス再生", - "delete_playlist": "プレイリストを削除", - "delete_playlist_confirmation": "このプレイリストを削除してもよろしいですか?", - "local_tracks": "ローカルトラック", + "delete_playlist": "再生リストを削除", + "delete_playlist_confirmation": "この再生リストを削除しますか?", + "local_tracks": "端末内の曲", + "local_tab": "端末内", "song_link": "曲のリンク", - "skip_this_nonsense": "この愚かなことをスキップ", + "skip_this_nonsense": "こんなことはスキップ", "freedom_of_music": "“音楽の自由”", - "freedom_of_music_palm": "“手のひらの中の音楽の自由”", + "freedom_of_music_palm": "“音楽の自由を思いのままに”", "get_started": "さあ始めましょう", "youtube_source_description": "推奨され、最適に機能します。", - "piped_source_description": "自由に感じますか? YouTubeと同じですが、はるかに無料です。", - "jiosaavn_source_description": "南アジア地域向けの最適です。", + "piped_source_description": "自由を感じる?YouTubeと同じだけど、はるかに自由です。", + "jiosaavn_source_description": "南アジア地域では最適です。", + "invidious_source_description": "Pipedに似ていますが、より利用性があります。", "highest_quality": "最高品質:{quality}", - "select_audio_source": "オーディオソースを選択", - "endless_playback_description": "新しい曲をキューの最後に自動的に追加", + "select_audio_source": "音声の提供元を選択", + "endless_playback_description": "キューの最後に新しい曲を自動で追加", "choose_your_region": "地域を選択", - "choose_your_region_description": "これにより、Spotubeがあなたの場所に適したコンテンツを表示できます。", + "choose_your_region_description": "Spotubeがあなたの地域に適したコンテンツを表示します。", "choose_your_language": "言語を選択してください", - "help_project_grow": "このプロジェクトの成長を支援する", - "help_project_grow_description": "Spotubeはオープンソースプロジェクトです。プロジェクトに貢献したり、バグを報告したり、新しい機能を提案することで、このプロジェクトの成長に貢献できます。", - "contribute_on_github": "GitHubで貢献する", - "donate_on_open_collective": "Open Collectiveで寄付する", + "help_project_grow": "プロジェクトの成長を支援する", + "help_project_grow_description": "SpoTubeはオープンソースプロジェクトです。貢献したり、バグ報告したり、新機能を提案することで、プロジェクトの成長に貢献できます。", + "contribute_on_github": "GitHubで貢献", + "donate_on_open_collective": "Open Collectiveで寄付", "browse_anonymously": "匿名で閲覧する", - "enable_connect": "接続を有効にする", - "enable_connect_description": "他のデバイスからSpotubeを制御する", - "devices": "デバイス", - "select": "選択する", - "connect_client_alert": "{client} によって操作されています", - "this_device": "このデバイス", + "enable_connect": "接続する", + "enable_connect_description": "他の端末からSpotubeを制御する", + "devices": "機器", + "select": "選択", + "connect_client_alert": "{client} から操作されています", + "this_device": "この端末", "remote": "リモート", - "local_library": "ローカルライブラリ", - "add_library_location": "ライブラリに追加", - "remove_library_location": "ライブラリから削除", - "local_tab": "ローカル", "stats": "統計", - "and_n_more": "そして {count} つのアイテム", - "recently_played": "最近再生された", - "browse_more": "もっと見る", + "and_n_more": "さらに {count} 項目", + "recently_played": "最近聴いた曲", + "browse_more": "もっと表示", "no_title": "タイトルなし", - "not_playing": "再生中ではありません", - "epic_failure": "壮大な失敗!", + "not_playing": "再生なし", + "epic_failure": "壮大なエラー!", "added_num_tracks_to_queue": "{tracks_length} 曲をキューに追加しました", - "spotube_has_an_update": "Spotube にアップデートがあります", + "spotube_has_an_update": "Spotube の最新版あり", "download_now": "今すぐダウンロード", "nightly_version": "Spotube Nightly {nightlyBuildNum} がリリースされました", "release_version": "Spotube v{version} がリリースされました", "read_the_latest": "最新の ", - "release_notes": "リリースノート", - "pick_color_scheme": "カラースキームを選択", + "release_notes": "更新情報を読む", + "pick_color_scheme": "カラーテーマを選択", "save": "保存", - "choose_the_device": "デバイスを選択:", - "multiple_device_connected": "複数のデバイスが接続されています。\nこのアクションを実行するデバイスを選択してください", + "choose_the_device": "端末を選択:", + "multiple_device_connected": "複数の端末が接続されています。\nこの操作を実行する端末を選択", "nothing_found": "何も見つかりませんでした", "the_box_is_empty": "ボックスは空です", "top_artists": "トップアーティスト", @@ -357,7 +362,7 @@ "email": "メール", "profile_followers": "フォロワー", "birthday": "誕生日", - "subscription": "サブスクリプション", + "subscription": "登録", "not_born": "未出生", "hacker": "ハッカー", "profile": "プロフィール", @@ -365,33 +370,29 @@ "edit": "編集", "user_profile": "ユーザープロフィール", "count_plays": "{count} 回再生", - "streaming_fees_hypothetical": "*これは Spotify のストリームあたりの支払い\nが $0.003 から $0.005 であると仮定して計算されています。\nこれは、Spotify でその曲を聴いた場合にアーティストにいくら支払ったかの\n洞察を得るための仮定の計算です。", - "count_mins": "{minutes} 分", - "summary_minutes": "分", - "summary_listened_to_music": "音楽を聴いた", - "summary_songs": "曲", - "summary_streamed_overall": "全体のストリーミング", - "summary_owed_to_artists": "今月アーティストに支払うべき額", - "summary_artists": "アーティストの", - "summary_music_reached_you": "音楽があなたに届いた", - "summary_full_albums": "フルアルバム", - "summary_got_your_love": "あなたの愛を受け取った", - "summary_playlists": "プレイリスト", - "summary_were_on_repeat": "リピートしていた", - "total_money": "合計 {money}", - "minutes_listened": "リスニング時間", + "streaming_fees_hypothetical": "ストリーミング料金 (概算)", + "minutes_listened": "視聴時間", "streamed_songs": "ストリーミングされた曲", "count_streams": "{count} 回のストリーム", "owned_by_you": "あなたが所有", "copied_shareurl_to_clipboard": "{shareUrl} をクリップボードにコピーしました", - "spotify_hipotetical_calculation": "*これは、Spotifyのストリームごとの支払い\nが $0.003 から $0.005 の範囲で計算されています。これは仮想的な\n計算で、Spotify で曲を聴いた場合に、アーティストに\nどれくらい支払ったかをユーザーに示すためのものです。", + "spotify_hipotetical_calculation": "*これは、Spotifyのストリームあたり\n$0.003 から $0.005 として計算されています。\n概算であり、Spotify で曲を聴いていたら、アーティストに\nどれくらい支払われたかを示すものです。", + "count_mins": "{minutes} 分", + "summary_minutes": "分", + "summary_listened_to_music": "音楽を聴いた", + "summary_songs": "曲", + "summary_streamed_overall": "まるごと聴いた", + "summary_owed_to_artists": "今月アーティストに払う\nべき額", + "summary_artists": "アーティスト", + "summary_music_reached_you": "の音楽が届いた", + "summary_full_albums": "フルアルバム", + "summary_got_your_love": "があなたの愛を受け取った", + "summary_playlists": "再生リスト", + "summary_were_on_repeat": "をリピートしました", + "total_money": "計 {money}", "webview_not_found": "Webviewが見つかりません", - "webview_not_found_description": "デバイスにWebviewランタイムがインストールされていません。\nインストールされている場合は、environment PATHにあることを確認してください\n\nインストール後、アプリを再起動してください", - "unsupported_platform": "サポートされていないプラットフォーム", - "invidious_instance": "Invidiousサーバーインスタンス", - "invidious_description": "トラックマッチングに使用するInvidiousサーバーインスタンス", - "invidious_warning": "一部はうまく機能しない可能性があります。自己責任で使用してください", - "invidious_source_description": "Pipedに似ていますが、より高い可用性があります。", + "webview_not_found_description": "端末にWebviewランタイムがインストールされていません。\nインストールされている場合は、環境変数のパスにあるか確認してください\n\nインストール後、アプリを再起動してください", + "unsupported_platform": "未対応のプラットフォーム", "cache_music": "音楽をキャッシュ", "open": "開く", "cache_folder": "キャッシュフォルダー", @@ -401,5 +402,92 @@ "export_cache_files": "キャッシュされたファイルをエクスポート", "found_n_files": "{count}ファイルが見つかりました", "export_cache_confirmation": "これらのファイルをエクスポートしますか", - "exported_n_out_of_m_files": "{filesExported} / {files}ファイルがエクスポートされました" + "exported_n_out_of_m_files": "{filesExported} / {files}ファイルがエクスポートされました", + "generate": "生成", + "undo": "元に戻す", + "download_all": "すべてダウンロード", + "add_all_to_playlist": "すべて再生リストに追加", + "add_all_to_queue": "すべてキューに追加", + "play_all_next": "すべてを次に再生", + "pause": "一時停止", + "view_all": "すべて表示", + "no_tracks_added_yet": "まだ曲を追加していないようです", + "no_tracks": "ここには曲がないようです", + "no_tracks_listened_yet": "まだ何も聞いていないようです", + "not_following_artists": "アーティストをフォローしていません", + "no_favorite_albums_yet": "まだお気に入りのアルバムを追加していないようです", + "no_logs_found": "ログなし", + "youtube_engine": "YouTubeエンジン", + "youtube_engine_not_installed_title": "{engine}はインストールされていません", + "youtube_engine_not_installed_message": "{engine}はシステムにインストールされていません。", + "youtube_engine_set_path": "PATH変数に設定されていることを確認するか\n{engine}実行ファイルの絶対パスを下記に設定してください", + "youtube_engine_unix_issue_message": "macOS/Linux/Unix系OSでは、.zshrc/.bashrc/.bash_profileなどでパスを設定しても動作しません。\nシェルの設定ファイルにパスを設定する必要があります", + "download": "ダウンロード", + "file_not_found": "ファイルが見つかりません", + "custom": "独自", + "add_custom_url": "独自にURLを追加", + "edit_port": "ポートを編集", + "port_helper_msg": "初期設定は-1で、ランダムな番号を示します。ファイアウォールを設定している場合に設定することを推奨します。", + "connect_request": "{client}の接続を許可しますか?", + "connection_request_denied": "接続が拒否されました。ユーザーがアクセスを拒否しました。", + "hipotetical_calculation": "*これは、オンライン音楽ストリーミングプラットフォームの1ストリームあたりの平均支払い額である$0.003〜$0.005に基づいて計算されています。これは、ユーザーが異なる音楽ストリーミングプラットフォームで曲を聴いた場合に、アーティストにどれだけ支払ったかを把握するための仮説的な計算です。", + "an_error_occurred": "エラーが発生しました", + "copy_to_clipboard": "クリップボードにコピー", + "view_logs": "ログを表示", + "retry": "再試行", + "no_default_metadata_provider_selected": "デフォルトのメタデータプロバイダーが設定されていません", + "manage_metadata_providers": "メタデータプロバイダーを管理", + "open_link_in_browser": "リンクをブラウザで開きますか?", + "do_you_want_to_open_the_following_link": "次のリンクを開きますか", + "unsafe_url_warning": "信頼できないソースからのリンクを開くのは安全ではない場合があります。注意してください!\nリンクをクリップボードにコピーすることもできます。", + "copy_link": "リンクをコピー", + "building_your_timeline": "あなたの視聴履歴に基づいてタイムラインを作成しています...", + "official": "公式", + "author_name": "作者: {author}", + "third_party": "サードパーティ", + "plugin_requires_authentication": "プラグインには認証が必要です", + "update_available": "アップデートが利用可能です", + "supports_scrobbling": "scrobblingに対応", + "plugin_scrobbling_info": "このプラグインは、あなたの音楽をscrobbleして視聴履歴を生成します。", + "default_plugin": "デフォルト", + "set_default": "デフォルトに設定", + "support": "サポート", + "support_plugin_development": "プラグイン開発をサポート", + "can_access_name_api": "- **{name}** APIにアクセスできます", + "do_you_want_to_install_this_plugin": "このプラグインをインストールしますか?", + "third_party_plugin_warning": "このプラグインはサードパーティのリポジトリからのものです。インストールする前にソースを信頼できるか確認してください。", + "author": "作者", + "this_plugin_can_do_following": "このプラグインは以下のことができます", + "install": "インストール", + "install_a_metadata_provider": "メタデータプロバイダーをインストール", + "no_tracks_playing": "現在再生中のトラックはありません", + "synced_lyrics_not_available": "この曲の同期歌詞は利用できません。代わりに", + "plain_lyrics": "シンプルな歌詞", + "tab_instead": "タブを使用してください。", + "disclaimer": "免責事項", + "third_party_plugin_dmca_notice": "Spotubeチームは、いかなる「サードパーティ」プラグインについても責任(法的責任を含む)を負いません。\nご自身の責任でご使用ください。バグや問題については、プラグインリポジトリに報告してください。\n\n「サードパーティ」プラグインが何らかのサービス/法人のToS/DMCAを侵害している場合、その「サードパーティ」プラグインの作者またはホスティングプラットフォーム(例:GitHub/Codeberg)に措置を講じるよう依頼してください。上記に記載されている(「サードパーティ」とラベル付けされた)ものはすべて、パブリック/コミュニティによって維持されているプラグインです。私たちはそれらをキュレーションしていないため、それらに対して措置を講じることはできません。\n\n", + "input_does_not_match_format": "入力が必須フォーマットと一致しません", + "metadata_provider_plugins": "メタデータプロバイダープラグイン", + "paste_plugin_download_url": "ダウンロードURL、GitHub/CodebergリポジトリURL、または.smplugファイルへの直接リンクを貼り付けます", + "download_and_install_plugin_from_url": "URLからプラグインをダウンロードしてインストール", + "failed_to_add_plugin_error": "プラグインの追加に失敗しました: {error}", + "upload_plugin_from_file": "ファイルからプラグインをアップロード", + "installed": "インストール済み", + "available_plugins": "利用可能なプラグイン", + "configure_your_own_metadata_plugin": "独自のプレイリスト/アルバム/アーティスト/フィードのメタデータプロバイダーを構成", + "audio_scrobblers": "オーディオスクロッブラー", + "scrobbling": "Scrobbling", + "download_music_format": "音楽ダウンロード形式", + "streaming_music_format": "音楽ストリーミング形式", + "download_music_quality": "音楽ダウンロード品質", + "streaming_music_quality": "音楽ストリーミング品質", + "default_metadata_source": "デフォルトメタデータソース", + "set_default_metadata_source": "デフォルトメタデータソースを設定", + "default_audio_source": "デフォルトオーディオソース", + "set_default_audio_source": "デフォルトオーディオソースを設定", + "plugins": "プラグイン", + "configure_plugins": "独自のメタデータプロバイダーとオーディオソースプラグインを設定", + "source": "ソース: ", + "uncompressed": "非圧縮", + "dab_music_source_description": "オーディオファイル向け。高品質/ロスレスオーディオストリームを提供。正確なISRCベースのトラックマッチング。" } \ No newline at end of file diff --git a/lib/l10n/app_ka.arb b/lib/l10n/app_ka.arb index 3bcd0748..6a0cb06c 100644 --- a/lib/l10n/app_ka.arb +++ b/lib/l10n/app_ka.arb @@ -401,5 +401,94 @@ "export_cache_files": "ქეშირებული ფაილების ექსპორტი", "found_n_files": "ნაპოვნია {count} ფაილი", "export_cache_confirmation": "გსურთ ამ ფაილების ექსპორტი", - "exported_n_out_of_m_files": "{filesExported} ფაილი {files}-დან ექსპორტირებულია" + "exported_n_out_of_m_files": "{filesExported} ფაილი {files}-დან ექსპორტირებულია", + "playlist": "პლეისთი", + "no_loop": "არ არის ციკლი", + "generate": "გააგენერირეთ", + "undo": "დაბრუნება", + "download_all": "ყველას ჩამოტვირთვა", + "add_all_to_playlist": "ყველა დაამატეთ პლეისთში", + "add_all_to_queue": "ყველა დაამატეთ რიგში", + "play_all_next": "ყველა შემდეგ ითამაშე", + "pause": "შეჩერება", + "view_all": "ყველა ნახვა", + "no_tracks_added_yet": "გაჩნდება რომ ჯერ არ გაქვთ დამატებული ტრეკები", + "no_tracks": "გავლებული არ ჩანს არ არსებობს ტრეკები", + "no_tracks_listened_yet": "გქონდეთ გრძნობა, რომ ჯერ არაფერი უსმენია", + "not_following_artists": "არ მიჰყვებით რომელიმე არტისტს", + "no_favorite_albums_yet": "გაჩნდება რომ ჯერ არ გაქვთ დამატებული ალბომები თქვენს ფავორიტებში", + "no_logs_found": "ჩაწერები ვერ მოიძებნა", + "youtube_engine": "YouTube ძრავა", + "youtube_engine_not_installed_title": "{engine} არ არის ინსტალირებული", + "youtube_engine_not_installed_message": "{engine} არ არის ინსტალირებული თქვენს სისტემაში.", + "youtube_engine_set_path": "დარწმუნდით, რომ ის ხელმისაწვდომია PATH ცვლადში ან\nდაუყავით {engine} პროგრამის ფაილის სრული გზა", + "youtube_engine_unix_issue_message": "macOS/Linux/Unix მსგავსი ოპერაციული სისტემებში, .zshrc/.bashrc/.bash_profile-ით პათის დაყენება ვერ იმუშავებს.\nთქვენ უნდა დააყენოთ პათი შელ ფაილში", + "download": "ჩამოტვირთვა", + "file_not_found": "ფაილი ვერ მოიძებნა", + "custom": "პერსონალიზირებული", + "add_custom_url": "დამატება პერსონალური URL", + "edit_port": "პორტის რედაქტირება", + "port_helper_msg": "ნაგულისხმევი არის -1, რაც შემთხვევითი ნომრის მითითებას ნიშნავს. თუ لديك firewall настроен, рекомендуется установить это.", + "connect_request": "{client}-ის დაკავშირების ნებართვა?", + "connection_request_denied": "კავშირი უარყოფილია. მომხმარებელმა უარყო წვდომა.", + "hipotetical_calculation": "*ეს გამოითვლება ონლაინ მუსიკალური სტრიმინგის პლატფორმების საშუალო ანაზღაურების საფუძველზე, რომელიც შეადგენს $0.003-დან $0.005-მდე. ეს არის ჰიპოთეტური გაანგარიშება, რომელიც მომხმარებელს აძლევს წარმოდგენას, თუ რამდენს გადაუხდიდნენ ისინი არტისტებს, თუ მათ სიმღერებს მოუსმენდნენ სხვადასხვა მუსიკალურ სტრიმინგ პლატფორმაზე.", + "an_error_occurred": "მოხდა შეცდომა", + "copy_to_clipboard": "კოპირება ბუფერში", + "view_logs": "იხილეთ ჟურნალები", + "retry": "ხელახლა ცდა", + "no_default_metadata_provider_selected": "თქვენ არ გაქვთ დაყენებული ნაგულისხმევი მეტამონაცემების პროვაიდერი", + "manage_metadata_providers": "მეტამონაცემების პროვაიდერების მართვა", + "open_link_in_browser": "ბმულის გახსნა ბრაუზერში?", + "do_you_want_to_open_the_following_link": "გსურთ გახსნათ შემდეგი ბმული", + "unsafe_url_warning": "შეიძლება სახიფათო იყოს ბმულების გახსნა უნდობელი წყაროებიდან. იყავით ფრთხილად!\nასევე შეგიძლიათ დააკოპიროთ ბმული თქვენს ბუფერში.", + "copy_link": "ბმულის კოპირება", + "building_your_timeline": "თქვენი დროის ხაზის აგება თქვენი მოსმენების საფუძველზე...", + "official": "ოფიციალური", + "author_name": "ავტორი: {author}", + "third_party": "მესამე მხარის", + "plugin_requires_authentication": "პლაგინი საჭიროებს ავთენტიფიკაციას", + "update_available": "განახლება ხელმისაწვდომია", + "supports_scrobbling": "მხარს უჭერს სქრობლინგს", + "plugin_scrobbling_info": "ეს პლაგინი აწარმოებს თქვენი მუსიკის სქრობლინგს, რათა შექმნას თქვენი მოსმენის ისტორია.", + "default_plugin": "ნაგულისხმევი", + "set_default": "ნაგულისხმევად დაყენება", + "support": "მხარდაჭერა", + "support_plugin_development": "პლაგინის განვითარების მხარდაჭერა", + "can_access_name_api": "- შეუძლია წვდომა **{name}** API-ზე", + "do_you_want_to_install_this_plugin": "გსურთ ამ პლაგინის დაყენება?", + "third_party_plugin_warning": "ეს პლაგინი არის მესამე მხარის საცავიდან. გთხოვთ, დარწმუნდეთ, რომ ენდობით წყაროს დაყენებამდე.", + "author": "ავტორი", + "this_plugin_can_do_following": "ამ პლაგინს შეუძლია შემდეგის გაკეთება", + "install": "დაყენება", + "install_a_metadata_provider": "დააყენეთ მეტამონაცემების პროვაიდერი", + "no_tracks_playing": "ამჟამად არ უკრავს არცერთი ტრეკი", + "synced_lyrics_not_available": "ამ სიმღერისთვის სინქრონიზებული ტექსტები არ არის ხელმისაწვდომი. გთხოვთ, გამოიყენოთ", + "plain_lyrics": "მარტივი ტექსტები", + "tab_instead": "ჩანართი, სანაცვლოდ.", + "disclaimer": "პასუხისმგებლობის უარყოფა", + "third_party_plugin_dmca_notice": "Spotube-ის გუნდი არ იღებს პასუხისმგებლობას (მათ შორის, იურიდიულს) არცერთ \"მესამე მხარის\" პლაგინზე.\nგთხოვთ, გამოიყენოთ ისინი თქვენი რისკის ქვეშ. ნებისმიერი ხარვეზის/პრობლემის შესახებ შეატყობინეთ პლაგინის საცავს.\n\nთუ რომელიმე \"მესამე მხარის\" პლაგინი არღვევს რაიმე სერვისის/იურიდიული პირის ToS/DMCA-ს, გთხოვთ, სთხოვეთ \"მესამე მხარის\" პლაგინის ავტორს ან ჰოსტინგის პლატფორმას, მაგალითად GitHub/Codeberg, მიიღოს ზომები. ზემოთ ჩამოთვლილი (\"მესამე მხარის\" ეტიკეტის მქონე) ყველა არის საჯარო/საზოგადოების მიერ შენარჩუნებული პლაგინები. ჩვენ მათ არ ვაკონტროლებთ, ამიტომ არ შეგვიძლია მათზე რაიმე ზომების მიღება.\n\n", + "input_does_not_match_format": "შეყვანა არ ემთხვევა საჭირო ფორმატს", + "metadata_provider_plugins": "მეტამონაცემების პროვაიდერების პლაგინები", + "paste_plugin_download_url": "ჩასვით ჩამოტვირთვის url ან GitHub/Codeberg-ის რეპოს url ან პირდაპირი ბმული .smplug ფაილზე", + "download_and_install_plugin_from_url": "პლაგინის ჩამოტვირთვა და დაყენება url-დან", + "failed_to_add_plugin_error": "პლაგინის დამატება ვერ მოხერხდა: {error}", + "upload_plugin_from_file": "პლაგინის ატვირთვა ფაილიდან", + "installed": "დაინსტალირებული", + "available_plugins": "ხელმისაწვდომი პლაგინები", + "configure_your_own_metadata_plugin": "დააყენეთ თქვენი საკუთარი პლეილისტის/ალბომის/არტისტის/ფიდის მეტამონაცემების პროვაიდერი", + "audio_scrobblers": "აუდიო სქრობლერები", + "scrobbling": "სქრობლინგი", + "download_music_format": "მუსიკის ჩამოტვირთვის ფორმატი", + "streaming_music_format": "სტრიმინგის მუსიკის ფორმატი", + "download_music_quality": "ჩამოტვირთვის ხარისხი", + "streaming_music_quality": "სტრიმინგის ხარისხი", + "default_metadata_source": "ნაგულისხმევი მეტამონაცემების წყარო", + "set_default_metadata_source": "ნაგულისხმევი მეტამონაცემების წყაროს დაყენება", + "default_audio_source": "ნაგულისხმევი აუდიო წყარო", + "set_default_audio_source": "ნაგულისხმევი აუდიო წყაროს დაყენება", + "plugins": "პლაგინები", + "configure_plugins": "თქვენი საკუთარი მეტამონაცემებისა და აუდიო წყაროს პლაგინების კონფიგურაცია", + "source": "წყარო: ", + "uncompressed": "შეუკუმშავი", + "dab_music_source_description": "აუდიოფილებისთვის. უზრუნველყოფს მაღალი ხარისხის/უკომპრესო აუდიო სტრიმებს. ზუსტი შესაბამისობა ISRC-ის მიხედვით." } \ No newline at end of file diff --git a/lib/l10n/app_ko.arb b/lib/l10n/app_ko.arb index 7e368081..70a68f8b 100644 --- a/lib/l10n/app_ko.arb +++ b/lib/l10n/app_ko.arb @@ -402,5 +402,94 @@ "export_cache_files": "캐시된 파일 내보내기", "found_n_files": "{count}개의 파일을 찾았습니다", "export_cache_confirmation": "이 파일들을 내보내시겠습니까", - "exported_n_out_of_m_files": "{files}개 중 {filesExported}개 파일을 내보냈습니다" + "exported_n_out_of_m_files": "{files}개 중 {filesExported}개 파일을 내보냈습니다", + "playlist": "재생 목록", + "no_loop": "반복 없음", + "generate": "생성", + "undo": "실행 취소", + "download_all": "모두 다운로드", + "add_all_to_playlist": "모두 재생 목록에 추가", + "add_all_to_queue": "모두 큐에 추가", + "play_all_next": "모두 다음에 재생", + "pause": "일시 정지", + "view_all": "모두 보기", + "no_tracks_added_yet": "아직 트랙을 추가하지 않은 것 같습니다", + "no_tracks": "여기에 트랙이 없는 것 같습니다", + "no_tracks_listened_yet": "아직 아무 것도 듣지 않은 것 같습니다", + "not_following_artists": "아티스트를 팔로우하지 않고 있습니다", + "no_favorite_albums_yet": "아직 즐겨찾기 앨범을 추가하지 않은 것 같습니다", + "no_logs_found": "로그를 찾을 수 없습니다", + "youtube_engine": "YouTube 엔진", + "youtube_engine_not_installed_title": "{engine}가 설치되지 않았습니다", + "youtube_engine_not_installed_message": "{engine}가 시스템에 설치되지 않았습니다.", + "youtube_engine_set_path": "PATH 변수에서 사용할 수 있는지 확인하거나\n아래에 {engine} 실행 파일의 절대 경로를 설정하세요", + "youtube_engine_unix_issue_message": "macOS/Linux/unix와 같은 운영 체제에서는 .zshrc/.bashrc/.bash_profile 등에 경로 설정이 작동하지 않습니다.\n셸 구성 파일에 경로를 설정해야 합니다", + "download": "다운로드", + "file_not_found": "파일을 찾을 수 없습니다", + "custom": "사용자 정의", + "add_custom_url": "사용자 정의 URL 추가", + "edit_port": "포트 편집", + "port_helper_msg": "기본값은 -1로 무작위 숫자를 나타냅니다. 방화벽이 구성된 경우 이를 설정하는 것이 좋습니다.", + "connect_request": "{client}의 연결을 허용하시겠습니까?", + "connection_request_denied": "연결이 거부되었습니다. 사용자가 액세스를 거부했습니다.", + "hipotetical_calculation": "*이것은 온라인 음악 스트리밍 플랫폼의 스트림당 평균 지불액인 $0.003에서 $0.005를 기준으로 계산됩니다. 이것은 사용자가 다른 음악 스트리밍 플랫폼에서 노래를 들었다면 아티스트에게 얼마를 지불했을지에 대한 통찰력을 제공하기 위한 가상 계산입니다.", + "an_error_occurred": "오류가 발생했습니다", + "copy_to_clipboard": "클립보드에 복사", + "view_logs": "로그 보기", + "retry": "다시 시도", + "no_default_metadata_provider_selected": "기본 메타데이터 제공자가 설정되지 않았습니다", + "manage_metadata_providers": "메타데이터 제공자 관리", + "open_link_in_browser": "브라우저에서 링크를 여시겠습니까?", + "do_you_want_to_open_the_following_link": "다음 링크를 여시겠습니까", + "unsafe_url_warning": "신뢰할 수 없는 출처의 링크를 여는 것은 안전하지 않을 수 있습니다. 주의하세요!\n링크를 클립보드에 복사할 수도 있습니다.", + "copy_link": "링크 복사", + "building_your_timeline": "청취 기록을 기반으로 타임라인을 구축하고 있습니다...", + "official": "공식", + "author_name": "저자: {author}", + "third_party": "타사", + "plugin_requires_authentication": "플러그인에 인증이 필요합니다", + "update_available": "업데이트 사용 가능", + "supports_scrobbling": "스크로블링 지원", + "plugin_scrobbling_info": "이 플러그인은 음악을 스크로블하여 청취 기록을 생성합니다.", + "default_plugin": "기본", + "set_default": "기본값으로 설정", + "support": "지원", + "support_plugin_development": "플러그인 개발 지원", + "can_access_name_api": "- **{name}** API에 액세스할 수 있습니다", + "do_you_want_to_install_this_plugin": "이 플러그인을 설치하시겠습니까?", + "third_party_plugin_warning": "이 플러그인은 타사 리포지토리에서 제공됩니다. 설치하기 전에 출처를 신뢰하는지 확인하세요.", + "author": "저자", + "this_plugin_can_do_following": "이 플러그인은 다음을 수행할 수 있습니다", + "install": "설치", + "install_a_metadata_provider": "메타데이터 제공자 설치", + "no_tracks_playing": "현재 재생 중인 트랙이 없습니다", + "synced_lyrics_not_available": "이 노래에 대한 동기화된 가사를 사용할 수 없습니다. 대신", + "plain_lyrics": "일반 가사", + "tab_instead": "탭을 사용하세요.", + "disclaimer": "면책 조항", + "third_party_plugin_dmca_notice": "Spotube 팀은 어떠한 \"타사\" 플러그인에 대해서도 (법적 포함) 어떠한 책임도 지지 않습니다.\n사용자 자신의 책임하에 사용하시기 바랍니다. 버그/문제에 대해서는 플러그인 리포지토리에 보고해 주세요.\n\n만약 \"타사\" 플러그인이 서비스/법인의 ToS/DMCA를 위반하는 경우, \"타사\" 플러그인 저자 또는 호스팅 플랫폼(예: GitHub/Codeberg)에 조치를 취하도록 요청해 주세요. 위에 나열된 (\"타사\"로 표시된) 플러그인은 모두 공개/커뮤니티에서 유지 관리하는 플러그인입니다. 저희는 이를 큐레이션하지 않으므로 어떠한 조치도 취할 수 없습니다.\n\n", + "input_does_not_match_format": "입력이 필요한 형식과 일치하지 않습니다", + "metadata_provider_plugins": "메타데이터 제공자 플러그인", + "paste_plugin_download_url": "다운로드 URL, GitHub/Codeberg 리포지토리 URL 또는 .smplug 파일에 대한 직접 링크를 붙여넣으세요", + "download_and_install_plugin_from_url": "URL에서 플러그인 다운로드 및 설치", + "failed_to_add_plugin_error": "플러그인 추가 실패: {error}", + "upload_plugin_from_file": "파일에서 플러그인 업로드", + "installed": "설치됨", + "available_plugins": "사용 가능한 플러그인", + "configure_your_own_metadata_plugin": "자신만의 플레이리스트/앨범/아티스트/피드 메타데이터 제공자 구성", + "audio_scrobblers": "오디오 스크로블러", + "scrobbling": "스크로블링", + "download_music_format": "다운로드 음악 포맷", + "streaming_music_format": "스트리밍 음악 포맷", + "download_music_quality": "다운로드 음질", + "streaming_music_quality": "스트리밍 음질", + "default_metadata_source": "기본 메타데이터 소스", + "set_default_metadata_source": "기본 메타데이터 소스 설정", + "default_audio_source": "기본 오디오 소스", + "set_default_audio_source": "기본 오디오 소스 설정", + "plugins": "플러그인", + "configure_plugins": "직접 메타데이터 제공자와 오디오 소스 플러그인을 구성하세요", + "source": "출처: ", + "uncompressed": "비압축", + "dab_music_source_description": "오디오파일을 위한 소스입니다. 고음질/무손실 오디오 스트림을 제공하며 ISRC 기반으로 정확한 트랙 매칭을 지원합니다." } \ No newline at end of file diff --git a/lib/l10n/app_ne.arb b/lib/l10n/app_ne.arb index 77eea7d0..874c28a5 100644 --- a/lib/l10n/app_ne.arb +++ b/lib/l10n/app_ne.arb @@ -401,5 +401,94 @@ "export_cache_files": "क्यास फाइलहरू निर्यात गर्नुहोस्", "found_n_files": "{count} फाइलहरू फेला परे", "export_cache_confirmation": "यी फाइलहरू निर्यात गर्न चाहनुहुन्छ", - "exported_n_out_of_m_files": "{filesExported} मध्ये {files} फाइलहरू निर्यात गरियो" + "exported_n_out_of_m_files": "{filesExported} मध्ये {files} फाइलहरू निर्यात गरियो", + "playlist": "प्लेलिस्ट", + "no_loop": "कोई लूप नहीं", + "generate": "जनरेट", + "undo": "पूर्ववत", + "download_all": "सभी डाउनलोड करें", + "add_all_to_playlist": "सभी को प्लेलिस्ट में जोड़ें", + "add_all_to_queue": "सभी को कतार में जोड़ें", + "play_all_next": "सभी को अगला प्ले करें", + "pause": "विराम", + "view_all": "सभी देखें", + "no_tracks_added_yet": "लगता है आपने अभी तक कोई ट्रैक नहीं जोड़ा है", + "no_tracks": "यहाँ कोई ट्रैक नहीं दिख रहे हैं", + "no_tracks_listened_yet": "आपने अभी तक कुछ नहीं सुना है ऐसा लगता है", + "not_following_artists": "आप किसी कलाकार को फॉलो नहीं कर रहे हैं", + "no_favorite_albums_yet": "लगता है आपने अभी तक कोई एल्बम पसंदीदा में नहीं जोड़ा है", + "no_logs_found": "कोई लॉग नहीं मिला", + "youtube_engine": "YouTube इंजन", + "youtube_engine_not_installed_title": "{engine} इंस्टॉल नहीं है", + "youtube_engine_not_installed_message": "{engine} आपके सिस्टम में इंस्टॉल नहीं है।", + "youtube_engine_set_path": "सुनिश्चित करें कि यह PATH वेरिएबल में उपलब्ध है या\nनीचे {engine} एक्जीक्यूटेबल का पूर्ण पथ सेट करें", + "youtube_engine_unix_issue_message": "macOS/Linux/unix जैसे ऑपरेटिंग सिस्टम में, .zshrc/.bashrc/.bash_profile आदि में पथ सेट करना काम नहीं करेगा।\nआपको शेल कॉन्फ़िगरेशन फ़ाइल में पथ सेट करना होगा", + "download": "डाउनलोड", + "file_not_found": "फ़ाइल नहीं मिली", + "custom": "कस्टम", + "add_custom_url": "कस्टम URL जोड़ें", + "edit_port": "पोर्ट सम्पादन गर्नुहोस्", + "port_helper_msg": "डिफ़ॉल्ट -1 हो जुन यादृच्छिक संख्या जनाउँछ। यदि तपाईंले फायरवाल कन्फिगर गर्नुभएको छ भने, यसलाई सेट गर्न सिफारिस गरिन्छ।", + "connect_request": "{client} लाई जडान गर्न अनुमति दिनुहोस्?", + "connection_request_denied": "जडान अस्वीकृत। प्रयोगकर्ताले पहुँच अस्वीकृत गर्यो।", + "hipotetical_calculation": "*यो अनलाइन संगीत स्ट्रिमिङ प्लेटफर्मको प्रति स्ट्रिम भुक्तानी $0.003 देखि $0.005 को औसतमा आधारित छ। यो एक काल्पनिक गणना हो जुन प्रयोगकर्तालाई उनीहरूले विभिन्न संगीत स्ट्रिमिङ प्लेटफर्ममा आफ्ना गीतहरू सुनेमा कलाकारहरूलाई कति भुक्तानी गर्ने थिए भन्ने बारेमा अन्तरदृष्टि दिनको लागि हो।", + "an_error_occurred": "त्रुटि भयो", + "copy_to_clipboard": "क्लिपबोर्डमा प्रतिलिपि गर्नुहोस्", + "view_logs": "लगहरू हेर्नुहोस्", + "retry": "पुनः प्रयास गर्नुहोस्", + "no_default_metadata_provider_selected": "तपाईंले कुनै पूर्वनिर्धारित मेटाडेटा प्रदायक सेट गर्नुभएको छैन", + "manage_metadata_providers": "मेटाडेटा प्रदायकहरू प्रबन्ध गर्नुहोस्", + "open_link_in_browser": "ब्राउजरमा लिङ्क खोल्ने?", + "do_you_want_to_open_the_following_link": "के तपाईं निम्न लिङ्क खोल्न चाहनुहुन्छ", + "unsafe_url_warning": "अविश्वसनीय स्रोतहरूबाट लिङ्कहरू खोल्नु असुरक्षित हुन सक्छ। सावधान रहनुहोस्!\nतपाईं लिङ्कलाई आफ्नो क्लिपबोर्डमा पनि प्रतिलिपि गर्न सक्नुहुन्छ।", + "copy_link": "लिङ्क प्रतिलिपि गर्नुहोस्", + "building_your_timeline": "तपाईंको सुन्ने आधारमा तपाईंको समयरेखा निर्माण गर्दै...", + "official": "आधिकारिक", + "author_name": "लेखक: {author}", + "third_party": "तेस्रो-पक्ष", + "plugin_requires_authentication": "प्लगइनलाई प्रमाणीकरण चाहिन्छ", + "update_available": "अपडेट उपलब्ध छ", + "supports_scrobbling": "स्क्रब्बलिंगलाई समर्थन गर्दछ", + "plugin_scrobbling_info": "यो प्लगइनले तपाईंको सुन्ने इतिहास उत्पन्न गर्न तपाईंको संगीतलाई स्क्रब्बल गर्दछ।", + "default_plugin": "पूर्वनिर्धारित", + "set_default": "पूर्वनिर्धारित सेट गर्नुहोस्", + "support": "समर्थन", + "support_plugin_development": "प्लगइन विकासलाई समर्थन गर्नुहोस्", + "can_access_name_api": "- **{name}** API मा पहुँच गर्न सक्छ", + "do_you_want_to_install_this_plugin": "के तपाईं यो प्लगइन स्थापना गर्न चाहनुहुन्छ?", + "third_party_plugin_warning": "यो प्लगइन तेस्रो-पक्ष रिपोसिटरीबाट हो। कृपया स्थापना गर्नु अघि तपाईंले स्रोतमा विश्वास गर्नुहुन्छ भनी सुनिश्चित गर्नुहोस्।", + "author": "लेखक", + "this_plugin_can_do_following": "यो प्लगइनले निम्न गर्न सक्छ", + "install": "स्थापना गर्नुहोस्", + "install_a_metadata_provider": "मेटाडेटा प्रदायक स्थापना गर्नुहोस्", + "no_tracks_playing": "हाल कुनै ट्र्याक बजिरहेको छैन", + "synced_lyrics_not_available": "यो गीतको लागि सिङ्क गरिएका बोलहरू उपलब्ध छैनन्। कृपया यसको सट्टा", + "plain_lyrics": "सादा बोलहरू", + "tab_instead": "ट्याब प्रयोग गर्नुहोस्।", + "disclaimer": "अस्वीकरण", + "third_party_plugin_dmca_notice": "स्पोट्यूब टोलीले कुनै पनि \"तेस्रो-पक्ष\" प्लगइनहरूको लागि कुनै जिम्मेवारी (कानुनी सहित) लिँदैन।\nकृपया तिनीहरूलाई आफ्नो जोखिममा प्रयोग गर्नुहोस्। कुनै पनि बग/समस्याहरूको लागि, कृपया तिनीहरूलाई प्लगइन रिपोसिटरीमा रिपोर्ट गर्नुहोस्।\n\nयदि कुनै \"तेस्रो-पक्ष\" प्लगइनले कुनै सेवा/कानुनी संस्थाको ToS/DMCA तोडिरहेको छ भने, कृपया \"तेस्रो-पक्ष\" प्लगइन लेखक वा होस्टिङ प्लेटफर्म e.g. GitHub/Codeberg लाई कारबाही गर्न अनुरोध गर्नुहोस्। माथि सूचीबद्ध (\"तेस्रो-पक्ष\" लेबल गरिएका) सबै सार्वजनिक/सामुदायिक रूपमा राखिएका प्लगइनहरू हुन्। हामी तिनीहरूलाई क्युरेट गरिरहेका छैनौं, त्यसैले हामी तिनीहरूमा कुनै कारबाही गर्न सक्दैनौं।\n\n", + "input_does_not_match_format": "इनपुट आवश्यक ढाँचासँग मेल खाँदैन", + "metadata_provider_plugins": "मेटाडेटा प्रदायक प्लगइनहरू", + "paste_plugin_download_url": "डाउनलोड url वा GitHub/Codeberg repo url वा .smplug फाइलमा सिधा लिङ्क टाँस्नुहोस्", + "download_and_install_plugin_from_url": "url बाट प्लगइन डाउनलोड र स्थापना गर्नुहोस्", + "failed_to_add_plugin_error": "प्लगइन थप्न असफल: {error}", + "upload_plugin_from_file": "फाइलबाट प्लगइन अपलोड गर्नुहोस्", + "installed": "स्थापित", + "available_plugins": "उपलब्ध प्लगइनहरू", + "configure_your_own_metadata_plugin": "तपाईंको आफ्नै प्लेलिस्ट/एल्बम/कलाकार/फिड मेटाडेटा प्रदायक कन्फिगर गर्नुहोस्", + "audio_scrobblers": "अडियो स्क्रब्बलरहरू", + "scrobbling": "स्क्रब्बलिंग", + "download_music_format": "सङ्गीत डाउनलोड ढाँचा", + "streaming_music_format": "स्ट्रिमिङ सङ्गीत ढाँचा", + "download_music_quality": "डाउनलोड गुणस्तर", + "streaming_music_quality": "स्ट्रिमिङ गुणस्तर", + "default_metadata_source": "पूर्वनिर्धारित मेटाडाटा स्रोत", + "set_default_metadata_source": "पूर्वनिर्धारित मेटाडाटा स्रोत सेट गर्नुहोस्", + "default_audio_source": "पूर्वनिर्धारित अडियो स्रोत", + "set_default_audio_source": "पूर्वनिर्धारित अडियो स्रोत सेट गर्नुहोस्", + "plugins": "प्लगइनहरू", + "configure_plugins": "आफ्नै मेटाडाटा प्रदायक र अडियो स्रोत प्लगइनहरू कन्फिगर गर्नुहोस्", + "source": "स्रोत: ", + "uncompressed": "असंक्षिप्त", + "dab_music_source_description": "अडियोप्रेमीहरूका लागि। उच्च गुणस्तर/लसलेस अडियो स्ट्रिमहरू उपलब्ध गराउँछ। ISRC-मा आधारित सटीक ट्र्याक मिलान।" } \ No newline at end of file diff --git a/lib/l10n/app_nl.arb b/lib/l10n/app_nl.arb index 50b5e3bd..4d8deac1 100644 --- a/lib/l10n/app_nl.arb +++ b/lib/l10n/app_nl.arb @@ -9,7 +9,7 @@ "genre": "Genre", "personalized": "Gepersonaliseerd", "featured": "Aanbevolen", - "new_releases": "Nieuwe uitgaves", + "new_releases": "Nieuwe uitgaven", "songs": "Liedjes", "playing_track": "{track} afspelen", "queue_clear_alert": "Dit zal de huidige wachtrij wissen. {track_length} nummers worden verwijderd\nWil je doorgaan?", @@ -41,14 +41,15 @@ "sort_z_a": "Sorteren op Z-A", "sort_artist": "Sorteren op artiest", "sort_album": "Sorteren op album", + "sort_duration": "Sorteren op lengte", "sort_tracks": "Nummers sorteren", "currently_downloading": "Momenteel aan het downloaden ({tracks_length})", - "cancel_all": "Alle annuleren", + "cancel_all": "Alles annuleren", "filter_artist": "Artiesten filteren…", "followers": "{followers} volgers", "add_artist_to_blacklist": "Artiest toevoegen aan zwarte lijst", - "top_tracks": "Topsporen", - "fans_also_like": "Liefhebbers willen ook", + "top_tracks": "Topnummers", + "fans_also_like": "Fans luisteren ook", "loading": "Laden…", "artist": "Artiest", "blacklisted": "Zwarte lijst", @@ -89,8 +90,8 @@ "share": "Delen", "mini_player": "Minispeler", "slide_to_seek": "Schuiven om vooruit of achteruit te zoeken", - "shuffle_playlist": "Afspeellijst schuifelen", - "unshuffle_playlist": "Afspeellijst onschuifelen", + "shuffle_playlist": "Afspeellijst willekeurig", + "unshuffle_playlist": "Afspeellijst op volgorde", "previous_track": "Vorige nummer", "next_track": "Volgende nummer", "pause_playback": "Afspelen pauzeren", @@ -98,7 +99,7 @@ "loop_track": "Nummer herhalen", "repeat_playlist": "Afspeellijst herhalen", "queue": "Wachtrij", - "alternative_track_sources": "Alternatieve nummerbronnen", + "alternative_track_sources": "Alternatieve bronnen voor nummers", "download_track": "Nummer downloaden", "tracks_in_queue": "{tracks} nummers in wachtrij", "clear_all": "Alles wissen", @@ -240,8 +241,8 @@ "views": "Weergaven", "streamUrl": "Stream-URL", "stop": "Stoppen", - "sort_newest": "Sorteren op nieuwste toegevoegd", - "sort_oldest": "Sorteren op oudste toegevoegd", + "sort_newest": "Sorteren op recent toegevoegd", + "sort_oldest": "Sorteren op langst toegevoegd", "sleep_timer": "Slaaptimer", "mins": "{minutes} minuten", "hours": "{hours} uren", @@ -287,34 +288,32 @@ "explore_genres": "Genres verkennen", "friends": "Vrienden", "no_lyrics_available": "Sorry, geen teksten gevonden voor dit nummer", - "sort_duration": "Sorteer op Duur", - "audio_source": "Audiobron", - "start_a_radio": "Start een Radio", - "how_to_start_radio": "Hoe wilt u de radio starten?", - "replace_queue_question": "Wilt u de huidige wachtrij vervangen of eraan toevoegen?", - "endless_playback": "Eindeloze Afspelen", - "delete_playlist": "Verwijder Afspeellijst", - "delete_playlist_confirmation": "Weet u zeker dat u deze afspeellijst wilt verwijderen?", - "local_tracks": "Lokale Nummers", - "song_link": "Nummer Link", - "skip_this_nonsense": "Sla deze onzin over", - "freedom_of_music": "“Vrijheid van Muziek”", - "freedom_of_music_palm": "“Vrijheid van Muziek in de palm van je hand”", + "start_a_radio": "Een radio starten", + "how_to_start_radio": "Hoe wil je de radio starten?", + "replace_queue_question": "Wil je de huidige wachtrij vervangen of eraan toevoegen?", + "endless_playback": "Oneindig afspelen", + "delete_playlist": "Afspeellijst verwijderen", + "delete_playlist_confirmation": "Weet je zeker dat je deze afspeellijst wilt verwijderen?", + "local_tracks": "Lokale nummers", + "song_link": "Song-link", + "skip_this_nonsense": "Deze onzin overslaan", + "freedom_of_music": "“Vrijheid van muziek”", + "freedom_of_music_palm": "“Vrijheid van muziek in je hand”", "get_started": "Laten we beginnen", - "youtube_source_description": "Aanbevolen en werkt het beste.", - "piped_source_description": "Voel je vrij? Hetzelfde als YouTube maar veel gratis.", - "jiosaavn_source_description": "Het beste voor de Zuid-Aziatische regio.", - "highest_quality": "Hoogste Kwaliteit: {quality}", - "select_audio_source": "Selecteer Audiobron", - "endless_playback_description": "Voeg automatisch nieuwe nummers toe aan het einde van de wachtrij", - "choose_your_region": "Kies uw regio", - "choose_your_region_description": "Dit zal Spotube helpen om de juiste inhoud voor uw locatie te tonen.", - "choose_your_language": "Kies uw taal", - "help_project_grow": "Help dit project groeien", - "help_project_grow_description": "Spotube is een open-source project. U kunt dit project helpen groeien door bij te dragen aan het project, bugs te melden of nieuwe functies voor te stellen.", - "contribute_on_github": "Bijdragen op GitHub", - "donate_on_open_collective": "Doneren op Open Collective", - "browse_anonymously": "Anoniem Bladeren", + "youtube_source_description": "Aangeraden en werkt het best.", + "piped_source_description": "Voel je je vrij? Net als YouTube, maar meer vrij.", + "jiosaavn_source_description": "Het beste voor de regio Zuid-Azië.", + "highest_quality": "Hoogste kwaliteit: {quality}", + "select_audio_source": "Audiobron kiezen", + "endless_playback_description": "Nieuwe nummers automatisch achteraan de wachtrij toevoegen", + "choose_your_region": "Kies je regio", + "choose_your_region_description": "Dit helpt Spotube om de juiste inhoud\nvoor jouw locatie te tonen.", + "choose_your_language": "Kies je taal", + "help_project_grow": "Help dit project met groeien", + "help_project_grow_description": "Spotube is een open-source project. Je kunt dit project helpen groeien door eraan bij te dragen, problemen te melden of nieuwe functies voor te stellen.", + "contribute_on_github": "Bijdragen on GitHub", + "donate_on_open_collective": "Doneren on Open Collective", + "browse_anonymously": "Anoniem browsen", "enable_connect": "Verbinding inschakelen", "enable_connect_description": "Spotube bedienen vanaf andere apparaten", "devices": "Apparaten", @@ -402,5 +401,95 @@ "export_cache_files": "Gecacheerde bestanden exporteren", "found_n_files": "{count} bestanden gevonden", "export_cache_confirmation": "Wilt u deze bestanden exporteren naar", - "exported_n_out_of_m_files": "{filesExported} van de {files} bestanden geëxporteerd" + "exported_n_out_of_m_files": "{filesExported} van de {files} bestanden geëxporteerd", + "playlist": "Afspeellijst", + "no_loop": "Geen herhaling", + "generate": "Genereren", + "undo": "Ongedaan maken", + "download_all": "Alles downloaden", + "add_all_to_playlist": "Voeg alles toe aan afspeellijst", + "add_all_to_queue": "Voeg alles toe aan wachtrij", + "play_all_next": "Speel alles volgende", + "pause": "Pauzeren", + "view_all": "Bekijk alles", + "no_tracks_added_yet": "Het lijkt erop dat je nog geen nummers hebt toegevoegd", + "no_tracks": "Het lijkt erop dat er hier geen nummers zijn", + "no_tracks_listened_yet": "Het lijkt erop dat je nog niets hebt beluisterd", + "not_following_artists": "Je volgt geen artiesten", + "no_favorite_albums_yet": "Het lijkt erop dat je nog geen albums aan je favorieten hebt toegevoegd", + "no_logs_found": "Geen logbestanden gevonden", + "youtube_engine": "YouTube Engine", + "youtube_engine_not_installed_title": "{engine} is niet geïnstalleerd", + "youtube_engine_not_installed_message": "{engine} is niet geïnstalleerd op je systeem.", + "youtube_engine_set_path": "Zorg ervoor dat het beschikbaar is in de PATH-variabele of\nstel het absolute pad naar de {engine} uitvoerbare bestanden in", + "youtube_engine_unix_issue_message": "Op macOS/Linux/unix-achtige besturingssystemen werkt het instellen van paden in .zshrc/.bashrc/.bash_profile enz. niet.\nJe moet het pad instellen in het shell-configuratiebestand", + "download": "Downloaden", + "file_not_found": "Bestand niet gevonden", + "custom": "Aangepast", + "add_custom_url": "Voeg aangepaste URL toe", + "edit_port": "Poort bewerken", + "port_helper_msg": "Standaard is -1, wat een willekeurig nummer aangeeft. Als je een firewall hebt geconfigureerd, wordt aanbevolen dit in te stellen.", + "connect_request": "Toestaan dat {client} verbinding maakt?", + "connection_request_denied": "Verbinding geweigerd. Gebruiker heeft toegang geweigerd.", + "hipotetical_calculation": "*Dit is berekend op basis van de gemiddelde uitbetaling per stream van online muziekstreamingplatforms van $0,003 tot $0,005. Dit is een hypothetische berekening om de gebruiker inzicht te geven in hoeveel ze aan de artiesten zouden hebben betaald als ze hun nummer op een ander muziekstreamingplatform zouden beluisteren.", + "an_error_occurred": "Er is een fout opgetreden", + "copy_to_clipboard": "Kopiëren naar klembord", + "view_logs": "Logboeken bekijken", + "retry": "Opnieuw proberen", + "no_default_metadata_provider_selected": "U heeft geen standaard metadata-aanbieder ingesteld", + "manage_metadata_providers": "Metadata-aanbieders beheren", + "open_link_in_browser": "Link openen in browser?", + "do_you_want_to_open_the_following_link": "Wilt u de volgende link openen", + "unsafe_url_warning": "Het kan onveilig zijn om links van onbetrouwbare bronnen te openen. Wees voorzichtig!\nU kunt de link ook naar uw klembord kopiëren.", + "copy_link": "Link kopiëren", + "building_your_timeline": "Uw tijdlijn wordt opgebouwd op basis van uw luistergedrag...", + "official": "Officieel", + "author_name": "Auteur: {author}", + "third_party": "Derden", + "plugin_requires_authentication": "Plugin vereist authenticatie", + "update_available": "Update beschikbaar", + "supports_scrobbling": "Ondersteunt scrobbling", + "plugin_scrobbling_info": "Deze plugin scrobblet uw muziek om uw luistergeschiedenis te genereren.", + "default_plugin": "Standaard", + "set_default": "Instellen als standaard", + "support": "Ondersteuning", + "support_plugin_development": "Ondersteun plugin-ontwikkeling", + "can_access_name_api": "- Kan de **{name}** API benaderen", + "do_you_want_to_install_this_plugin": "Wilt u deze plugin installeren?", + "third_party_plugin_warning": "Deze plugin is afkomstig van een repository van derden. Zorg ervoor dat u de bron vertrouwt voordat u installeert.", + "author": "Auteur", + "this_plugin_can_do_following": "Deze plugin kan het volgende doen", + "install": "Installeren", + "install_a_metadata_provider": "Een metadata-aanbieder installeren", + "no_tracks_playing": "Er wordt momenteel geen nummer afgespeeld", + "synced_lyrics_not_available": "Gesynchroniseerde songteksten zijn niet beschikbaar voor dit nummer. Gebruik in plaats daarvan het tabblad", + "plain_lyrics": "Eenvoudige songteksten", + "tab_instead": "in plaats daarvan.", + "disclaimer": "Disclaimer", + "third_party_plugin_dmca_notice": "Het Spotube-team draagt geen enkele verantwoordelijkheid (inclusief juridische) voor \"derden\" plugins.\nGebruik ze op eigen risico. Voor bugs/problemen kunt u deze melden bij de plugin-repository.\n\nAls een \"derden\" plugin de ToS/DMCA van een service/juridische entiteit schendt, vraag dan de auteur van de \"derden\" plugin of het hostingplatform, bijvoorbeeld GitHub/Codeberg, om actie te ondernemen. De hierboven vermelde (gelabelde \"derden\") plugins zijn allemaal openbare/door de gemeenschap onderhouden plugins. We beheren ze niet, dus we kunnen geen actie tegen ze ondernemen.\n\n", + "input_does_not_match_format": "Invoer komt niet overeen met het vereiste formaat", + "metadata_provider_plugins": "Metadata-aanbieder Plugins", + "paste_plugin_download_url": "Plak de download-URL of de URL van de GitHub/Codeberg-repository of een directe link naar het .smplug-bestand", + "download_and_install_plugin_from_url": "Download en installeer de plugin via URL", + "failed_to_add_plugin_error": "Kon de plugin niet toevoegen: {error}", + "upload_plugin_from_file": "Plugin uploaden vanuit bestand", + "installed": "Geïnstalleerd", + "available_plugins": "Beschikbare plugins", + "configure_your_own_metadata_plugin": "Configureer uw eigen metadata-aanbieder voor afspeellijst/album/artiest/feed", + "audio_scrobblers": "Audioscrobblers", + "scrobbling": "Scrobbling", + "download_music_format": "Download muziekformaat", + "streaming_music_format": "Streaming muziekformaat", + "download_music_quality": "Downloadkwaliteit", + "streaming_music_quality": "Streamingkwaliteit", + "default_metadata_source": "Standaard metadata-bron", + "set_default_metadata_source": "Standaard metadata-bron instellen", + "default_audio_source": "Standaard audiobron", + "set_default_audio_source": "Standaard audiobron instellen", + "plugins": "Plug-ins", + "configure_plugins": "Configureer je eigen metadata- en audiobron-plug-ins", + "source": "Bron: ", + "uncompressed": "Ongecomprimeerd", + "dab_music_source_description": "Voor audiofielen. Biedt hoge kwaliteit/lossless audiostreams. Nauwkeurige trackmatching op basis van ISRC.", + "audio_source": "Audiobron" } \ No newline at end of file diff --git a/lib/l10n/app_pl.arb b/lib/l10n/app_pl.arb index 11ab51ce..80da1e89 100644 --- a/lib/l10n/app_pl.arb +++ b/lib/l10n/app_pl.arb @@ -401,5 +401,94 @@ "export_cache_files": "Eksportuj pliki z pamięci podręcznej", "found_n_files": "Znaleziono {count} plików", "export_cache_confirmation": "Czy chcesz wyeksportować te pliki do", - "exported_n_out_of_m_files": "Wyeksportowano {filesExported} z {files} plików" + "exported_n_out_of_m_files": "Wyeksportowano {filesExported} z {files} plików", + "playlist": "Playlista", + "no_loop": "Brak pętli", + "generate": "Generuj", + "undo": "Cofnij", + "download_all": "Pobierz wszystko", + "add_all_to_playlist": "Dodaj wszystko do playlisty", + "add_all_to_queue": "Dodaj wszystko do kolejki", + "play_all_next": "Odtwórz wszystko następnie", + "pause": "Pauza", + "view_all": "Zobacz wszystko", + "no_tracks_added_yet": "Wygląda na to, że jeszcze nie dodałeś żadnych utworów", + "no_tracks": "Wygląda na to, że tutaj nie ma żadnych utworów", + "no_tracks_listened_yet": "Wygląda na to, że jeszcze nic nie słuchałeś", + "not_following_artists": "Nie obserwujesz żadnych artystów", + "no_favorite_albums_yet": "Wygląda na to, że jeszcze nie dodałeś żadnych albumów do ulubionych", + "no_logs_found": "Nie znaleziono żadnych logów", + "youtube_engine": "Silnik YouTube", + "youtube_engine_not_installed_title": "{engine} nie jest zainstalowany", + "youtube_engine_not_installed_message": "{engine} nie jest zainstalowany w systemie.", + "youtube_engine_set_path": "Upewnij się, że jest dostępny w zmiennej PATH lub\nustaw absolutną ścieżkę do pliku wykonywalnego {engine} poniżej", + "youtube_engine_unix_issue_message": "W systemach macOS/Linux/unix, ustawianie ścieżki w .zshrc/.bashrc/.bash_profile itp. nie będzie działać.\nMusisz ustawić ścieżkę w pliku konfiguracyjnym powłoki", + "download": "Pobierz", + "file_not_found": "Plik nie znaleziony", + "custom": "Niestandardowy", + "add_custom_url": "Dodaj niestandardowy URL", + "edit_port": "Edytuj port", + "port_helper_msg": "Domyślna wartość to -1, co oznacza losową liczbę. Jeśli masz skonfigurowany zaporę, zaleca się jej ustawienie.", + "connect_request": "Zezwolić {client} na połączenie?", + "connection_request_denied": "Połączenie odrzucone. Użytkownik odmówił dostępu.", + "hipotetical_calculation": "*Jest to obliczone na podstawie średniej wypłaty z internetowych platform streamingowych za jeden stream w wysokości 0,003 do 0,005 USD. Jest to hipotetyczne obliczenie, które ma na celu dać użytkownikowi wgląd w to, ile zapłaciłby artystom, gdyby słuchał ich piosenek na różnych platformach streamingowych.", + "an_error_occurred": "Wystąpił błąd", + "copy_to_clipboard": "Kopiuj do schowka", + "view_logs": "Wyświetl logi", + "retry": "Ponów", + "no_default_metadata_provider_selected": "Nie masz ustawionego domyślnego dostawcy metadanych", + "manage_metadata_providers": "Zarządzaj dostawcami metadanych", + "open_link_in_browser": "Otworzyć link w przeglądarce?", + "do_you_want_to_open_the_following_link": "Czy chcesz otworzyć następujący link", + "unsafe_url_warning": "Otwieranie linków z niezaufanych źródeł może być niebezpieczne. Zachowaj ostrożność!\nMożesz również skopiować link do schowka.", + "copy_link": "Kopiuj link", + "building_your_timeline": "Budowanie Twojej osi czasu na podstawie Twoich odsłuchań...", + "official": "Oficjalny", + "author_name": "Autor: {author}", + "third_party": "Zewnętrzny", + "plugin_requires_authentication": "Wtyczka wymaga uwierzytelnienia", + "update_available": "Dostępna aktualizacja", + "supports_scrobbling": "Obsługuje scrobbling", + "plugin_scrobbling_info": "Ta wtyczka scrobbluje Twoją muzykę, aby wygenerować historię odsłuchań.", + "default_plugin": "Domyślna", + "set_default": "Ustaw jako domyślną", + "support": "Wsparcie", + "support_plugin_development": "Wspieraj rozwój wtyczki", + "can_access_name_api": "- Może uzyskać dostęp do API **{name}**", + "do_you_want_to_install_this_plugin": "Czy chcesz zainstalować tę wtyczkę?", + "third_party_plugin_warning": "Ta wtyczka pochodzi z zewnętrznego repozytorium. Upewnij się, że ufasz źródłu przed instalacją.", + "author": "Autor", + "this_plugin_can_do_following": "Ta wtyczka może wykonywać następujące czynności", + "install": "Instaluj", + "install_a_metadata_provider": "Zainstaluj dostawcę metadanych", + "no_tracks_playing": "Obecnie nie odtwarzany jest żaden utwór", + "synced_lyrics_not_available": "Zsynchronizowane teksty nie są dostępne dla tego utworu. Zamiast tego użyj zakładki", + "plain_lyrics": "Zwykłe teksty", + "tab_instead": "zamiast tego.", + "disclaimer": "Zastrzeżenie", + "third_party_plugin_dmca_notice": "Zespół Spotube nie ponosi żadnej odpowiedzialności (w tym prawnej) za żadne wtyczki \"zewnętrzne\".\nUżywaj ich na własne ryzyko. Wszelkie błędy/problemy prosimy zgłaszać w repozytorium wtyczki.\n\nJeśli jakakolwiek wtyczka \"zewnętrzna\" narusza ToS/DMCA jakiejkolwiek usługi/podmiotu prawnego, prosimy o kontakt z autorem wtyczki \"zewnętrznej\" lub platformą hostingową, np. GitHub/Codeberg, w celu podjęcia działań. Wymienione powyżej (oznaczone jako \"zewnętrzne\") są publicznymi wtyczkami utrzymywanymi przez społeczność. Nie kuratujemy ich, więc nie możemy podjąć żadnych działań w ich sprawie.\n\n", + "input_does_not_match_format": "Wprowadzony tekst nie pasuje do wymaganego formatu", + "metadata_provider_plugins": "Wtyczki dostawców metadanych", + "paste_plugin_download_url": "Wklej adres URL do pobrania lub adres URL repozytorium GitHub/Codeberg lub bezpośredni link do pliku .smplug", + "download_and_install_plugin_from_url": "Pobierz i zainstaluj wtyczkę z adresu URL", + "failed_to_add_plugin_error": "Nie udało się dodać wtyczki: {error}", + "upload_plugin_from_file": "Prześlij wtyczkę z pliku", + "installed": "Zainstalowane", + "available_plugins": "Dostępne wtyczki", + "configure_your_own_metadata_plugin": "Skonfiguruj własnego dostawcę metadanych dla playlisty/albumu/artysty/kanału", + "audio_scrobblers": "Scrobblery audio", + "scrobbling": "Scrobbling", + "download_music_format": "Format pobierania muzyki", + "streaming_music_format": "Format strumieniowania muzyki", + "download_music_quality": "Jakość pobierania", + "streaming_music_quality": "Jakość strumieniowania", + "default_metadata_source": "Domyślne źródło metadanych", + "set_default_metadata_source": "Ustaw domyślne źródło metadanych", + "default_audio_source": "Domyślne źródło audio", + "set_default_audio_source": "Ustaw domyślne źródło audio", + "plugins": "Wtyczki", + "configure_plugins": "Skonfiguruj własne wtyczki dostawców metadanych i źródeł audio", + "source": "Źródło: ", + "uncompressed": "Nieskompresowany", + "dab_music_source_description": "Dla audiofilów. Oferuje strumienie audio wysokiej jakości/lossless. Precyzyjne dopasowanie utworów na podstawie ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_pt.arb b/lib/l10n/app_pt.arb index 72841eab..fa7845c3 100644 --- a/lib/l10n/app_pt.arb +++ b/lib/l10n/app_pt.arb @@ -401,5 +401,94 @@ "export_cache_files": "Exportar Arquivos em Cache", "found_n_files": "Encontrados {count} arquivos", "export_cache_confirmation": "Deseja exportar estes arquivos para", - "exported_n_out_of_m_files": "Exportados {filesExported} de {files} arquivos" + "exported_n_out_of_m_files": "Exportados {filesExported} de {files} arquivos", + "playlist": "Playlist", + "no_loop": "Sem loop", + "generate": "Gerar", + "undo": "Desfazer", + "download_all": "Baixar tudo", + "add_all_to_playlist": "Adicionar tudo à playlist", + "add_all_to_queue": "Adicionar tudo à fila", + "play_all_next": "Reproduzir tudo a seguir", + "pause": "Pausar", + "view_all": "Ver tudo", + "no_tracks_added_yet": "Parece que você ainda não adicionou nenhuma faixa", + "no_tracks": "Parece que não há faixas aqui", + "no_tracks_listened_yet": "Parece que você ainda não ouviu nada", + "not_following_artists": "Você não está seguindo nenhum artista", + "no_favorite_albums_yet": "Parece que você ainda não adicionou nenhum álbum aos favoritos", + "no_logs_found": "Nenhum log encontrado", + "youtube_engine": "Motor YouTube", + "youtube_engine_not_installed_title": "{engine} não está instalado", + "youtube_engine_not_installed_message": "{engine} não está instalado no seu sistema.", + "youtube_engine_set_path": "Certifique-se de que está disponível na variável PATH ou\ndefina o caminho absoluto para o executável {engine} abaixo", + "youtube_engine_unix_issue_message": "Em sistemas macOS/Linux/unix, definir o caminho no .zshrc/.bashrc/.bash_profile etc. não funcionará.\nVocê precisa definir o caminho no arquivo de configuração do shell", + "download": "Baixar", + "file_not_found": "Arquivo não encontrado", + "custom": "Personalizado", + "add_custom_url": "Adicionar URL personalizada", + "edit_port": "Editar porta", + "port_helper_msg": "O padrão é -1, que indica um número aleatório. Se você tiver um firewall configurado, é recomendável definir isso.", + "connect_request": "Permitir que {client} se conecte?", + "connection_request_denied": "Conexão negada. O usuário negou o acesso .", + "hipotetical_calculation": "*Isso é calculado com base no pagamento médio por stream de plataformas de streaming de música online de US$ 0,003 a US$ 0,005. Esta é uma estimativa hipotética para dar ao usuário uma ideia de quanto ele teria pago aos artistas se ouvisse sua música em diferentes plataformas de streaming de música.", + "an_error_occurred": "Ocorreu um erro", + "copy_to_clipboard": "Copiar para a área de transferência", + "view_logs": "Ver logs", + "retry": "Tentar novamente", + "no_default_metadata_provider_selected": "Você não tem um provedor de metadados padrão definido", + "manage_metadata_providers": "Gerenciar provedores de metadados", + "open_link_in_browser": "Abrir link no navegador?", + "do_you_want_to_open_the_following_link": "Você deseja abrir o seguinte link", + "unsafe_url_warning": "Pode ser inseguro abrir links de fontes não confiáveis. Tenha cautela!\nVocê também pode copiar o link para sua área de transferência.", + "copy_link": "Copiar link", + "building_your_timeline": "Construindo sua linha do tempo com base em suas audições...", + "official": "Oficial", + "author_name": "Autor: {author}", + "third_party": "Terceiros", + "plugin_requires_authentication": "Plugin requer autenticação", + "update_available": "Atualização disponível", + "supports_scrobbling": "Suporta scrobbling", + "plugin_scrobbling_info": "Este plugin faz o scrobbling de sua música para gerar seu histórico de audição.", + "default_plugin": "Padrão", + "set_default": "Definir como padrão", + "support": "Suporte", + "support_plugin_development": "Apoiar o desenvolvimento do plugin", + "can_access_name_api": "- Pode acessar a API **{name}**", + "do_you_want_to_install_this_plugin": "Você deseja instalar este plugin?", + "third_party_plugin_warning": "Este plugin é de um repositório de terceiros. Certifique-se de que você confia na fonte antes de instalá-lo.", + "author": "Autor", + "this_plugin_can_do_following": "Este plugin pode fazer o seguinte", + "install": "Instalar", + "install_a_metadata_provider": "Instalar um provedor de metadados", + "no_tracks_playing": "Nenhuma música sendo reproduzida no momento", + "synced_lyrics_not_available": "As letras sincronizadas não estão disponíveis para esta música. Por favor, use a aba", + "plain_lyrics": "Letras simples", + "tab_instead": "em vez disso.", + "disclaimer": "Aviso", + "third_party_plugin_dmca_notice": "A equipe Spotube não se responsabiliza (incluindo legalmente) por quaisquer plugins de \"terceiros\".\nUse-os por sua conta e risco. Para quaisquer bugs/problemas, por favor, relate-os ao repositório do plugin.\n\nSe algum plugin de \"terceiros\" estiver violando os Termos de Serviço/DMCA de qualquer serviço/entidade legal, por favor, peça ao autor do plugin \"terceiro\" ou à plataforma de hospedagem, por exemplo, GitHub/Codeberg, para tomar medidas. Os plugins listados acima (rotulados como \"terceiros\") são todos plugins públicos/mantidos pela comunidade. Não os estamos curando, então não podemos tomar nenhuma medida sobre eles.\n\n", + "input_does_not_match_format": "A entrada não corresponde ao formato exigido", + "metadata_provider_plugins": "Plugins do provedor de metadados", + "paste_plugin_download_url": "Cole a url de download ou a url do repositório GitHub/Codeberg ou o link direto para o arquivo .smplug", + "download_and_install_plugin_from_url": "Baixar e instalar o plugin a partir da url", + "failed_to_add_plugin_error": "Falha ao adicionar plugin: {error}", + "upload_plugin_from_file": "Carregar plugin a partir de arquivo", + "installed": "Instalado", + "available_plugins": "Plugins disponíveis", + "configure_your_own_metadata_plugin": "Configure seu próprio provedor de metadados de playlist/álbum/artista/feed", + "audio_scrobblers": "Scrobblers de áudio", + "scrobbling": "Scrobbling", + "download_music_format": "Formato de download de música", + "streaming_music_format": "Formato de streaming de música", + "download_music_quality": "Qualidade de download", + "streaming_music_quality": "Qualidade de streaming", + "default_metadata_source": "Fonte padrão de metadados", + "set_default_metadata_source": "Definir fonte padrão de metadados", + "default_audio_source": "Fonte de áudio padrão", + "set_default_audio_source": "Definir fonte de áudio padrão", + "plugins": "Plugins", + "configure_plugins": "Configure seus próprios plugins de provedores de metadados e fontes de áudio", + "source": "Fonte: ", + "uncompressed": "Não comprimido", + "dab_music_source_description": "Para audiófilos. Fornece streams de áudio de alta qualidade/sem perdas. Correspondência precisa de faixas baseada em ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_ru.arb b/lib/l10n/app_ru.arb index 6be53ba9..2e864268 100644 --- a/lib/l10n/app_ru.arb +++ b/lib/l10n/app_ru.arb @@ -401,5 +401,94 @@ "export_cache_files": "Экспортировать кэшированные файлы", "found_n_files": "Найдено {count} файлов", "export_cache_confirmation": "Вы хотите экспортировать эти файлы в", - "exported_n_out_of_m_files": "Экспортировано {filesExported} из {files} файлов" + "exported_n_out_of_m_files": "Экспортировано {filesExported} из {files} файлов", + "playlist": "Плейлист", + "no_loop": "Без повтора", + "generate": "Генерировать", + "undo": "Отменить", + "download_all": "Скачать все", + "add_all_to_playlist": "Добавить все в плейлист", + "add_all_to_queue": "Добавить все в очередь", + "play_all_next": "Воспроизвести все следующее", + "pause": "Пауза", + "view_all": "Просмотреть все", + "no_tracks_added_yet": "Похоже, вы ещё не добавили ни одного трека", + "no_tracks": "Похоже, здесь нет треков", + "no_tracks_listened_yet": "Похоже, вы ещё ничего не слушали", + "not_following_artists": "Вы не подписаны на художников", + "no_favorite_albums_yet": "Похоже, вы ещё не добавили ни одного альбома в избранное", + "no_logs_found": "Логи не найдены", + "youtube_engine": "YouTube Движок", + "youtube_engine_not_installed_title": "{engine} не установлен", + "youtube_engine_not_installed_message": "{engine} не установлен в вашей системе.", + "youtube_engine_set_path": "Убедитесь, что он доступен в переменной PATH или\nустановите абсолютный путь к исполнимому файлу {engine} ниже", + "youtube_engine_unix_issue_message": "В macOS/Linux/Unix-подобных ОС, установка пути в .zshrc/.bashrc/.bash_profile и т.д. не будет работать.\nВы должны установить путь в файле конфигурации оболочки", + "download": "Скачать", + "file_not_found": "Файл не найден", + "custom": "Пользовательский", + "add_custom_url": "Добавить пользовательский URL", + "edit_port": "Редактировать порт", + "port_helper_msg": "По умолчанию -1, что означает случайное число. Если у вас настроен брандмауэр, рекомендуется установить это.", + "connect_request": "Разрешить {client} подключение?", + "connection_request_denied": "Подключение отклонено. Пользователь отказал в доступе.", + "hipotetical_calculation": "*Это рассчитано на основе средней выплаты за прослушивание на онлайн-платформах для потоковой передачи музыки в размере от 0,003 до 0,005 долларов США. Это гипотетический расчет, чтобы дать пользователю представление о том, сколько бы они заплатили артистам, если бы слушали их песни на разных музыкальных стриминговых платформах.", + "an_error_occurred": "Произошла ошибка", + "copy_to_clipboard": "Скопировать в буфер обмена", + "view_logs": "Просмотреть журналы", + "retry": "Повторить", + "no_default_metadata_provider_selected": "Вы не выбрали поставщика метаданных по умолчанию", + "manage_metadata_providers": "Управление поставщиками метаданных", + "open_link_in_browser": "Открыть ссылку в браузере?", + "do_you_want_to_open_the_following_link": "Вы хотите открыть следующую ссылку", + "unsafe_url_warning": "Открытие ссылок из ненадежных источников может быть небезопасным. Будьте осторожны!\nВы также можете скопировать ссылку в буфер обмена.", + "copy_link": "Копировать ссылку", + "building_your_timeline": "Создание вашей временной шкалы на основе ваших прослушиваний...", + "official": "Официальный", + "author_name": "Автор: {author}", + "third_party": "Сторонний", + "plugin_requires_authentication": "Плагин требует аутентификации", + "update_available": "Доступно обновление", + "supports_scrobbling": "Поддерживает скробблинг", + "plugin_scrobbling_info": "Этот плагин скробблит вашу музыку для создания вашей истории прослушиваний.", + "default_plugin": "По умолчанию", + "set_default": "Установить по умолчанию", + "support": "Поддержка", + "support_plugin_development": "Поддержать разработку плагина", + "can_access_name_api": "- Может получить доступ к API **{name}**", + "do_you_want_to_install_this_plugin": "Вы хотите установить этот плагин?", + "third_party_plugin_warning": "Этот плагин из стороннего репозитория. Пожалуйста, убедитесь, что вы доверяете источнику перед установкой.", + "author": "Автор", + "this_plugin_can_do_following": "Этот плагин может выполнять следующее", + "install": "Установить", + "install_a_metadata_provider": "Установить поставщика метаданных", + "no_tracks_playing": "В настоящее время не воспроизводится ни один трек", + "synced_lyrics_not_available": "Синхронизированные тексты недоступны для этой песни. Пожалуйста, используйте вкладку", + "plain_lyrics": "Простые тексты", + "tab_instead": "вместо этого.", + "disclaimer": "Отказ от ответственности", + "third_party_plugin_dmca_notice": "Команда Spotube не несет никакой ответственности (в том числе юридической) за какие-либо \"сторонние\" плагины.\nПожалуйста, используйте их на свой страх и риск. О любых ошибках/проблемах сообщайте в репозиторий плагина.\n\nЕсли какой-либо \"сторонний\" плагин нарушает ToS/DMCA какого-либо сервиса/юридического лица, пожалуйста, попросите автора плагина \"стороннего\" или хостинговую платформу, например, GitHub/Codeberg, принять меры. Перечисленные выше (помеченные как \"сторонние\") являются общедоступными/поддерживаемыми сообществом плагинами. Мы не курируем их, поэтому не можем принимать по ним никаких мер.\n\n", + "input_does_not_match_format": "Введенные данные не соответствуют требуемому формату", + "metadata_provider_plugins": "Плагины поставщика метаданных", + "paste_plugin_download_url": "Вставьте URL-адрес для загрузки или URL-адрес репозитория GitHub/Codeberg или прямую ссылку на файл .smplug", + "download_and_install_plugin_from_url": "Загрузить и установить плагин по URL-адресу", + "failed_to_add_plugin_error": "Не удалось добавить плагин: {error}", + "upload_plugin_from_file": "Загрузить плагин из файла", + "installed": "Установлено", + "available_plugins": "Доступные плагины", + "configure_your_own_metadata_plugin": "Настройте свой собственный поставщик метаданных для плейлиста/альбома/артиста/ленты", + "audio_scrobblers": "Аудио скробблеры", + "scrobbling": "Скробблинг", + "download_music_format": "Формат загрузки музыки", + "streaming_music_format": "Формат потоковой музыки", + "download_music_quality": "Качество загрузки", + "streaming_music_quality": "Качество стриминга", + "default_metadata_source": "Источник метаданных по умолчанию", + "set_default_metadata_source": "Задать источник метаданных по умолчанию", + "default_audio_source": "Источник аудио по умолчанию", + "set_default_audio_source": "Задать источник аудио по умолчанию", + "plugins": "Плагины", + "configure_plugins": "Настройте собственные плагины провайдеров метаданных и источников аудио", + "source": "Источник: ", + "uncompressed": "Несжатый", + "dab_music_source_description": "Для аудиофилов. Предоставляет высококачественные/lossless аудиопотоки. Точное совпадение треков по ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_ta.arb b/lib/l10n/app_ta.arb new file mode 100644 index 00000000..6cea7b1a --- /dev/null +++ b/lib/l10n/app_ta.arb @@ -0,0 +1,492 @@ +{ + "guest": "விருந்தினர்", + "browse": "உலாவு", + "search": "தேடுக", + "library": "நூலகம்", + "lyrics": "பாடல் வரிகள்", + "settings": "அமைப்புகள்", + "genre_categories_filter": "வகைகள் அல்லது பாணிகளை வடிகட்டுக...", + "genre": "பாணி", + "personalized": "தனிப்பயனாக்கப்பட்ட", + "featured": "சிறப்பிடம் பெற்ற", + "new_releases": "புதிய வெளியீடுகள்", + "songs": "பாடல்கள்", + "playing_track": "{track} இயங்குகிறது", + "queue_clear_alert": "இது தற்போதைய வரிசையை அழிக்கும். {track_length} பாடல்கள் நீக்கப்படும்\nதொடர விரும்புகிறீர்களா?", + "load_more": "மேலும் ஏற்றுக", + "playlists": "பாடல் பட்டியல்கள்", + "artists": "கலைஞர்கள்", + "albums": "ஆல்பங்கள்", + "tracks": "பாடல்கள்", + "downloads": "பதிவிறக்கங்கள்", + "filter_playlists": "உங்கள் பாடல் பட்டியல்களை வடிகட்டுக...", + "liked_tracks": "விரும்பிய பாடல்கள்", + "liked_tracks_description": "உங்கள் விரும்பிய பாடல்கள் அனைத்தும்", + "playlist": "பாடல் பட்டியல்", + "create_a_playlist": "பாடல் பட்டியலை உருவாக்குக", + "update_playlist": "பாடல் பட்டியலைப் புதுப்பிக்க", + "create": "உருவாக்கு", + "cancel": "ரத்து செய்", + "update": "புதுப்பி", + "playlist_name": "பாடல் பட்டியல் பெயர்", + "name_of_playlist": "பாடல் பட்டியலின் பெயர்", + "description": "விளக்கம்", + "public": "பொது", + "collaborative": "கூட்டு", + "search_local_tracks": "உள்ளூர் பாடல்களைத் தேடுக...", + "play": "இயக்கு", + "delete": "அழி", + "none": "எதுவுமில்லை", + "sort_a_z": "A-Z வரிசைப்படுத்து", + "sort_z_a": "Z-A வரிசைப்படுத்து", + "sort_artist": "கலைஞர் மூலம் வரிசைப்படுத்து", + "sort_album": "ஆல்பம் மூலம் வரிசைப்படுத்து", + "sort_duration": "கால அளவு மூலம் வரிசைப்படுத்து", + "sort_tracks": "பாடல்களை வரிசைப்படுத்து", + "currently_downloading": "தற்போது பதிவிறக்குகிறது ({tracks_length})", + "cancel_all": "அனைத்தையும் ரத்து செய்", + "filter_artist": "கலைஞர்களை வடிகட்டுக...", + "followers": "{followers} பின்தொடர்பவர்கள்", + "add_artist_to_blacklist": "கலைஞரை தடைப்பட்டியலில் சேர்க்க", + "top_tracks": "சிறந்த பாடல்கள்", + "fans_also_like": "ரசிகர்கள் விரும்புவது", + "loading": "ஏற்றுகிறது...", + "artist": "கலைஞர்", + "blacklisted": "தடைப்பட்டியலில் உள்ளது", + "following": "பின்தொடர்கிறது", + "follow": "பின்தொடர்", + "artist_url_copied": "கலைஞர் URL கிளிப்போர்டுக்கு நகலெடுக்கப்பட்டது", + "added_to_queue": "{tracks} பாடல்கள் வரிசையில் சேர்க்கப்பட்டன", + "filter_albums": "ஆல்பங்களை வடிகட்டுக...", + "synced": "ஒத்திசைக்கப்பட்டது", + "plain": "சாதாரண", + "shuffle": "கலக்கு", + "search_tracks": "பாடல்களைத் தேடுக...", + "released": "வெளியிடப்பட்டது", + "error": "பிழை {error}", + "title": "தலைப்பு", + "time": "நேரம்", + "more_actions": "மேலும் செயல்கள்", + "download_count": "பதிவிறக்கு ({count})", + "add_count_to_playlist": "({count}) பாடல் பட்டியலில் சேர்", + "add_count_to_queue": "({count}) வரிசையில் சேர்", + "play_count_next": "({count}) அடுத்து இயக்கு", + "album": "ஆல்பம்", + "copied_to_clipboard": "{data} கிளிப்போர்டுக்கு நகலெடுக்கப்பட்டது", + "add_to_following_playlists": "{track} பின்வரும் பாடல் பட்டியல்களில் சேர்", + "add": "சேர்", + "added_track_to_queue": "{track} வரிசையில் சேர்க்கப்பட்டது", + "add_to_queue": "வரிசையில் சேர்", + "track_will_play_next": "{track} அடுத்து இயக்கப்படும்", + "play_next": "அடுத்து இயக்கு", + "removed_track_from_queue": "{track} வரிசையிலிருந்து நீக்கப்பட்டது", + "remove_from_queue": "வரிசையிலிருந்து நீக்கு", + "remove_from_favorites": "பிடித்தவையிலிருந்து நீக்கு", + "save_as_favorite": "பிடித்தவையாக சேமி", + "add_to_playlist": "பாடல் பட்டியலில் சேர்", + "remove_from_playlist": "பாடல் பட்டியலிலிருந்து நீக்கு", + "add_to_blacklist": "தடைப்பட்டியலில் சேர்", + "remove_from_blacklist": "தடைப்பட்டியலிலிருந்து நீக்கு", + "share": "பகிர்", + "mini_player": "சிறிய இயக்கி", + "slide_to_seek": "முன்னோக்கி அல்லது பின்னோக்கி செல்ல சறுக்கவும்", + "shuffle_playlist": "பாடல் பட்டியலை கலக்கு", + "unshuffle_playlist": "பாடல் பட்டியலை கலக்காதே", + "previous_track": "முந்தைய பாடல்", + "next_track": "அடுத்த பாடல்", + "pause_playback": "இயக்கத்தை நிறுத்து", + "resume_playback": "இயக்கத்தை தொடர்", + "loop_track": "பாடலை சுழற்று", + "no_loop": "சுழற்சி இல்லை", + "repeat_playlist": "பாடல் பட்டியலை மீண்டும் இயக்கு", + "queue": "வரிசை", + "alternative_track_sources": "மாற்று பாடல் மூலங்கள்", + "download_track": "பாடலைப் பதிவிறக்கு", + "tracks_in_queue": "வரிசையில் {tracks} பாடல்கள்", + "clear_all": "அனைத்தையும் அழி", + "show_hide_ui_on_hover": "மேலே வரும்போது UI ஐக் காட்டு/மறை", + "always_on_top": "எப்போதும் மேலே", + "exit_mini_player": "சிறிய இயக்கியிலிருந்து வெளியேறு", + "download_location": "பதிவிறக்க இடம்", + "local_library": "உள்ளூர் நூலகம்", + "add_library_location": "நூலகத்தில் சேர்", + "remove_library_location": "நூலகத்திலிருந்து நீக்கு", + "account": "கணக்கு", + "login_with_spotify": "உங்கள் Spotify கணக்கில் உள்நுழைக", + "connect_with_spotify": "Spotify உடன் இணைக்கவும்", + "logout": "வெளியேறு", + "logout_of_this_account": "இந்த கணக்கிலிருந்து வெளியேறு", + "language_region": "மொழி & பிராந்தியம்", + "language": "மொழி", + "system_default": "கணினி இயல்புநிலை", + "market_place_region": "சந்தை பிராந்தியம்", + "recommendation_country": "பரிந்துரை நாடு", + "appearance": "தோற்றம்", + "layout_mode": "அமைப்பு முறை", + "override_layout_settings": "தளவமைப்பு அமைப்புகளை மாற்றியமை", + "adaptive": "தகவமைப்பு", + "compact": "சுருக்கமான", + "extended": "விரிவான", + "theme": "தீம்", + "dark": "இருள்", + "light": "வெளிர்", + "system": "கணினி வழி", + "accent_color": "அழுத்த நிறம்", + "sync_album_color": "ஆல்பம் நிறத்தை ஒத்திசை", + "sync_album_color_description": "ஆல்பம் படத்தின் முக்கிய நிறத்தை அழுத்த நிறமாகப் பயன்படுத்துகிறது", + "playback": "பின்னணி", + "audio_quality": "ஒலி தரம்", + "high": "உயர்", + "low": "குறைந்த", + "pre_download_play": "முன்பதிவிறக்கம் மற்றும் இயக்கம்", + "pre_download_play_description": "ஒலியை ஸ்ட்ரீம் செய்வதற்குப் பதிலாக, பைட்டுகளைப் பதிவிறக்கி இயக்கவும் (அதிக பேண்ட்விட்த் பயனர்களுக்கு பரிந்துரைக்கப்படுகிறது)", + "skip_non_music": "இசையல்லாத பகுதிகளைத் தவிர் (SponsorBlock)", + "blacklist_description": "தடைசெய்யப்பட்ட பாடல்கள் மற்றும் கலைஞர்கள்", + "wait_for_download_to_finish": "தற்போதைய பதிவிறக்கம் முடியும் வரை காத்திருக்கவும்", + "desktop": "கணினி", + "close_behavior": "மூடும் நடத்தை", + "close": "மூடு", + "minimize_to_tray": "ட்ரேயை குறைக்கவும்", + "show_tray_icon": "ட்ரே ஐகானைக் காட்டு", + "about": "பற்றி", + "u_love_spotube": "நீங்கள் Spotube ஐ நேசிக்கிறீர்கள் என்பது எங்களுக்குத் தெரியும்", + "check_for_updates": "புதுப்பிப்புகளைச் சரிபார்", + "about_spotube": "Spotube பற்றி", + "blacklist": "தடைப்பட்டியல்", + "please_sponsor": "தயவுசெய்து ஆதரவு/நன்கொடை அளியுங்கள்", + "spotube_description": "Spotube, ஒரு லேசான, பல தளங்களில் இயங்கும், அனைவருக்கும் இலவசமான spotify கிளையன்ட்", + "version": "பதிப்பு", + "build_number": "கட்டமைப்பு எண்", + "founder": "நிறுவனர்", + "repository": "களஞ்சியம்", + "bug_issues": "பிழை_சிக்கல்கள்", + "made_with": "வங்காளதேசத்திலிருந்து🇧🇩 ❤️ உருவாக்கப்பட்டது", + "kingkor_roy_tirtho": "கிங்கர் ராய் திர்தோ", + "copyright": "© 2021-{current_year} கிங்கர் ராய் திர்தோ", + "license": "உரிமம்", + "add_spotify_credentials": "தொடங்குவதற்கு உங்கள் spotify சான்றுகளைச் சேர்க்கவும்", + "credentials_will_not_be_shared_disclaimer": "கவலைப்பட வேண்டாம், உங்கள் சான்றுகள் எதுவும் சேகரிக்கப்படாது அல்லது யாருடனும் பகிரப்படாது", + "know_how_to_login": "இதை எப்படி செய்வது என்று தெரியவில்லையா?", + "follow_step_by_step_guide": "படிப்படியான வழிகாட்டியைப் பின்பற்றவும்", + "spotify_cookie": "Spotify {name} நட்புநிரல்", + "cookie_name_cookie": "{name} நட்புநிரல்", + "fill_in_all_fields": "அனைத்து களங்களையும் நிரப்பவும்", + "submit": "சமர்ப்பி", + "exit": "வெளியேறு", + "previous": "முந்தைய", + "next": "அடுத்து", + "done": "முடிந்தது", + "step_1": "முதல் படி", + "first_go_to": "முதலில், செல்லவேண்டியது", + "login_if_not_logged_in": "நீங்கள் உள்நுழையவில்லை என்றால் உள்நுழைக/பதிவுசெய்க", + "step_2": "இரண்டாம் படி", + "step_2_steps": "1. நீங்கள் உள்நுழைந்தவுடன், F12 ஐ அழுத்தவும் அல்லது வலது கிளிக் செய்து > ஆய்வு செய்யவும் உலாவி டெவ்டூல்களைத் திறக்கவும்.\n2. பின்னர் \"பயன்பாடு\" தாவலுக்குச் செல்லவும் (Chrome, Edge, Brave போன்றவை) அல்லது \"சேமிப்பகம்\" தாவல் (Firefox, Palemoon போன்றவை)\n3. \"குக்கிகள்\" பிரிவுக்குச் சென்று பின்னர் \"https://accounts.spotify.com\" பிரிவுக்குச் செல்லவும்", + "step_3": "மூன்றாம் படி", + "step_3_steps": "\"sp_dc\" நட்புநிரலின் மதிப்பை நகலெடுக்கவும்", + "success_emoji": "வெற்றி🥳", + "success_message": "இப்போது நீங்கள் உங்கள் Spotify கணக்கில் வெற்றிகரமாக உள்நுழைந்துள்ளீர்கள். நல்லது, நண்பரே!", + "step_4": "நான்காம் படி", + "step_4_steps": "நகலெடுக்கப்பட்ட \"sp_dc\" மதிப்பை ஒட்டவும்", + "something_went_wrong": "ஏதோ தவறு நடந்துவிட்டது", + "piped_instance": "Piped சேவையகம் நிகழ்வு", + "piped_description": "பாடல் பொருத்தத்திற்குப் பயன்படுத்த வேண்டிய Piped சேவையகம் நிகழ்வு", + "piped_warning": "அவற்றில் சில நன்றாக வேலை செய்யாமல் இருக்கலாம். எனவே உங்கள் சொந்த ஆபத்தில் பயன்படுத்தவும்", + "invidious_instance": "Invidious சேவையக நிகழ்வு", + "invidious_description": "பாடல் பொருத்தத்திற்குப் பயன்படுத்த வேண்டிய Invidious சேவையக நிகழ்வு", + "invidious_warning": "அவற்றில் சில நன்றாக வேலை செய்யாமல் இருக்கலாம். எனவே உங்கள் சொந்த ஆபத்தில் பயன்படுத்தவும்", + "generate": "உருவாக்கு", + "track_exists": "பாடல் {track} ஏற்கனவே உள்ளது", + "replace_downloaded_tracks": "பதிவிறக்கம் செய்யப்பட்ட அனைத்து பாடல்களையும் மாற்றவும்", + "skip_download_tracks": "பதிவிறக்கம் செய்யப்பட்ட அனைத்து பாடல்களையும் தவிர்க்கவும்", + "do_you_want_to_replace": "ஏற்கனவே உள்ள பாடலை மாற்ற விரும்புகிறீர்களா?", + "replace": "மாற்று", + "skip": "தவிர்", + "select_up_to_count_type": "{count} {type} வரை தேர்ந்தெடுக்கவும்", + "select_genres": "வகைகளைத் தேர்ந்தெடுக்கவும்", + "add_genres": "வகைகளைச் சேர்க்கவும்", + "country": "நாடு", + "number_of_tracks_generate": "உருவாக்க வேண்டிய பாடல்களின் எண்ணிக்கை", + "acousticness": "அகவுஸ்டிக்னெஸ்", + "danceability": "நடனத்தன்மை", + "energy": "ஆற்றல்", + "instrumentalness": "கருவித்தன்மை", + "liveness": "உயிர்ப்புத்தன்மை", + "loudness": "ஒலி அளவு", + "speechiness": "பேச்சுத்தன்மை", + "valence": "உணர்வு", + "popularity": "பிரபலம்", + "key": "இசை குறிப்பு", + "duration": "கால அளவு (வினாடிகள்)", + "tempo": "வேகம் (BPM)", + "mode": "முறை", + "time_signature": "நேர கையொப்பம்", + "short": "குறுகிய", + "medium": "நடுத்தர", + "long": "நீண்ட", + "min": "குறைந்தபட்சம்", + "max": "அதிகபட்சம்", + "target": "இலக்கு", + "moderate": "மிதமான", + "deselect_all": "அனைத்தையும் தேர்வுநீக்கு", + "select_all": "அனைத்தையும் தேர்ந்தெடு", + "are_you_sure": "உறுதியாக இருக்கிறீர்களா?", + "generating_playlist": "உங்கள் தனிப்பயன்பாட்டிற்கான பாடல் பட்டியலை உருவாக்குகிறது...", + "selected_count_tracks": "{count} பாடல்கள் தேர்ந்தெடுக்கப்பட்டன", + "download_warning": "நீங்கள் அனைத்து பாடல்களையும் மொத்தமாக பதிவிறக்கினால், நீங்கள் தெளிவாக இசையைத் திருடுகிறீர்கள் மற்றும் இசையின் படைப்பாற்றல் சமூகத்திற்கு சேதம் விளைவிக்கிறீர்கள். நீங்கள் இதை அறிந்திருக்கிறீர்கள் என்று நம்புகிறேன். எப்போதும், கலைஞரின் கடின உழைப்பை மதித்து ஆதரிக்க முயற்சி செய்யுங்கள்", + "download_ip_ban_warning": "மேலும், அதிகப்படியான பதிவிறக்க கோரிக்கைகள் காரணமாக உங்கள் IP YouTube இல் தடைசெய்யப்படலாம். IP தடை என்பது குறைந்தது 2-3 மாதங்களுக்கு அந்த IP சாதனத்திலிருந்து YouTube ஐப் பயன்படுத்த முடியாது (நீங்கள் உள்நுழைந்திருந்தாலும் கூட). இது ஒருபோதும் நடந்தால் Spotube பொறுப்பேற்காது", + "by_clicking_accept_terms": "'ஏற்றுக்கொள்' என்பதைக் கிளிக் செய்வதன் மூலம் பின்வரும் விதிமுறைகளுக்கு நீங்கள் ஒப்புக்கொள்கிறீர்கள்:", + "download_agreement_1": "நான் இசையைத் திருடுகிறேன் என்பது எனக்குத் தெரியும். நான் கெட்டவன்", + "download_agreement_2": "நான் கலைஞரை முடிந்தவரை ஆதரிப்பேன், அவர்களின் கலைக்கு பணம் செலுத்த எனக்கு பணம் இல்லாததால் மட்டுமே இதைச் செய்கிறேன்", + "download_agreement_3": "என் IP YouTube இல் தடைசெய்யப்படலாம் என்பதை நான் முழுமையாக அறிவேன், மேலும் என் தற்போதைய செயலால் ஏற்படும் எந்த விபத்துகளுக்கும் Spotube அல்லது அதன் உரிமையாளர்கள்/பங்களிப்பாளர்களை பொறுப்பாக்க மாட்டேன்", + "decline": "மறு", + "accept": "ஏற்றுக்கொள்", + "details": "விவரங்கள்", + "youtube": "YouTube", + "channel": "சேனல்", + "likes": "விருப்பங்கள்", + "dislikes": "விருப்பமில்லாதவை", + "views": "பார்வைகள்", + "streamUrl": "ஸ்ட்ரீம் URL", + "stop": "நிறுத்து", + "sort_newest": "புதிதாக சேர்க்கப்பட்டவற்றை வரிசைப்படுத்து", + "sort_oldest": "பழமையானவற்றை வரிசைப்படுத்து", + "sleep_timer": "உறக்க நேரம்", + "mins": "{minutes} நிமிடங்கள்", + "hours": "{hours} மணிநேரங்கள்", + "hour": "{hours} மணிநேரம்", + "custom_hours": "தனிப்பயன் மணிநேரங்கள்", + "logs": "பதிவுகள்", + "developers": "உருவாக்குநர்கள்", + "not_logged_in": "நீங்கள் உள்நுழையவில்லை", + "search_mode": "தேடல் முறை", + "audio_source": "ஒலி மூலம்", + "ok": "சரி", + "failed_to_encrypt": "குறியாக்கம் தோல்வியடைந்தது", + "encryption_failed_warning": "Spotube உங்கள் தரவை பாதுகாப்பாக சேமிக்க குறியாக்கத்தைப் பயன்படுத்துகிறது. ஆனால் அவ்வாறு செய்ய முடியவில்லை. எனவே இது பாதுகாப்பற்ற சேமிப்பகத்திற்கு மாறும்\nநீங்கள் லினக்ஸ் பயன்படுத்துகிறீர்கள் என்றால், எந்த ரகசிய சேவையும் (gnome-keyring, kde-wallet, keepassxc போன்றவை) நிறுவப்பட்டுள்ளதா என்பதை உறுதிப்படுத்தவும்", + "querying_info": "தகவலைக் கேட்கிறது...", + "piped_api_down": "Piped API செயலிழந்துள்ளது", + "piped_down_error_instructions": "Piped நிகழ்வு {pipedInstance} தற்போது செயலிழந்துள்ளது\n\nநிகழ்வை மாற்றவும் அல்லது 'API வகை'யை அதிகாரப்பூர்வ YouTube API க்கு மாற்றவும்\n\nமாற்றத்திற்குப் பிறகு பயன்பாட்டை மறுதொடக்கம் செய்வதை உறுதிப்படுத்தவும்", + "you_are_offline": "நீங்கள் தற்போது ஆஃப்லைனில் உள்ளீர்கள்", + "connection_restored": "உங்கள் இணைய இணைப்பு மீட்டெடுக்கப்பட்டது", + "use_system_title_bar": "கணினி தலைப்புப் பட்டியைப் பயன்படுத்தவும்", + "crunching_results": "முடிவுகளை செயலாக்குகிறது...", + "search_to_get_results": "முடிவுகளைப் பெற தேடவும்", + "use_amoled_mode": "கருமை நிற இருண்ட தீம்", + "pitch_dark_theme": "AMOLED முறை", + "normalize_audio": "ஒலியை சீரமை", + "change_cover": "அட்டையை மாற்று", + "add_cover": "அட்டையைச் சேர்", + "restore_defaults": "இயல்புநிலைகளை மீட்டமை", + "download_music_codec": "இசை கோடெக்கை பதிவிறக்கு", + "streaming_music_codec": "இசை கோடெக்கை ஸ்ட்ரீம் செய்", + "login_with_lastfm": "Last.fm உடன் உள்நுழைக", + "connect": "இணை", + "disconnect_lastfm": "Last.fm இலிருந்து துண்டி", + "disconnect": "துண்டி", + "username": "பயனர்பெயர்", + "password": "கடவுச்சொல்", + "login": "உள்நுழைக", + "login_with_your_lastfm": "உங்கள் Last.fm கணக்குடன் உள்நுழைக", + "scrobble_to_lastfm": "Last.fm க்கு ஸ்க்ரோபிள் செய்", + "go_to_album": "ஆல்பத்திற்குச் செல்", + "discord_rich_presence": "Discord செழுமையான தோற்றம்", + "browse_all": "அனைத்தையும் உலாவு", + "genres": "வகைகள்", + "explore_genres": "வகைகளை ஆராயுங்கள்", + "friends": "நண்பர்கள்", + "no_lyrics_available": "மன்னிக்கவும், இந்தப் பாடலுக்கான பாடல் வரிகளைக் கண்டுபிடிக்க முடியவில்லை", + "start_a_radio": "வானொலியைத் தொடங்கு", + "how_to_start_radio": "வானொலியை எவ்வாறு தொடங்க விரும்புகிறீர்கள்?", + "replace_queue_question": "தற்போதைய வரிசையை மாற்ற விரும்புகிறீர்களா அல்லது அதனுடன் சேர்க்க விரும்புகிறீர்களா?", + "endless_playback": "முடிவற்ற இயக்கம்", + "delete_playlist": "பாடல் பட்டியலை நீக்கு", + "delete_playlist_confirmation": "இந்த பாடல் பட்டியலை நீக்க விரும்புகிறீர்களா?", + "local_tracks": "உள்ளூர் பாடல்கள்", + "local_tab": "உள்ளூர்", + "song_link": "பாடல் இணைப்பு", + "skip_this_nonsense": "இந்த அர்த்தமற்றதைத் தவிர்", + "freedom_of_music": "\"இசையின் சுதந்திரம்\"", + "freedom_of_music_palm": "\"உங்கள் கைகளில் இசையின் சுதந்திரம்\"", + "get_started": "தொடங்குவோம்", + "youtube_source_description": "பரிந்துரைக்கப்படுகிறது மற்றும் சிறப்பாக செயல்படுகிறது.", + "piped_source_description": "சுதந்திரமாக உணர்கிறீர்களா? YouTube போலவே ஆனால் மிகவும் சுதந்திரமானது.", + "jiosaavn_source_description": "தெற்காசியப் பிராந்தியத்திற்கு சிறந்தது.", + "invidious_source_description": "Piped ஐப் போன்றது ஆனால் அதிக கிடைக்கும் தன்மையுடன்.", + "highest_quality": "உயர்ந்த தரம்: {quality}", + "select_audio_source": "ஒலி மூலத்தைத் தேர்ந்தெடுக்கவும்", + "endless_playback_description": "வரிசையின் இறுதியில் புதிய பாடல்களை\nதானாகவே சேர்க்கவும்", + "choose_your_region": "உங்கள் பிராந்தியத்தைத் தேர்ந்தெடுக்கவும்", + "choose_your_region_description": "இது உங்கள் இருப்பிடத்திற்கான சரியான உள்ளடக்கத்தை\nSpotube காட்ட உதவும்.", + "choose_your_language": "உங்கள் மொழியைத் தேர்ந்தெடுக்கவும்", + "help_project_grow": "இந்த திட்டம் வளர உதவுங்கள்", + "help_project_grow_description": "Spotube ஒரு திறந்த மூல திட்டம். திட்டத்திற்கு பங்களிப்பு செய்வதன் மூலம், பிழைகளைப் புகாரளிப்பதன் மூலம் அல்லது புதிய அம்சங்களைப் பரிந்துரைப்பதன் மூலம் இந்தத் திட்டம் வளர உதவலாம்.", + "contribute_on_github": "GitHub இல் பங்களியுங்கள்", + "donate_on_open_collective": "Open Collective இல் நன்கொடை அளியுங்கள்", + "browse_anonymously": "அநாமதேயமாக உலாவுக", + "enable_connect": "இணைப்பை இயக்கு", + "enable_connect_description": "மற்ற சாதனங்களிலிருந்து Spotube ஐக் கட்டுப்படுத்தவும்", + "devices": "சாதனங்கள்", + "select": "தேர்ந்தெடு", + "connect_client_alert": "நீங்கள் {client} ஆல் கட்டுப்படுத்தப்படுகிறீர்கள்", + "this_device": "இந்த சாதனம்", + "remote": "தொலைநிலை", + "stats": "புள்ளிவிவரங்கள்", + "and_n_more": "மற்றும் {count} கூடுதலாக", + "recently_played": "சமீபத்தில் இயக்கியவை", + "browse_more": "மேலும் உலாவு", + "no_title": "தலைப்பு இல்லை", + "not_playing": "இயக்கப்படவில்லை", + "epic_failure": "மோசமான தோல்வி!", + "added_num_tracks_to_queue": "{tracks_length} பாடல்கள் வரிசையில் சேர்க்கப்பட்டன", + "spotube_has_an_update": "Spotube க்கு ஒரு புதுப்பிப்பு உள்ளது", + "download_now": "இப்போது பதிவிறக்கு", + "nightly_version": "Spotube Nightly {nightlyBuildNum} வெளியிடப்பட்டுள்ளது", + "release_version": "Spotube v{version} வெளியிடப்பட்டுள்ளது", + "read_the_latest": "சமீபத்திய ", + "release_notes": "வெளியீட்டு குறிப்புகளைப் படிக்கவும்", + "pick_color_scheme": "வண்ணத் திட்டத்தைத் தேர்ந்தெடுக்கவும்", + "save": "சேமி", + "choose_the_device": "சாதனத்தைத் தேர்ந்தெடுக்கவும்:", + "multiple_device_connected": "பல சாதனங்கள் இணைக்கப்பட்டுள்ளன.\nஇந்த செயல் நடைபெற வேண்டிய சாதனத்தைத் தேர்ந்தெடுக்கவும்", + "nothing_found": "எதுவும் கிடைக்கவில்லை", + "the_box_is_empty": "பெட்டி காலியாக உள்ளது", + "top_artists": "சிறந்த கலைஞர்கள்", + "top_albums": "சிறந்த ஆல்பங்கள்", + "this_week": "இந்த வாரம்", + "this_month": "இந்த மாதம்", + "last_6_months": "கடந்த 6 மாதங்கள்", + "this_year": "இந்த ஆண்டு", + "last_2_years": "கடந்த 2 ஆண்டுகள்", + "all_time": "எல்லா நேரமும்", + "powered_by_provider": "{providerName} ஆல் இயக்கப்படுகிறது", + "email": "மின்னஞ்சல்", + "profile_followers": "பின்தொடர்பவர்கள்", + "birthday": "பிறந்த நாள்", + "subscription": "சந்தா", + "not_born": "பிறக்கவில்லை", + "hacker": "ஹேக்கர்", + "profile": "சுயவிவரம்", + "no_name": "பெயர் இல்லை", + "edit": "திருத்து", + "user_profile": "பயனர் சுயவிவரம்", + "count_plays": "{count} முறை இசைக்கப்பட்டது", + "streaming_fees_hypothetical": "ஸ்ட்ரீமிங் கட்டணங்கள் (கற்பனை)", + "minutes_listened": "காலம் கேட்டது", + "streamed_songs": "ஸ்ட்ரீமிங் செய்யப்பட்ட பாடல்கள்", + "count_streams": "{count} ஸ்ட்ரீம்கள்", + "owned_by_you": "உங்களால் கொண்டது", + "copied_shareurl_to_clipboard": "நகலெடுக்கப்பட்டது {shareUrl} கிளிப்போர்டுக்காக", + "spotify_hipotetical_calculation": "*இது Spotify இன் ஒவ்வொரு ஸ்ட்ரீமிற்கும்\n$0.003 முதல் $0.005 வரை அளவீடு அடிப்படையில் கணக்கிடப்படுகிறது. இது ஒரு கற்பனை\nகணக்கீடு ஆகும், பயனர் எந்த அளவிற்கு கலைஞர்களுக்கு\nஅதோர் பாடலை Spotify மென்பொருளில் கேட்டால் எவ்வளவு பணம் செலுத்தினார்கள் என்பதைக் கண்டுபிடிக்க.", + "count_mins": "{minutes} நிமிடங்கள்", + "summary_minutes": "நிமிடங்கள்", + "summary_listened_to_music": "இசை கேட்டது", + "summary_songs": "பாடல்கள்", + "summary_streamed_overall": "மொத்தமாக ஸ்ட்ரீமிங்", + "summary_owed_to_artists": "கலைஞர்களுக்கு\nஇந்த மாதம் சொந்தமானது", + "summary_artists": "கலைஞர்கள்", + "summary_music_reached_you": "இசை உங்களுக்கு வந்தது", + "summary_full_albums": "முழு ஆல்பங்கள்", + "summary_got_your_love": "உங்கள் அன்பை பெற்றுக்கொண்டேன்", + "summary_playlists": "பாடல் பட்டியல்கள்", + "summary_were_on_repeat": "மீண்டும் மீண்டும் இருந்தன", + "total_money": "மொத்தம் {money}", + "webview_not_found": "வெப்வியூ கிடைக்கவில்லை", + "webview_not_found_description": "உங்கள் சாதனத்தில் எந்தவொரு வெப்வியூ இயக்கத்தை நிறுவவில்லை.\nஇது நிறுவப்பட்டிருந்தால், சுற்றுச்சூழல் பாதையில் PATH உள்ளது என்பதை உறுதிபடுத்தவும்\n\nநிறுவித்த பிறகு, செயலியை மறுதொடக்கம் செய்யவும்", + "unsupported_platform": "அதிர்ஷ்டகாத உருப்படியை ஆதரிக்கவில்லை", + "cache_music": "இசையை கேஷ் செய்", + "open": "திறக்கவும்", + "cache_folder": "கேஷ் அடைவு", + "export": "ஏற்றுமதி", + "clear_cache": "கேஷ் அழிக்கவும்", + "clear_cache_confirmation": "கேஷைப் அழிக்க விரும்புகிறீர்களா?", + "export_cache_files": "கேஷில் உள்ள கோப்புகளை ஏற்றுமதி செய்யவும்", + "found_n_files": "{count} கோப்புகள் கிடைத்தன", + "export_cache_confirmation": "இந்த கோப்புகளை ஏற்றுமதி செய்ய விரும்புகிறீர்களா?", + "exported_n_out_of_m_files": "{filesExported} கோப்புகள் ஏற்றுமதி செய்யப்பட்டன, {files} கோப்புகளில்", + "undo": "செயல்தவிர்", + "download_all": "அனைத்தையும் பதிவிறக்குக", + "add_all_to_playlist": "அனைத்தையும் பாடல் பட்டியலில் சேர்க்கவும்", + "add_all_to_queue": "அனைத்தையும் வரிசைப்படுத்து", + "play_all_next": "அடுத்த உள்ள அனைத்தையும் இயக்கு", + "pause": "நிறுத்து", + "view_all": "அனைத்தையும் காண்க", + "no_tracks_added_yet": "உங்கள் பாடல்களை இன்னும் சேர்க்கவில்லை என்றால் தெரியாதே", + "no_tracks": "இங்கு பாடல்கள் எதுவும் இல்லை", + "no_tracks_listened_yet": "இன்னும் எதையும் கேள்வியில்லை", + "not_following_artists": "நீங்கள் எந்த கலைஞரையும் பின்தொடரவில்லை", + "no_favorite_albums_yet": "நீங்கள் இன்னும் எந்த ஆல்பங்களையும் பிடித்தவையாகச் சேர்க்கவில்லை", + "no_logs_found": "பதிவுகள் எதுவும் கிடைக்கவில்லை", + "youtube_engine": "YouTube இயந்திரம்", + "youtube_engine_not_installed_title": "{engine} நிறுவியதில்லை", + "youtube_engine_not_installed_message": "{engine} உங்கள் கணினியில் நிறுவியதில்லை.", + "youtube_engine_set_path": "PATH மாறியில் கிடைக்கிறதா என்பதை உறுதிப்படுத்தவும் அல்லது\n{engine} செயல் செய்யக்கூடிய முறையை கீழே அமைக்கவும்", + "youtube_engine_unix_issue_message": "macOS/Linux/unix போல் OS இல், .zshrc/.bashrc/.bash_profile போன்றவை அமைப்பில் பாதையை PATH அமைப்பது இயலாது.\nநீங்கள்.shell configuration file இல் பாதையை அமைக்க வேண்டும்", + "download": "பதிவிறக்கு", + "file_not_found": "கோப்பு கிடைக்கவில்லை", + "custom": "தனிப்பயன்", + "add_custom_url": "தனிப்பயன் URL ஐச் சேர்க்கவும்", + "edit_port": "போர்டு திருத்தவும்", + "port_helper_msg": "இயல்புநிலை -1 ஆகும், இது சீரற்ற எண்ணை குறிக்கிறது. நீங்கள் தீயணைப்பு அமைக்கப்பட்டிருந்தால், இதை அமைப்பது பரிந்துரைக்கப்படுகிறது.", + "connect_request": "{client} க்கு இணைக்க அனுமதிக்கவா?", + "connection_request_denied": "இணைப்பு மறுக்கப்பட்டது. பயனர் அணுகலை மறுத்தார்.", + "hipotetical_calculation": "*இது சராசரி ஆன்லைன் இசை ஸ்ட்ரீமிங் தளத்தின் ஒரு ஸ்ட்ரீமிற்கான $0.003 முதல் $0.005 வரையிலான கட்டணத்தின் அடிப்படையில் கணக்கிடப்படுகிறது. இது ஒரு கற்பனையான கணக்கீடு ஆகும், இது பயனர்கள் வெவ்வேறு இசை ஸ்ட்ரீமிங் தளங்களில் தங்கள் பாடல்களைக் கேட்டால் கலைஞர்களுக்கு எவ்வளவு பணம் செலுத்தியிருப்பார்கள் என்பது குறித்த நுண்ணறிவை வழங்குகிறது.", + "an_error_occurred": "ஒரு பிழை ஏற்பட்டது", + "copy_to_clipboard": "கிளிப்போர்டுக்கு நகலெடுக்கவும்", + "view_logs": "பதிவுகளைப் பார்க்கவும்", + "retry": "மீண்டும் முயற்சிக்கவும்", + "no_default_metadata_provider_selected": "நீங்கள் எந்த இயல்புநிலை மெட்டாடேட்டா வழங்குநரையும் அமைக்கவில்லை", + "manage_metadata_providers": "மெட்டாடேட்டா வழங்குநர்களை நிர்வகிக்கவும்", + "open_link_in_browser": "இணைப்பை உலாவியில் திறக்கவா?", + "do_you_want_to_open_the_following_link": "பின்வரும் இணைப்பை நீங்கள் திறக்க விரும்புகிறீர்களா", + "unsafe_url_warning": "நம்பத்தகாத மூலங்களிலிருந்து இணைப்புகளைத் திறப்பது பாதுகாப்பற்றதாக இருக்கலாம். எச்சரிக்கையாக இருங்கள்!\nநீங்கள் இணைப்பை உங்கள் கிளிப்போர்டுக்கு நகலெடுக்கலாம்.", + "copy_link": "இணைப்பை நகலெடுக்கவும்", + "building_your_timeline": "உங்கள் கேட்டலின் அடிப்படையில் உங்கள் காலவரிசையை உருவாக்குகிறது...", + "official": "அதிகாரபூர்வமானது", + "author_name": "ஆசிரியர்: {author}", + "third_party": "மூன்றாம் தரப்பு", + "plugin_requires_authentication": "பிளகின் அங்கீகாரத்தைக் கோருகிறது", + "update_available": "புதுப்பிப்பு உள்ளது", + "supports_scrobbling": "ஸ்க்ரோப்ளிங்கை ஆதரிக்கிறது", + "plugin_scrobbling_info": "இந்த பிளகின் உங்கள் கேட்பதின் வரலாற்றை உருவாக்க உங்கள் இசையை ஸ்க்ரோப்ள் செய்கிறது.", + "default_plugin": "இயல்புநிலை", + "set_default": "இயல்புநிலையாக அமைக்கவும்", + "support": "ஆதரவு", + "support_plugin_development": "பிளகின் வளர்ச்சிக்கு ஆதரவு", + "can_access_name_api": "- **{name}** API ஐ அணுக முடியும்", + "do_you_want_to_install_this_plugin": "இந்த பிளகினை நீங்கள் நிறுவ விரும்புகிறீர்களா?", + "third_party_plugin_warning": "இந்த பிளகின் மூன்றாம் தரப்பு களஞ்சியத்திலிருந்து வருகிறது. நிறுவும் முன் மூலத்தை நீங்கள் நம்புகிறீர்கள் என்பதை உறுதிப்படுத்தவும்.", + "author": "ஆசிரியர்", + "this_plugin_can_do_following": "இந்த பிளகின் பின்வருவனவற்றைச் செய்ய முடியும்", + "install": "நிறுவவும்", + "install_a_metadata_provider": "மெட்டாடேட்டா வழங்குநரை நிறுவவும்", + "no_tracks_playing": "தற்போது எந்த பாடலும் இயங்கவில்லை", + "synced_lyrics_not_available": "இந்த பாடலுக்கு ஒத்திசைக்கப்பட்ட வரிகள் கிடைக்கவில்லை. தயவுசெய்து", + "plain_lyrics": "சாதாரண வரிகள்", + "tab_instead": "தாவலை அதற்கு பதிலாக பயன்படுத்தவும்.", + "disclaimer": "துறப்பு", + "third_party_plugin_dmca_notice": "ஸ்பாட்யூப் குழு எந்த \"மூன்றாம் தரப்பு\" பிளகின்களுக்கும் எந்தப் பொறுப்பையும் (சட்டரீதியான உட்பட) ஏற்காது.\nதயவுசெய்து உங்கள் சொந்த ஆபத்தில் அவற்றைப் பயன்படுத்தவும். ஏதேனும் பிழைகள்/சிக்கல்களுக்கு, பிளகின் களஞ்சியத்தில் அவற்றைப் புகாரளிக்கவும்.\n\nஏதேனும் ஒரு \"மூன்றாம் தரப்பு\" பிளகின் ஒரு சேவை/சட்ட நிறுவனத்தின் ToS/DMCA ஐ மீறினால், தயவுசெய்து \"மூன்றாம் தரப்பு\" பிளகின் ஆசிரியரையோ அல்லது ஹோஸ்டிங் தளத்தையோ, எ.கா. GitHub/Codeberg, நடவடிக்கை எடுக்கக் கோரவும். மேலே பட்டியலிடப்பட்ட (\"மூன்றாம் தரப்பு\" என பெயரிடப்பட்ட) அனைத்து பொதுவான/சமூகத்தால் பராமரிக்கப்படும் பிளகின்கள். நாங்கள் அவற்றை க்யூரேட் செய்யவில்லை, எனவே அவற்றின் மீது எந்த நடவடிக்கையும் எடுக்க முடியாது.\n\n", + "input_does_not_match_format": "உள்ளீடு தேவையான வடிவத்துடன் பொருந்தவில்லை", + "metadata_provider_plugins": "மெட்டாடேட்டா வழங்குநர் பிளகின்கள்", + "paste_plugin_download_url": "பதிவிறக்க url அல்லது GitHub/Codeberg repo url அல்லது .smplug கோப்பிற்கான நேரடி இணைப்பை ஒட்டவும்", + "download_and_install_plugin_from_url": "url இலிருந்து பிளகினைப் பதிவிறக்கி நிறுவவும்", + "failed_to_add_plugin_error": "பிளகினைச் சேர்க்கத் தவறிவிட்டது: {error}", + "upload_plugin_from_file": "கோப்பிலிருந்து பிளகினைப் பதிவேற்றவும்", + "installed": "நிறுவப்பட்டது", + "available_plugins": "கிடைக்கக்கூடிய பிளகின்கள்", + "configure_your_own_metadata_plugin": "உங்கள் சொந்த பிளேலிஸ்ட்/ஆல்பம்/கலைஞர்/ஊட்ட மெட்டாடேட்டா வழங்குநரை உள்ளமைக்கவும்", + "audio_scrobblers": "ஆடியோ ஸ்க்ரோப்ளர்கள்", + "scrobbling": "ஸ்க்ரோப்ளிங்", + "download_music_format": "இசை பதிவிறக்க வடிவம்", + "streaming_music_format": "இசை ஸ்ட்ரீமிங் வடிவம்", + "download_music_quality": "பதிவிறக்க தரம்", + "streaming_music_quality": "ஸ்ட்ரீமிங் தரம்", + "default_metadata_source": "இயல்புநிலை மெட்டாடேட்டா மூலம்", + "set_default_metadata_source": "இயல்புநிலை மெட்டாடேட்டா மூலத்தை அமை", + "default_audio_source": "இயல்புநிலை ஆடியோ மூலம்", + "set_default_audio_source": "இயல்புநிலை ஆடியோ மூலத்தை அமை", + "plugins": "செருகுநிரல்கள்", + "configure_plugins": "உங்கள் சொந்த மெட்டாடேட்டா வழங்குநர் மற்றும் ஆடியோ மூல செருகுநிரல்களை அமைக்கவும்", + "source": "மூலம்: ", + "uncompressed": "அழுத்தப்படாத", + "dab_music_source_description": "ஆடியோஃபைல்களுக்காக. உயர்தர/லாஸ்லெஸ் ஆடியோ ஸ்ட்ரீம்களை வழங்குகிறது. ISRC அடிப்படையில் துல்லியமான பாடல் பொருத்தம்." +} \ No newline at end of file diff --git a/lib/l10n/app_th.arb b/lib/l10n/app_th.arb index 1b72f1a3..4f2efc0e 100644 --- a/lib/l10n/app_th.arb +++ b/lib/l10n/app_th.arb @@ -402,5 +402,94 @@ "export_cache_files": "ส่งออกไฟล์แคช", "found_n_files": "พบ {count} ไฟล์", "export_cache_confirmation": "คุณต้องการส่งออกไฟล์เหล่านี้ไปยัง", - "exported_n_out_of_m_files": "ส่งออก {filesExported} จาก {files} ไฟล์" + "exported_n_out_of_m_files": "ส่งออก {filesExported} จาก {files} ไฟล์", + "playlist": "เพลย์ลิสต์", + "no_loop": "ไม่มีการวนซ้ำ", + "generate": "สร้าง", + "undo": "ย้อนกลับ", + "download_all": "ดาวน์โหลดทั้งหมด", + "add_all_to_playlist": "เพิ่มทั้งหมดในเพลย์ลิสต์", + "add_all_to_queue": "เพิ่มทั้งหมดในคิว", + "play_all_next": "เล่นทั้งหมดถัดไป", + "pause": "หยุดชั่วคราว", + "view_all": "ดูทั้งหมด", + "no_tracks_added_yet": "ดูเหมือนคุณยังไม่ได้เพิ่มเพลงใด ๆ", + "no_tracks": "ดูเหมือนจะไม่มีเพลงที่นี่", + "no_tracks_listened_yet": "ดูเหมือนคุณยังไม่ได้ฟังอะไรเลย", + "not_following_artists": "คุณไม่ได้ติดตามศิลปินใด ๆ", + "no_favorite_albums_yet": "ดูเหมือนคุณยังไม่ได้เพิ่มอัลบัมใด ๆ ในรายการโปรด", + "no_logs_found": "ไม่พบบันทึก", + "youtube_engine": "เครื่องมือ YouTube", + "youtube_engine_not_installed_title": "{engine} ยังไม่ได้ติดตั้ง", + "youtube_engine_not_installed_message": "{engine} ยังไม่ได้ติดตั้งในระบบของคุณ", + "youtube_engine_set_path": "ตรวจสอบให้แน่ใจว่ามันมีอยู่ในตัวแปร PATH หรือ\nตั้งค่าพาธที่แท้จริงของไฟล์ที่สามารถทำงานได้ {engine} ด้านล่าง", + "youtube_engine_unix_issue_message": "ใน macOS/Linux/Unix อย่าง OS การตั้งค่าพาธใน .zshrc/.bashrc/.bash_profile เป็นต้น จะไม่ทำงาน\nคุณต้องตั้งค่าพาธในไฟล์การกำหนดค่า shell", + "download": "ดาวน์โหลด", + "file_not_found": "ไม่พบไฟล์", + "custom": "กำหนดเอง", + "add_custom_url": "เพิ่ม URL แบบกำหนดเอง", + "edit_port": "แก้ไขพอร์ต", + "port_helper_msg": "ค่าเริ่มต้นคือ -1 ซึ่งหมายถึงหมายเลขสุ่ม หากคุณได้กำหนดค่าไฟร์วอลล์แล้ว แนะนำให้ตั้งค่านี้", + "connect_request": "อนุญาตให้ {client} เชื่อมต่อหรือไม่?", + "connection_request_denied": "การเชื่อมต่อล้มเหลว ผู้ใช้ปฏิเสธการเข้าถึง", + "hipotetical_calculation": "*การคำนวณนี้อิงจากค่าเฉลี่ยการจ่ายเงินต่อสตรีมของแพลตฟอร์มสตรีมมิ่งเพลงออนไลน์ที่ $0.003 ถึง $0.005 นี่เป็นการคำนวณสมมติฐานเพื่อให้ผู้ใช้เข้าใจว่าพวกเขาจะต้องจ่ายเงินให้ศิลปินเท่าไหร่หากพวกเขาฟังเพลงบนแพลตฟอร์มสตรีมมิ่งเพลงที่แตกต่างกัน", + "an_error_occurred": "เกิดข้อผิดพลาด", + "copy_to_clipboard": "คัดลอกไปยังคลิปบอร์ด", + "view_logs": "ดูบันทึก", + "retry": "ลองใหม่", + "no_default_metadata_provider_selected": "คุณไม่ได้ตั้งค่าผู้ให้บริการเมตาดาต้าเริ่มต้น", + "manage_metadata_providers": "จัดการผู้ให้บริการเมตาดาต้า", + "open_link_in_browser": "เปิดลิงก์ในเบราว์เซอร์หรือไม่?", + "do_you_want_to_open_the_following_link": "คุณต้องการเปิดลิงก์ต่อไปนี้หรือไม่", + "unsafe_url_warning": "การเปิดลิงก์จากแหล่งที่ไม่น่าเชื่อถืออาจไม่ปลอดภัย โปรดระมัดระวัง!\nคุณยังสามารถคัดลอกลิงก์ไปยังคลิปบอร์ดของคุณได้", + "copy_link": "คัดลอกลิงก์", + "building_your_timeline": "กำลังสร้างไทม์ไลน์ของคุณตามการฟังของคุณ...", + "official": "อย่างเป็นทางการ", + "author_name": "ผู้เขียน: {author}", + "third_party": "บุคคลที่สาม", + "plugin_requires_authentication": "ปลั๊กอินต้องมีการรับรองความถูกต้อง", + "update_available": "มีการอัปเดต", + "supports_scrobbling": "รองรับการ scrobbling", + "plugin_scrobbling_info": "ปลั๊กอินนี้จะ scrobble เพลงของคุณเพื่อสร้างประวัติการฟังของคุณ", + "default_plugin": "ค่าเริ่มต้น", + "set_default": "ตั้งค่าเริ่มต้น", + "support": "สนับสนุน", + "support_plugin_development": "สนับสนุนการพัฒนาปลั๊กอิน", + "can_access_name_api": "- สามารถเข้าถึง API **{name}**", + "do_you_want_to_install_this_plugin": "คุณต้องการติดตั้งปลั๊กอินนี้หรือไม่?", + "third_party_plugin_warning": "ปลั๊กอินนี้มาจากที่เก็บของบุคคลที่สาม โปรดตรวจสอบให้แน่ใจว่าคุณเชื่อถือแหล่งที่มาก่อนทำการติดตั้ง", + "author": "ผู้เขียน", + "this_plugin_can_do_following": "ปลั๊กอินนี้สามารถทำสิ่งต่อไปนี้", + "install": "ติดตั้ง", + "install_a_metadata_provider": "ติดตั้งผู้ให้บริการเมตาดาต้า", + "no_tracks_playing": "ขณะนี้ไม่มีเพลงที่กำลังเล่นอยู่", + "synced_lyrics_not_available": "ไม่มีเนื้อเพลงที่ซิงค์สำหรับเพลงนี้ กรุณาใช้แท็บ", + "plain_lyrics": "เนื้อเพลงธรรมดา", + "tab_instead": "แทน", + "disclaimer": "ข้อสงวนสิทธิ์", + "third_party_plugin_dmca_notice": "ทีม Spotube ไม่รับผิดชอบใดๆ (รวมถึงทางกฎหมาย) สำหรับปลั๊กอิน \"บุคคลที่สาม\" ใดๆ\nโปรดใช้งานด้วยความเสี่ยงของคุณเอง สำหรับข้อบกพร่อง/ปัญหาใดๆ โปรดรายงานไปยังที่เก็บปลั๊กอิน\n\nหากปลั๊กอิน \"บุคคลที่สาม\" ใดๆ ละเมิด ToS/DMCA ของบริการ/นิติบุคคลใดๆ โปรดขอให้ผู้เขียนปลั๊กอิน \"บุคคลที่สาม\" หรือแพลตฟอร์มโฮสติ้ง เช่น GitHub/Codeberg ดำเนินการ ที่ระบุไว้ข้างต้น (ที่ติดป้าย \"บุคคลที่สาม\") เป็นปลั๊กอินสาธารณะ/ที่ดูแลโดยชุมชนทั้งหมด เราไม่ได้จัดการดูแล ดังนั้นเราจึงไม่สามารถดำเนินการใดๆ กับพวกเขาได้\n\n", + "input_does_not_match_format": "อินพุตไม่ตรงกับรูปแบบที่ต้องการ", + "metadata_provider_plugins": "ปลั๊กอินผู้ให้บริการเมตาดาต้า", + "paste_plugin_download_url": "วาง url ดาวน์โหลดหรือ url ที่เก็บ GitHub/Codeberg หรือลิงก์โดยตรงไปยังไฟล์ .smplug", + "download_and_install_plugin_from_url": "ดาวน์โหลดและติดตั้งปลั๊กอินจาก url", + "failed_to_add_plugin_error": "ไม่สามารถเพิ่มปลั๊กอินได้: {error}", + "upload_plugin_from_file": "อัปโหลดปลั๊กอินจากไฟล์", + "installed": "ติดตั้งแล้ว", + "available_plugins": "ปลั๊กอินที่มีอยู่", + "configure_your_own_metadata_plugin": "กำหนดค่าผู้ให้บริการเมตาดาต้าเพลย์ลิสต์/อัลบั้ม/ศิลปิน/ฟีดของคุณเอง", + "audio_scrobblers": "เครื่อง scrobbler เสียง", + "scrobbling": "Scrobbling", + "download_music_format": "รูปแบบการดาวน์โหลดเพลง", + "streaming_music_format": "รูปแบบการสตรีมเพลง", + "download_music_quality": "คุณภาพการดาวน์โหลด", + "streaming_music_quality": "คุณภาพการสตรีม", + "default_metadata_source": "แหล่งเมตาดาต้าพื้นฐาน", + "set_default_metadata_source": "ตั้งค่าแหล่งเมตาดาต้าพื้นฐาน", + "default_audio_source": "แหล่งเสียงพื้นฐาน", + "set_default_audio_source": "ตั้งค่าแหล่งเสียงพื้นฐาน", + "plugins": "ปลั๊กอิน", + "configure_plugins": "กำหนดค่าปลั๊กอินผู้ให้บริการเมตาดาต้าและแหล่งเสียงของคุณเอง", + "source": "แหล่งที่มา: ", + "uncompressed": "ไม่บีบอัด", + "dab_music_source_description": "สำหรับคนรักเสียงเพลง ให้สตรีมเสียงคุณภาพสูง/ไร้การสูญเสียการบีบอัด การจับคู่แทร็กแม่นยำตาม ISRC" } \ No newline at end of file diff --git a/lib/l10n/app_tl.arb b/lib/l10n/app_tl.arb new file mode 100644 index 00000000..bf1f174c --- /dev/null +++ b/lib/l10n/app_tl.arb @@ -0,0 +1,492 @@ +{ + "guest": "Bisita", + "browse": "Mag-browse", + "search": "Maghanap", + "library": "Silid-aklatan", + "lyrics": "Mga Liriko", + "settings": "Mga Setting", + "genre_categories_filter": "I-filter ang mga kategorya o genre...", + "genre": "Genre", + "personalized": "Naka-personalize", + "featured": "Tampok", + "new_releases": "Mga Bagong Paglabas", + "songs": "Mga Kanta", + "playing_track": "Tumutugtog ang {track}", + "queue_clear_alert": "Ito ay magbubura ng kasalukuyang pila. {track_length} na mga track ang tatanggalin\nGusto mo bang magpatuloy?", + "load_more": "Mag-load pa", + "playlists": "Mga Playlist", + "artists": "Mga Artista", + "albums": "Mga Album", + "tracks": "Mga Track", + "downloads": "Mga Download", + "filter_playlists": "I-filter ang iyong mga playlist...", + "liked_tracks": "Mga Nagustuhang Track", + "liked_tracks_description": "Lahat ng mga track na iyong nagustuhan", + "playlist": "Playlist", + "create_a_playlist": "Gumawa ng playlist", + "update_playlist": "I-update ang playlist", + "create": "Lumikha", + "cancel": "Ikansela", + "update": "I-update", + "playlist_name": "Pangalan ng Playlist", + "name_of_playlist": "Pangalan ng playlist", + "description": "Paglalarawan", + "public": "Pampubliko", + "collaborative": "Pakikipagtulungan", + "search_local_tracks": "Maghanap ng mga lokal na track...", + "play": "I-play", + "delete": "Burahin", + "none": "Wala", + "sort_a_z": "Ayusin ayon sa A-Z", + "sort_z_a": "Ayusin ayon sa Z-A", + "sort_artist": "Ayusin ayon sa Artista", + "sort_album": "Ayusin ayon sa Album", + "sort_duration": "Ayusin ayon sa Tagal", + "sort_tracks": "Ayusin ang mga Track", + "currently_downloading": "Kasalukuyang Nagda-download ({tracks_length})", + "cancel_all": "Kanselahin Lahat", + "filter_artist": "I-filter ang mga artista...", + "followers": "{followers} na mga Tagasunod", + "add_artist_to_blacklist": "Idagdag ang artista sa blacklist", + "top_tracks": "Mga Nangungunang Track", + "fans_also_like": "Gusto rin ng mga tagahanga", + "loading": "Naglo-load...", + "artist": "Artista", + "blacklisted": "Naka-blacklist", + "following": "Sinusundan", + "follow": "Sundan", + "artist_url_copied": "Na-copy sa clipboard ang URL ng artista", + "added_to_queue": "Idinagdag ang {tracks} na mga track sa pila", + "filter_albums": "I-filter ang mga album...", + "synced": "Naka-sync", + "plain": "Simpleng", + "shuffle": "I-shuffle", + "search_tracks": "Maghanap ng mga track...", + "released": "Inilabas", + "error": "Error {error}", + "title": "Pamagat", + "time": "Oras", + "more_actions": "Higit pang mga aksyon", + "download_count": "I-download ({count})", + "add_count_to_playlist": "Idagdag ({count}) sa Playlist", + "add_count_to_queue": "Idagdag ({count}) sa Pila", + "play_count_next": "I-play ({count}) kasunod", + "album": "Album", + "copied_to_clipboard": "Na-copy ang {data} sa clipboard", + "add_to_following_playlists": "Idagdag ang {track} sa mga sumusunod na Playlist", + "add": "Idagdag", + "added_track_to_queue": "Idinagdag ang {track} sa pila", + "add_to_queue": "Idagdag sa pila", + "track_will_play_next": "Ang {track} ay tutugtog susunod", + "play_next": "I-play susunod", + "removed_track_from_queue": "Tinanggal ang {track} mula sa pila", + "remove_from_queue": "Alisin mula sa pila", + "remove_from_favorites": "Alisin mula sa mga paborito", + "save_as_favorite": "I-save bilang paborito", + "add_to_playlist": "Idagdag sa playlist", + "remove_from_playlist": "Alisin mula sa playlist", + "add_to_blacklist": "Idagdag sa blacklist", + "remove_from_blacklist": "Alisin mula sa blacklist", + "share": "Ibahagi", + "mini_player": "Mini Player", + "slide_to_seek": "I-slide para mag-seek pasulong o pabalik", + "shuffle_playlist": "I-shuffle ang playlist", + "unshuffle_playlist": "I-unshuffle ang playlist", + "previous_track": "Nakaraang track", + "next_track": "Susunod na track", + "pause_playback": "I-pause ang Playback", + "resume_playback": "Ipagpatuloy ang Playback", + "loop_track": "I-loop ang track", + "no_loop": "Walang loop", + "repeat_playlist": "Ulitin ang playlist", + "queue": "Pila", + "alternative_track_sources": "Alternatibong mga pinagmulan ng track", + "download_track": "I-download ang track", + "tracks_in_queue": "{tracks} na mga track sa pila", + "clear_all": "Burahin lahat", + "show_hide_ui_on_hover": "Ipakita/Itago ang UI sa hover", + "always_on_top": "Palaging nasa ibabaw", + "exit_mini_player": "Lumabas sa Mini player", + "download_location": "Lokasyon ng pag-download", + "local_library": "Lokal na silid-aklatan", + "add_library_location": "Idagdag sa silid-aklatan", + "remove_library_location": "Alisin mula sa silid-aklatan", + "account": "Account", + "login_with_spotify": "Mag-login gamit ang iyong Spotify account", + "connect_with_spotify": "Kumonekta sa Spotify", + "logout": "Mag-logout", + "logout_of_this_account": "Mag-logout sa account na ito", + "language_region": "Wika at Rehiyon", + "language": "Wika", + "system_default": "Default ng Sistema", + "market_place_region": "Rehiyon ng Marketplace", + "recommendation_country": "Bansang Inirerekomenda", + "appearance": "Hitsura", + "layout_mode": "Mode ng Layout", + "override_layout_settings": "I-override ang mga setting ng responsive layout mode", + "adaptive": "Umaangkop", + "compact": "Kompakto", + "extended": "Pinalawig", + "theme": "Tema", + "dark": "Madilim", + "light": "Maliwanag", + "system": "Sistema", + "accent_color": "Kulay ng Accent", + "sync_album_color": "I-sync ang kulay ng album", + "sync_album_color_description": "Ginagamit ang pangunahing kulay ng album art bilang kulay ng accent", + "playback": "Playback", + "audio_quality": "Kalidad ng Audio", + "high": "Mataas", + "low": "Mababa", + "pre_download_play": "Mag-pre-download at i-play", + "pre_download_play_description": "Sa halip na mag-stream ng audio, mag-download ng bytes at i-play sa halip (Inirerekomenda para sa mga gumagamit ng mataas na bandwidth)", + "skip_non_music": "Laktawan ang mga segment na hindi musika (SponsorBlock)", + "blacklist_description": "Mga track at artista na nasa blacklist", + "wait_for_download_to_finish": "Mangyaring maghintay para matapos ang kasalukuyang pag-download", + "desktop": "Desktop", + "close_behavior": "Pag-uugali ng Pagsara", + "close": "Isara", + "minimize_to_tray": "I-minimize sa tray", + "show_tray_icon": "Ipakita ang icon ng System tray", + "about": "Tungkol sa", + "u_love_spotube": "Alam naming gusto mo ang Spotube", + "check_for_updates": "Maghanap ng mga update", + "about_spotube": "Tungkol sa Spotube", + "blacklist": "Blacklist", + "please_sponsor": "Mangyaring Mag-sponsor/Mag-donate", + "spotube_description": "Spotube, isang magaan, cross-platform, libreng-para-sa-lahat na spotify client", + "version": "Bersyon", + "build_number": "Build Number", + "founder": "Nagtatag", + "repository": "Repository", + "bug_issues": "Bug+Mga Isyu", + "made_with": "Ginawa nang may ❤️ sa Bangladesh🇧🇩", + "kingkor_roy_tirtho": "Kingkor Roy Tirtho", + "copyright": "© 2021-{current_year} Kingkor Roy Tirtho", + "license": "Lisensya", + "add_spotify_credentials": "Idagdag ang iyong mga kredensyal sa spotify para makapagsimula", + "credentials_will_not_be_shared_disclaimer": "Huwag mag-alala, ang alinman sa iyong mga kredensyal ay hindi kokolektahin o ibabahagi sa sinuman", + "know_how_to_login": "Hindi mo alam kung paano gawin ito?", + "follow_step_by_step_guide": "Sundin ang Hakbang-hakbang na gabay", + "spotify_cookie": "Spotify {name} Cookie", + "cookie_name_cookie": "{name} Cookie", + "fill_in_all_fields": "Mangyaring punan ang lahat ng field", + "submit": "Isumite", + "exit": "Lumabas", + "previous": "Nakaraan", + "next": "Susunod", + "done": "Tapos na", + "step_1": "Hakbang 1", + "first_go_to": "Una, Pumunta sa", + "login_if_not_logged_in": "at Mag-login/Mag-signup kung hindi ka naka-log in", + "step_2": "Hakbang 2", + "step_2_steps": "1. Kapag naka-log in ka na, pindutin ang F12 o i-right click ang Mouse > Inspect para Buksan ang Browser devtools.\n2. Pagkatapos ay pumunta sa \"Application\" Tab (Chrome, Edge, Brave atbp..) o \"Storage\" Tab (Firefox, Palemoon atbp..)\n3. Pumunta sa \"Cookies\" na seksyon at pagkatapos sa \"https://accounts.spotify.com\" na subseksyon", + "step_3": "Hakbang 3", + "step_3_steps": "Kopyahin ang halaga ng \"sp_dc\" Cookie", + "success_emoji": "Tagumpay🥳", + "success_message": "Ngayon ay matagumpay kang Naka-log in gamit ang iyong Spotify account. Magaling, kaibigan!", + "step_4": "Hakbang 4", + "step_4_steps": "I-paste ang na-kopyang halaga ng \"sp_dc\"", + "something_went_wrong": "May nangyaring mali", + "piped_instance": "Instance ng Piped Server", + "piped_description": "Ang instance ng Piped server na gagamitin para sa pagtutugma ng track", + "piped_warning": "Maaaring hindi gumagana nang mabuti ang ilan sa mga ito. Kaya gamitin sa sarili mong peligro", + "invidious_instance": "Instance ng Invidious Server", + "invidious_description": "Ang instance ng Invidious server na gagamitin para sa pagtutugma ng track", + "invidious_warning": "Maaaring hindi gumagana nang mabuti ang ilan sa mga ito. Kaya gamitin sa sarili mong peligro", + "generate": "Gumawa", + "track_exists": "Ang Track na {track} ay umiiral na", + "replace_downloaded_tracks": "Palitan ang lahat ng na-download na mga track", + "skip_download_tracks": "Laktawan ang pag-download ng lahat ng na-download na mga track", + "do_you_want_to_replace": "Gusto mo bang palitan ang umiiral na track??", + "replace": "Palitan", + "skip": "Laktawan", + "select_up_to_count_type": "Pumili ng hanggang {count} {type}", + "select_genres": "Pumili ng mga Genre", + "add_genres": "Magdagdag ng mga Genre", + "country": "Bansa", + "number_of_tracks_generate": "Bilang ng mga track na gagawin", + "acousticness": "Acoustic-ness", + "danceability": "Kakayahang Sayawin", + "energy": "Enerhiya", + "instrumentalness": "Instrumental-ness", + "liveness": "Liveness", + "loudness": "Lakas", + "speechiness": "Pagsasalita", + "valence": "Valence", + "popularity": "Popularidad", + "key": "Key", + "duration": "Tagal (s)", + "tempo": "Tempo (BPM)", + "mode": "Mode", + "time_signature": "Time Signature", + "short": "Maikli", + "medium": "Katamtaman", + "long": "Mahaba", + "min": "Min", + "max": "Max", + "target": "Target", + "moderate": "Katamtaman", + "deselect_all": "Alisin ang Pagkakapili sa Lahat", + "select_all": "Piliin Lahat", + "are_you_sure": "Sigurado ka ba?", + "generating_playlist": "Gumagawa ng iyong custom na playlist...", + "selected_count_tracks": "Napili ang {count} na mga track", + "download_warning": "Kung nag-download ka ng lahat ng Track sa maramihan, malinaw na nagpa-pirate ka ng Musika at nagsasanhi ng pinsala sa creative society ng Musika. Sana ay alam mo ito. Palaging, subukang igalang at suportahan ang masipag na paggawa ng Artist", + "download_ip_ban_warning": "Sa nga pala, ang iyong IP ay maaaring ma-block sa YouTube dahil sa sobrang mga kahilingan sa pag-download kaysa sa karaniwan. Ang IP block ay nangangahulugang hindi mo magagamit ang YouTube (kahit na naka-log in ka) sa loob ng hindi bababa sa 2-3 buwan mula sa device na may IP na iyon. At hindi pinanghahawakan ng Spotube ang anumang responsibilidad kung mangyayari ito", + "by_clicking_accept_terms": "Sa pamamagitan ng pag-click sa 'tanggapin', sumasang-ayon ka sa mga sumusunod na tuntunin:", + "download_agreement_1": "Alam kong nagpa-pirate ako ng Musika. Masama ako", + "download_agreement_2": "Susuportahan ko ang Artist saan man ako maaari at ginagawa ko lang ito dahil wala akong pera para bumili ng kanilang sining", + "download_agreement_3": "Lubos kong nauunawaan na ang aking IP ay maaaring ma-block sa YouTube at hindi ko pinanghahawakan ang Spotube o ang kanyang mga may-ari/nag-ambag na responsable para sa anumang aksidente na sanhi ng aking kasalukuyang aksyon", + "decline": "Tanggihan", + "accept": "Tanggapin", + "details": "Mga Detalye", + "youtube": "YouTube", + "channel": "Channel", + "likes": "Mga Like", + "dislikes": "Mga Dislike", + "views": "Mga View", + "streamUrl": "Stream URL", + "stop": "Ihinto", + "sort_newest": "Ayusin ayon sa pinakabagong idinagdag", + "sort_oldest": "Ayusin ayon sa pinakalumang idinagdag", + "sleep_timer": "Sleep Timer", + "mins": "{minutes} Minuto", + "hours": "{hours} Oras", + "hour": "{hours} Oras", + "custom_hours": "Custom na Oras", + "logs": "Mga Log", + "developers": "Mga Developer", + "not_logged_in": "Hindi ka naka-log in", + "search_mode": "Mode ng Paghahanap", + "audio_source": "Pinagmulan ng Audio", + "ok": "Ok", + "failed_to_encrypt": "Nabigong i-encrypt", + "encryption_failed_warning": "Gumagamit ng encryption ang Spotube para ligtas na i-store ang iyong data. Ngunit nabigo. Kaya babalik ito sa hindi secure na storage\nKung gumagamit ka ng linux, mangyaring tiyakin na mayroon kang anumang secret-service na naka-install (gnome-keyring, kde-wallet, keepassxc atbp)", + "querying_info": "Kinukuha ang impormasyon...", + "piped_api_down": "Ang Piped API ay hindi gumagana", + "piped_down_error_instructions": "Ang instance ng Piped na {pipedInstance} ay kasalukuyang hindi gumagana\n\nMaaari mong baguhin ang instance o baguhin ang 'Uri ng API' sa opisyal na YouTube API\n\nSiguraduhing i-restart ang app pagkatapos ng pagbabago", + "you_are_offline": "Kasalukuyan kang offline", + "connection_restored": "Naibalik na ang iyong koneksyon sa internet", + "use_system_title_bar": "Gamitin ang title bar ng system", + "crunching_results": "Pinaproseso ang mga resulta...", + "search_to_get_results": "Maghanap para makakuha ng mga resulta", + "use_amoled_mode": "Matingkad na itim na madilim na tema", + "pitch_dark_theme": "AMOLED Mode", + "normalize_audio": "I-normalize ang audio", + "change_cover": "Baguhin ang cover", + "add_cover": "Magdagdag ng cover", + "restore_defaults": "Ibalik ang mga default", + "download_music_codec": "Codec para sa pag-download ng musika", + "streaming_music_codec": "Codec para sa pag-stream ng musika", + "login_with_lastfm": "Mag-login gamit ang Last.fm", + "connect": "Kumonekta", + "disconnect_lastfm": "Idiskonekta ang Last.fm", + "disconnect": "Idiskonekta", + "username": "Username", + "password": "Password", + "login": "Mag-login", + "login_with_your_lastfm": "Mag-login gamit ang iyong Last.fm account", + "scrobble_to_lastfm": "I-scrobble sa Last.fm", + "go_to_album": "Pumunta sa Album", + "discord_rich_presence": "Discord Rich Presence", + "browse_all": "I-browse Lahat", + "genres": "Mga Genre", + "explore_genres": "Tuklasin ang mga Genre", + "friends": "Mga Kaibigan", + "no_lyrics_available": "Paumanhin, hindi mahanap ang lyrics para sa track na ito", + "start_a_radio": "Magsimula ng Radio", + "how_to_start_radio": "Paano mo gustong simulan ang radio?", + "replace_queue_question": "Gusto mo bang palitan ang kasalukuyang pila o idagdag dito?", + "endless_playback": "Walang Hanggang Playback", + "delete_playlist": "Burahin ang Playlist", + "delete_playlist_confirmation": "Sigurado ka bang gusto mong burahin ang playlist na ito?", + "local_tracks": "Mga Lokal na Track", + "local_tab": "Lokal", + "song_link": "Link ng Kanta", + "skip_this_nonsense": "Laktawan ang kalokohan na ito", + "freedom_of_music": "\"Kalayaan ng Musika\"", + "freedom_of_music_palm": "\"Kalayaan ng Musika sa iyong palad\"", + "get_started": "Magsimula na tayo", + "youtube_source_description": "Inirerekomenda at pinakamahusay na gumagana.", + "piped_source_description": "Gusto ng kalayaan? Kapareho ng YouTube ngunit mas malaya.", + "jiosaavn_source_description": "Pinakamahusay para sa rehiyon ng South Asia.", + "invidious_source_description": "Katulad ng Piped ngunit may mas mataas na availability.", + "highest_quality": "Pinakamataas na Kalidad: {quality}", + "select_audio_source": "Pumili ng Pinagmulan ng Audio", + "endless_playback_description": "Awtomatikong magdagdag ng mga bagong kanta\nsa dulo ng pila", + "choose_your_region": "Piliin ang iyong rehiyon", + "choose_your_region_description": "Ito ay tutulong sa Spotube na ipakita sa iyo ang tamang content\npara sa iyong lokasyon.", + "choose_your_language": "Piliin ang iyong wika", + "help_project_grow": "Tulungan ang proyektong ito na lumago", + "help_project_grow_description": "Ang Spotube ay isang open-source na proyekto. Maaari mong tulungan ang proyektong ito na lumago sa pamamagitan ng pag-contribute sa proyekto, pag-ulat ng mga bug, o pagmungkahi ng mga bagong feature.", + "contribute_on_github": "Mag-contribute sa GitHub", + "donate_on_open_collective": "Mag-donate sa Open Collective", + "browse_anonymously": "Mag-browse nang Anonymous", + "enable_connect": "I-enable ang Connect", + "enable_connect_description": "Kontrolin ang Spotube mula sa ibang mga device", + "devices": "Mga Device", + "select": "Pumili", + "connect_client_alert": "Ikaw ay kontrolado ng {client}", + "this_device": "Ang Device na ito", + "remote": "Remote", + "stats": "Mga Stat", + "and_n_more": "at {count} pa", + "recently_played": "Kamakailan Lang na Ni-play", + "browse_more": "Mag-browse pa", + "no_title": "Walang Pamagat", + "not_playing": "Hindi tumutugtog", + "epic_failure": "Epic na pagkabigo!", + "added_num_tracks_to_queue": "Nagdagdag ng {tracks_length} na mga track sa pila", + "spotube_has_an_update": "Ang Spotube ay may update", + "download_now": "I-download Ngayon", + "nightly_version": "Ang Spotube Nightly {nightlyBuildNum} ay inilabas na", + "release_version": "Ang Spotube v{version} ay inilabas na", + "read_the_latest": "Basahin ang pinakabagong ", + "release_notes": "release notes", + "pick_color_scheme": "Pumili ng color scheme", + "save": "I-save", + "choose_the_device": "Piliin ang device:", + "multiple_device_connected": "Mayroong maraming device na nakakonekta.\nPiliin ang device kung saan mo gustong maganap ang aksyon na ito", + "nothing_found": "Walang nahanap", + "the_box_is_empty": "Ang kahon ay walang laman", + "top_artists": "Nangungunang mga Artista", + "top_albums": "Nangungunang mga Album", + "this_week": "Ngayong linggo", + "this_month": "Ngayong buwan", + "last_6_months": "Nakaraang 6 na buwan", + "this_year": "Ngayong taon", + "last_2_years": "Nakaraang 2 taon", + "all_time": "Lahat ng panahon", + "powered_by_provider": "Pinapagana ng {providerName}", + "email": "Email", + "profile_followers": "Mga Tagasunod", + "birthday": "Kaarawan", + "subscription": "Subscription", + "not_born": "Hindi pa ipinanganak", + "hacker": "Hacker", + "profile": "Profile", + "no_name": "Walang Pangalan", + "edit": "I-edit", + "user_profile": "Profile ng User", + "count_plays": "{count} na mga play", + "streaming_fees_hypothetical": "Mga bayarin sa streaming (hypothetical)", + "minutes_listened": "Mga minutong pinapakinggan", + "streamed_songs": "Mga na-stream na kanta", + "count_streams": "{count} na mga stream", + "owned_by_you": "Pag-aari mo", + "copied_shareurl_to_clipboard": "Na-kopya ang {shareUrl} sa clipboard", + "spotify_hipotetical_calculation": "*Ito ay kinalkula batay sa bawat stream\nna bayad ng Spotify na $0.003 hanggang $0.005. Ito ay isang hypothetical\nna pagkalkula para bigyan ang user ng ideya kung magkano\nang kanilang ibabayad sa mga artista kung sila ay nakikinig\nng kanilang kanta sa Spotify.", + "count_mins": "{minutes} minuto", + "summary_minutes": "minuto", + "summary_listened_to_music": "Nakinig sa musika", + "summary_songs": "mga kanta", + "summary_streamed_overall": "Na-stream sa kabuuan", + "summary_owed_to_artists": "Utang sa mga artista\nngayong buwan", + "summary_artists": "artista", + "summary_music_reached_you": "Umabot sa iyo ang musika", + "summary_full_albums": "buong album", + "summary_got_your_love": "Nakuha ang iyong pagmamahal", + "summary_playlists": "mga playlist", + "summary_were_on_repeat": "Pinu-playlst muli", + "total_money": "Kabuuang {money}", + "webview_not_found": "Hindi nahanap ang Webview", + "webview_not_found_description": "Walang webview runtime na naka-install sa iyong device.\nKung naka-install ito, siguraduhing nasa Environment PATH\n\nPagkatapos mag-install, i-restart ang app", + "unsupported_platform": "Hindi suportadong platform", + "cache_music": "I-cache ang musika", + "open": "Buksan", + "cache_folder": "Folder ng cache", + "export": "I-export", + "clear_cache": "Burahin ang cache", + "clear_cache_confirmation": "Gusto mo bang burahin ang cache?", + "export_cache_files": "I-export ang mga Naka-cache na File", + "found_n_files": "Nahanap ang {count} na mga file", + "export_cache_confirmation": "Gusto mo bang i-export ang mga file na ito sa", + "exported_n_out_of_m_files": "Na-export ang {filesExported} mula sa {files} na mga file", + "undo": "I-undo", + "download_all": "I-download lahat", + "add_all_to_playlist": "Idagdag lahat sa playlist", + "add_all_to_queue": "Idagdag lahat sa pila", + "play_all_next": "I-play lahat susunod", + "pause": "Pause", + "view_all": "Tingnan lahat", + "no_tracks_added_yet": "Mukhang wala ka pang idinaragdag na mga track", + "no_tracks": "Mukhang walang mga track dito", + "no_tracks_listened_yet": "Mukhang wala ka pang pinakikinggan", + "not_following_artists": "Hindi ka sumusunod sa anumang mga artista", + "no_favorite_albums_yet": "Mukhang wala ka pang idinagdag na anumang mga album sa iyong mga paborito", + "no_logs_found": "Walang nahanap na mga log", + "youtube_engine": "YouTube Engine", + "youtube_engine_not_installed_title": "Hindi naka-install ang {engine}", + "youtube_engine_not_installed_message": "Hindi naka-install ang {engine} sa iyong sistema.", + "youtube_engine_set_path": "Siguraduhing available ito sa PATH variable o\ni-set ang absolute path sa {engine} executable sa ibaba", + "youtube_engine_unix_issue_message": "Sa macOS/Linux/unix tulad ng OS, ang pag-set ng path sa .zshrc/.bashrc/.bash_profile atbp. ay hindi gagana.\nKailangan mong i-set ang path sa configuration file ng shell", + "download": "I-download", + "file_not_found": "Hindi nahanap ang file", + "custom": "Custom", + "add_custom_url": "Magdagdag ng custom URL", + "edit_port": "I-edit ang port", + "port_helper_msg": "Ang default ay -1 na nagpapahiwatig ng random na numero. Kung na-configure mo ang firewall, inirerekomenda na itakda ito.", + "connect_request": "Payagan ang {client} na kumonekta?", + "connection_request_denied": "Tanggihan ang koneksyon. Tinanggihan ng gumagamit ang pag-access.", + "hipotetical_calculation": "*Ito ay kinakalkula batay sa average na payout ng online music streaming platform na $0.003 hanggang $0.005 kada stream. Ito ay isang hypothetical na kalkulasyon upang bigyan ang user ng insight kung magkano ang babayaran nila sa mga artist kung sakaling makinig sila ng kanilang kanta sa iba't ibang music streaming platform.", + "an_error_occurred": "May naganap na error", + "copy_to_clipboard": "Kopyahin sa clipboard", + "view_logs": "Tingnan ang mga log", + "retry": "Subukang muli", + "no_default_metadata_provider_selected": "Wala kang nakatakdang default na metadata provider", + "manage_metadata_providers": "Pamahalaan ang mga metadata provider", + "open_link_in_browser": "Buksan ang Link sa Browser?", + "do_you_want_to_open_the_following_link": "Gusto mo bang buksan ang sumusunod na link", + "unsafe_url_warning": "Maaaring hindi ligtas ang pagbukas ng mga link mula sa hindi pinagkakatiwalaang pinagmulan. Mag-ingat!\nMaaari mo ring kopyahin ang link sa iyong clipboard.", + "copy_link": "Kopyahin ang Link", + "building_your_timeline": "Binubuo ang iyong timeline batay sa iyong mga pinakinggan...", + "official": "Opisyal", + "author_name": "May-akda: {author}", + "third_party": "Third-party", + "plugin_requires_authentication": "Nangangailangan ng authentication ang plugin", + "update_available": "May available na update", + "supports_scrobbling": "Sinusuportahan ang scrobbling", + "plugin_scrobbling_info": "Sinis-scrobble ng plugin na ito ang iyong musika upang mabuo ang iyong kasaysayan ng pakikinig.", + "default_plugin": "Default", + "set_default": "Itakda bilang default", + "support": "Suporta", + "support_plugin_development": "Suportahan ang pagbuo ng plugin", + "can_access_name_api": "- Maaaring i-access ang **{name}** API", + "do_you_want_to_install_this_plugin": "Gusto mo bang i-install ang plugin na ito?", + "third_party_plugin_warning": "Ang plugin na ito ay mula sa third-party na repository. Mangyaring tiyakin na pinagkakatiwalaan mo ang pinagmulan bago mag-install.", + "author": "May-akda", + "this_plugin_can_do_following": "Maaaring gawin ng plugin na ito ang sumusunod", + "install": "I-install", + "install_a_metadata_provider": "Mag-install ng Metadata Provider", + "no_tracks_playing": "Walang Track na kasalukuyang tumutugtog", + "synced_lyrics_not_available": "Hindi available ang mga naka-sync na lyrics para sa kantang ito. Mangyaring gamitin ang", + "plain_lyrics": "Simpleng Lyrics", + "tab_instead": "na tab sa halip.", + "disclaimer": "Disclaimer", + "third_party_plugin_dmca_notice": "Ang Spotube team ay walang hawak na anumang responsibilidad (kabilang ang legal) para sa anumang \"Third-party\" plugins.\nMangyaring gamitin ang mga ito sa iyong sariling peligro. Para sa anumang mga bug/isyu, mangyaring iulat ang mga ito sa repository ng plugin.\n\nKung ang anumang \"Third-party\" plugin ay lumalabag sa ToS/DMCA ng anumang serbisyo/legal na entity, mangyaring hilingin sa \"Third-party\" plugin author o sa hosting platform e.g. GitHub/Codeberg na gumawa ng aksyon. Ang nakalista sa itaas (\"Third-party\" na may label) ay lahat ng pampubliko/komunidad na pinananatiling mga plugin. Hindi namin sila kinukurusado, kaya hindi kami makakagawa ng anumang aksyon sa kanila.\n\n", + "input_does_not_match_format": "Ang input ay hindi tumutugma sa kinakailangang format", + "metadata_provider_plugins": "Mga Plugin ng Metadata Provider", + "paste_plugin_download_url": "I-paste ang download url o GitHub/Codeberg repo url o direktang link sa .smplug file", + "download_and_install_plugin_from_url": "I-download at i-install ang plugin mula sa url", + "failed_to_add_plugin_error": "Nabigo ang pagdagdag ng plugin: {error}", + "upload_plugin_from_file": "I-upload ang plugin mula sa file", + "installed": "Naka-install", + "available_plugins": "Mga available na plugin", + "configure_your_own_metadata_plugin": "I-configure ang iyong sariling playlist/album/artist/feed metadata provider", + "audio_scrobblers": "Mga Audio Scrobbler", + "scrobbling": "Scrobbling", + "download_music_format": "I-download na format ng musika", + "streaming_music_format": "Format ng streaming ng musika", + "download_music_quality": "Kalidad ng i-download na musika", + "streaming_music_quality": "Kalidad ng streaming ng musika", + "default_metadata_source": "Default na pinagmulan ng metadata", + "set_default_metadata_source": "Itakda ang default na pinagmulan ng metadata", + "default_audio_source": "Default na pinagmulan ng audio", + "set_default_audio_source": "Itakda ang default na pinagmulan ng audio", + "plugins": "Mga plugin", + "configure_plugins": "I-configure ang sarili mong metadata provider at mga audio source plugin", + "source": "Pinagmulan: ", + "uncompressed": "Hindi naka-compress", + "dab_music_source_description": "Para sa mga audiophile. Nagbibigay ng de-kalidad/walang loss na audio streams. Tumpak na pagtutugma ng track batay sa ISRC." +} \ No newline at end of file diff --git a/lib/l10n/app_tr.arb b/lib/l10n/app_tr.arb index 7f2bf5fb..72734d3b 100644 --- a/lib/l10n/app_tr.arb +++ b/lib/l10n/app_tr.arb @@ -401,5 +401,94 @@ "export_cache_files": "Önbelleğe Alınmış Dosyaları Dışa Aktar", "found_n_files": "{count} dosya bulundu", "export_cache_confirmation": "Bu dosyaları dışa aktarmak istiyor musunuz", - "exported_n_out_of_m_files": "{filesExported} / {files} dosya dışa aktarıldı" + "exported_n_out_of_m_files": "{filesExported} / {files} dosya dışa aktarıldı", + "playlist": "Çalma Listesi", + "no_loop": "Dönüş Yok", + "generate": "Oluştur", + "undo": "Geri Al", + "download_all": "Tümünü İndir", + "add_all_to_playlist": "Hepsini çalma listesine ekle", + "add_all_to_queue": "Hepsini kuyruğa ekle", + "play_all_next": "Hepsini bir sonraki çal", + "pause": "Duraklat", + "view_all": "Tümünü Gör", + "no_tracks_added_yet": "Henüz hiçbir şarkı eklemediniz gibi görünüyor", + "no_tracks": "Burada hiç şarkı yok gibi görünüyor", + "no_tracks_listened_yet": "Henüz hiçbir şey dinlemediniz gibi görünüyor", + "not_following_artists": "Hiçbir sanatçıyı takip etmiyorsunuz", + "no_favorite_albums_yet": "Henüz favorilerinize herhangi bir albüm eklemediniz gibi görünüyor", + "no_logs_found": "Log bulunamadı", + "youtube_engine": "YouTube Motoru", + "youtube_engine_not_installed_title": "{engine} Yüklü değil", + "youtube_engine_not_installed_message": "{engine} sisteminizde yüklü değil.", + "youtube_engine_set_path": "PATH değişkeninde kullanılabilir olduğundan emin olun veya\n{engine} çalıştırılabilir dosyasının mutlak yolunu aşağıda ayarlayın", + "youtube_engine_unix_issue_message": "macOS/Linux/Unix benzeri işletim sistemlerinde, .zshrc/.bashrc/.bash_profile gibi dosyalarda yol ayarlamak işe yaramaz.\nYolunuzu kabuk yapılandırma dosyasına ayarlamanız gerekir", + "download": "İndir", + "file_not_found": "Dosya bulunamadı", + "custom": "Özel", + "add_custom_url": "Özel URL ekle", + "edit_port": "Portu düzenle", + "port_helper_msg": "Varsayılan -1'dir, bu da rastgele bir sayıyı gösterir. Bir güvenlik duvarınız varsa, bunu ayarlamanız önerilir.", + "connect_request": "{client} bağlantısına izin verilsin mi?", + "connection_request_denied": "Bağlantı reddedildi. Kullanıcı erişimi reddetti.", + "hipotetical_calculation": "*Bu, çevrimiçi müzik akışı platformlarının ortalama akış başına $0,003 ile $0,005 arasındaki ödemesine göre hesaplanmıştır. Bu, kullanıcının farklı müzik akışı platformlarında şarkılarını dinleselerdi sanatçılara ne kadar ödeme yapacaklarına dair fikir vermek için yapılan varsayımsal bir hesaplamadır.", + "an_error_occurred": "Bir hata oluştu", + "copy_to_clipboard": "Panoya kopyala", + "view_logs": "Günlükleri görüntüle", + "retry": "Tekrar dene", + "no_default_metadata_provider_selected": "Varsayılan bir meta veri sağlayıcısı ayarlanmadı", + "manage_metadata_providers": "Meta veri sağlayıcılarını yönet", + "open_link_in_browser": "Bağlantıyı Tarayıcıda Aç?", + "do_you_want_to_open_the_following_link": "Aşağıdaki bağlantıyı açmak istiyor musunuz", + "unsafe_url_warning": "Güvenilmeyen kaynaklardan bağlantı açmak güvensiz olabilir. Dikkatli olun!\nBağlantıyı panonuza da kopyalayabilirsiniz.", + "copy_link": "Bağlantıyı Kopyala", + "building_your_timeline": "Dinlemelerinize göre zaman çizelgeniz oluşturuluyor...", + "official": "Resmi", + "author_name": "Yazar: {author}", + "third_party": "Üçüncü taraf", + "plugin_requires_authentication": "Eklenti kimlik doğrulama gerektirir", + "update_available": "Güncelleme mevcut", + "supports_scrobbling": "Scrobbling'i destekler", + "plugin_scrobbling_info": "Bu eklenti, dinleme geçmişinizi oluşturmak için müziğinizi scrobble eder.", + "default_plugin": "Varsayılan", + "set_default": "Varsayılan olarak ayarla", + "support": "Destek", + "support_plugin_development": "Eklenti geliştirmeyi destekle", + "can_access_name_api": "- **{name}** API'ye erişebilir", + "do_you_want_to_install_this_plugin": "Bu eklentiyi yüklemek istiyor musunuz?", + "third_party_plugin_warning": "Bu eklenti üçüncü taraf bir depodan gelmektedir. Lütfen yüklemeden önce kaynağa güvendiğinizden emin olun.", + "author": "Yazar", + "this_plugin_can_do_following": "Bu eklenti aşağıdakileri yapabilir", + "install": "Yükle", + "install_a_metadata_provider": "Bir Meta Veri Sağlayıcısı Yükle", + "no_tracks_playing": "Şu anda çalınan bir Parça yok", + "synced_lyrics_not_available": "Bu şarkı için senkronize şarkı sözleri mevcut değil. Lütfen", + "plain_lyrics": "Düz Şarkı Sözleri", + "tab_instead": "sekmesini kullanın.", + "disclaimer": "Sorumluluk Reddi", + "third_party_plugin_dmca_notice": "Spotube ekibi, herhangi bir \"Üçüncü taraf\" eklentisi için herhangi bir sorumluluk (yasal olanlar dahil) kabul etmez.\nLütfen bunları kendi riskinizde kullanın. Herhangi bir hata/sorun için lütfen bunları eklenti deposuna bildirin.\n\nHerhangi bir \"Üçüncü taraf\" eklentisi bir hizmetin/yasal varlığın ToS/DMCA'sını ihlal ediyorsa, lütfen \"Üçüncü taraf\" eklenti yazarından veya barındırma platformundan, örneğin GitHub/Codeberg'den harekete geçmesini isteyin. Yukarıda listelenen (\"Üçüncü taraf\" olarak etiketlenen) eklentilerin tümü genel/topluluk tarafından sürdürülen eklentilerdir. Biz bunları küratörlüğünü yapmıyoruz, bu yüzden onlar üzerinde herhangi bir işlem yapamayız.\n\n", + "input_does_not_match_format": "Girdi, gerekli biçimle eşleşmiyor", + "metadata_provider_plugins": "Meta Veri Sağlayıcısı Eklentileri", + "paste_plugin_download_url": "İndirme url'sini veya GitHub/Codeberg repo url'sini veya .smplug dosyasına doğrudan bağlantıyı yapıştırın", + "download_and_install_plugin_from_url": "url'den eklentiyi indir ve yükle", + "failed_to_add_plugin_error": "Eklenti eklenemedi: {error}", + "upload_plugin_from_file": "Dosyadan eklenti yükle", + "installed": "Yüklü", + "available_plugins": "Mevcut eklentiler", + "configure_your_own_metadata_plugin": "Kendi çalma listenizi/albümünüzü/sanatçınızı/akış meta veri sağlayıcınızı yapılandırın", + "audio_scrobblers": "Ses Scrobbler'lar", + "scrobbling": "Scrobbling", + "download_music_format": "Müzik indirme formatı", + "streaming_music_format": "Müzik akış formatı", + "download_music_quality": "İndirilen müzik kalitesi", + "streaming_music_quality": "Yayınlanan müzik kalitesi", + "default_metadata_source": "Varsayılan meta veri kaynağı", + "set_default_metadata_source": "Varsayılan meta veri kaynağını ayarla", + "default_audio_source": "Varsayılan ses kaynağı", + "set_default_audio_source": "Varsayılan ses kaynağını ayarla", + "plugins": "Eklentiler", + "configure_plugins": "Kendi meta veri sağlayıcı ve ses kaynağı eklentilerinizi yapılandırın", + "source": "Kaynak: ", + "uncompressed": "Sıkıştırılmamış", + "dab_music_source_description": "Audiophile'ler için. Yüksek kaliteli/kayıpsız ses akışları sağlar. Doğru ISRC tabanlı parça eşleştirme." } \ No newline at end of file diff --git a/lib/l10n/app_uk.arb b/lib/l10n/app_uk.arb index 245c87e1..bdb723ad 100644 --- a/lib/l10n/app_uk.arb +++ b/lib/l10n/app_uk.arb @@ -401,5 +401,94 @@ "export_cache_files": "Експортувати кешовані файли", "found_n_files": "Знайдено {count} файлів", "export_cache_confirmation": "Ви хочете експортувати ці файли до", - "exported_n_out_of_m_files": "Експортовано {filesExported} з {files} файлів" + "exported_n_out_of_m_files": "Експортовано {filesExported} з {files} файлів", + "playlist": "Плейлист", + "no_loop": "Без повтору", + "generate": "Генерувати", + "undo": "Скасувати", + "download_all": "Завантажити все", + "add_all_to_playlist": "Додати все до плейлиста", + "add_all_to_queue": "Додати все в чергу", + "play_all_next": "Відтворити все наступне", + "pause": "Пауза", + "view_all": "Переглянути все", + "no_tracks_added_yet": "Здається, ви ще не додали жодної пісні", + "no_tracks": "Здається, тут немає пісень", + "no_tracks_listened_yet": "Здається, ви ще нічого не слухали", + "not_following_artists": "Ви не підписані на жодного артиста", + "no_favorite_albums_yet": "Здається, ви ще не додали жодного альбому в улюблені", + "no_logs_found": "Жодних журналів не знайдено", + "youtube_engine": "YouTube Двигун", + "youtube_engine_not_installed_title": "{engine} не встановлено", + "youtube_engine_not_installed_message": "{engine} не встановлено на вашій системі.", + "youtube_engine_set_path": "Переконайтесь, що він доступний у змінній PATH або\nвстановіть абсолютний шлях до виконуваного файлу {engine} нижче", + "youtube_engine_unix_issue_message": "У macOS/Linux/Unix-подібних ОС, встановлення шляху в .zshrc/.bashrc/.bash_profile тощо не працює.\nВам потрібно налаштувати шлях у файлі конфігурації оболонки", + "download": "Завантажити", + "file_not_found": "Файл не знайдено", + "custom": "Користувацький", + "add_custom_url": "Додати користувацький URL", + "edit_port": "Редагувати порт", + "port_helper_msg": "За замовчуванням -1, що означає випадкове число. Якщо у вас налаштований брандмауер, рекомендується це налаштувати.", + "connect_request": "Дозволити {client} підключення?", + "connection_request_denied": "Підключення відхилено. Користувач відмовив у доступі.", + "hipotetical_calculation": "*Це розраховано на основі середньої виплати за стрім онлайн-платформ для потокового відтворення музики, що становить від $0,003 до $0,005. Це гіпотетичний розрахунок, щоб дати користувачеві уявлення про те, скільки б вони заплатили артистам, якщо б слухали їхні пісні на різних музичних стрімінгових платформах.", + "an_error_occurred": "Сталася помилка", + "copy_to_clipboard": "Копіювати в буфер обміну", + "view_logs": "Переглянути логи", + "retry": "Повторити", + "no_default_metadata_provider_selected": "Ви не встановили провайдера метаданих за замовчуванням", + "manage_metadata_providers": "Керувати провайдерами метаданих", + "open_link_in_browser": "Відкрити посилання в браузері?", + "do_you_want_to_open_the_following_link": "Ви хочете відкрити наступне посилання", + "unsafe_url_warning": "Відкриття посилань з ненадійних джерел може бути небезпечним. Будьте обережні!\nВи також можете скопіювати посилання в буфер обміну.", + "copy_link": "Копіювати посилання", + "building_your_timeline": "Створення вашої часової шкали на основі ваших прослуховувань...", + "official": "Офіційний", + "author_name": "Автор: {author}", + "third_party": "Сторонній", + "plugin_requires_authentication": "Плагін вимагає автентифікації", + "update_available": "Доступне оновлення", + "supports_scrobbling": "Підтримує скроблінг", + "plugin_scrobbling_info": "Цей плагін скроббить вашу музику, щоб створити вашу історію прослуховувань.", + "default_plugin": "За замовчуванням", + "set_default": "Встановити за замовчуванням", + "support": "Підтримка", + "support_plugin_development": "Підтримати розробку плагіна", + "can_access_name_api": "- Може отримати доступ до **{name}** API", + "do_you_want_to_install_this_plugin": "Ви хочете встановити цей плагін?", + "third_party_plugin_warning": "Цей плагін із стороннього репозиторію. Будь ласка, переконайтеся, що ви довіряєте джерелу перед встановленням.", + "author": "Автор", + "this_plugin_can_do_following": "Цей плагін може робити наступне", + "install": "Встановити", + "install_a_metadata_provider": "Встановити провайдера метаданих", + "no_tracks_playing": "Наразі не відтворюється жоден трек", + "synced_lyrics_not_available": "Синхронізовані тексти недоступні для цієї пісні. Будь ласка, використовуйте вкладку", + "plain_lyrics": "Звичайні тексти", + "tab_instead": "замість цього.", + "disclaimer": "Відмова від відповідальності", + "third_party_plugin_dmca_notice": "Команда Spotube не несе жодної відповідальності (включно з юридичною) за будь-які плагіни \"третіх сторін\".\nБудь ласка, використовуйте їх на свій страх і ризик. Про будь-які помилки/проблеми повідомляйте в репозиторій плагіна.\n\nЯкщо якийсь плагін \"третьої сторони\" порушує ToS/DMCA будь-якої служби/юридичної особи, будь ласка, попросіть автора плагіна \"третьої сторони\" або хостингову платформу, наприклад, GitHub/Codeberg, вжити заходів. Усі перераховані вище (позначені як \"треті сторони\") є плагінами, які підтримуються публічно/спільнотою. Ми не куруємо їх, тому не можемо вжити жодних заходів щодо них.\n\n", + "input_does_not_match_format": "Введені дані не відповідають необхідному формату", + "metadata_provider_plugins": "Плагіни провайдера метаданих", + "paste_plugin_download_url": "Вставте URL-адресу для завантаження або URL-адресу репозиторію GitHub/Codeberg або пряме посилання на файл .smplug", + "download_and_install_plugin_from_url": "Завантажити та встановити плагін з URL-адреси", + "failed_to_add_plugin_error": "Не вдалося додати плагін: {error}", + "upload_plugin_from_file": "Завантажити плагін з файлу", + "installed": "Встановлено", + "available_plugins": "Доступні плагіни", + "configure_your_own_metadata_plugin": "Налаштуйте свій власний провайдер метаданих для плейлиста/альбому/виконавця/стрічки", + "audio_scrobblers": "Аудіо скробблери", + "scrobbling": "Скроблінг", + "download_music_format": "Формат завантаження музики", + "streaming_music_format": "Формат потокової музики", + "download_music_quality": "Якість завантаженої музики", + "streaming_music_quality": "Якість потокової музики", + "default_metadata_source": "Джерело метаданих за замовчуванням", + "set_default_metadata_source": "Встановити джерело метаданих за замовчуванням", + "default_audio_source": "Джерело аудіо за замовчуванням", + "set_default_audio_source": "Встановити джерело аудіо за замовчуванням", + "plugins": "Плагіни", + "configure_plugins": "Налаштуйте власні плагіни метаданих і аудіоджерела", + "source": "Джерело: ", + "uncompressed": "Без стиснення", + "dab_music_source_description": "Для аудіофілів. Забезпечує високоякісні/без втрат аудіопотоки. Точна відповідність треків на основі ISRC." } \ No newline at end of file diff --git a/lib/l10n/app_vi.arb b/lib/l10n/app_vi.arb index 37f7f709..5733963e 100644 --- a/lib/l10n/app_vi.arb +++ b/lib/l10n/app_vi.arb @@ -401,5 +401,94 @@ "export_cache_files": "Xuất các tệp được lưu trong bộ nhớ đệm", "found_n_files": "Tìm thấy {count} tệp", "export_cache_confirmation": "Bạn có muốn xuất các tệp này đến", - "exported_n_out_of_m_files": "Đã xuất {filesExported} trên {files} tệp" + "exported_n_out_of_m_files": "Đã xuất {filesExported} trên {files} tệp", + "playlist": "Danh sách phát", + "no_loop": "Không lặp lại", + "generate": "Tạo", + "undo": "Hoàn tác", + "download_all": "Tải xuống tất cả", + "add_all_to_playlist": "Thêm tất cả vào danh sách phát", + "add_all_to_queue": "Thêm tất cả vào danh sách chờ", + "play_all_next": "Chơi tất cả tiếp theo", + "pause": "Tạm dừng", + "view_all": "Xem tất cả", + "no_tracks_added_yet": "Có vẻ bạn chưa thêm bất kỳ bài hát nào", + "no_tracks": "Có vẻ không có bài hát nào ở đây", + "no_tracks_listened_yet": "Có vẻ bạn chưa nghe gì cả", + "not_following_artists": "Bạn không đang theo dõi bất kỳ nghệ sĩ nào", + "no_favorite_albums_yet": "Có vẻ bạn chưa thêm album nào vào danh sách yêu thích", + "no_logs_found": "Không tìm thấy nhật ký", + "youtube_engine": "Công cụ YouTube", + "youtube_engine_not_installed_title": "{engine} chưa được cài đặt", + "youtube_engine_not_installed_message": "{engine} chưa được cài đặt trong hệ thống của bạn.", + "youtube_engine_set_path": "Đảm bảo nó có sẵn trong biến PATH hoặc\nđặt đường dẫn tuyệt đối đến tệp thực thi {engine} dưới đây", + "youtube_engine_unix_issue_message": "Trên macOS/Linux/Unix, việc thiết lập đường dẫn trong .zshrc/.bashrc/.bash_profile v.v. sẽ không hoạt động.\nBạn cần thiết lập đường dẫn trong tệp cấu hình shell", + "download": "Tải xuống", + "file_not_found": "Không tìm thấy tệp", + "custom": "Tùy chỉnh", + "add_custom_url": "Thêm URL tùy chỉnh", + "edit_port": "Chỉnh sửa cổng", + "port_helper_msg": "Mặc định là -1, có nghĩa là số ngẫu nhiên. Nếu bạn đã cấu hình tường lửa, nên đặt điều này.", + "connect_request": "Cho phép {client} kết nối?", + "connection_request_denied": "Kết nối bị từ chối. Người dùng đã từ chối quyền truy cập.", + "hipotetical_calculation": "*Điều này được tính toán dựa trên khoản thanh toán trung bình mỗi luồng của nền tảng phát nhạc trực tuyến là $0,003 đến $0,005. Đây là một phép tính giả định để cung cấp cho người dùng cái nhìn sâu sắc về số tiền họ đã trả cho các nghệ sĩ nếu họ nghe bài hát của họ trên các nền tảng phát nhạc trực tuyến khác nhau.", + "an_error_occurred": "Đã xảy ra lỗi", + "copy_to_clipboard": "Sao chép vào khay nhớ tạm", + "view_logs": "Xem nhật ký", + "retry": "Thử lại", + "no_default_metadata_provider_selected": "Bạn chưa đặt nhà cung cấp siêu dữ liệu mặc định nào", + "manage_metadata_providers": "Quản lý nhà cung cấp siêu dữ liệu", + "open_link_in_browser": "Mở liên kết trong Trình duyệt?", + "do_you_want_to_open_the_following_link": "Bạn có muốn mở liên kết sau không", + "unsafe_url_warning": "Việc mở các liên kết từ các nguồn không đáng tin cậy có thể không an toàn. Hãy thận trọng!\nBạn cũng có thể sao chép liên kết vào khay nhớ tạm của mình.", + "copy_link": "Sao chép liên kết", + "building_your_timeline": "Đang xây dựng dòng thời gian của bạn dựa trên những gì bạn đã nghe...", + "official": "Chính thức", + "author_name": "Tác giả: {author}", + "third_party": "Bên thứ ba", + "plugin_requires_authentication": "Plugin yêu cầu xác thực", + "update_available": "Có bản cập nhật", + "supports_scrobbling": "Hỗ trợ scrobbling", + "plugin_scrobbling_info": "Plugin này scrobble nhạc của bạn để tạo lịch sử nghe của bạn.", + "default_plugin": "Mặc định", + "set_default": "Đặt làm mặc định", + "support": "Hỗ trợ", + "support_plugin_development": "Hỗ trợ phát triển plugin", + "can_access_name_api": "- Có thể truy cập API **{name}**", + "do_you_want_to_install_this_plugin": "Bạn có muốn cài đặt plugin này không?", + "third_party_plugin_warning": "Plugin này đến từ một kho lưu trữ của bên thứ ba. Vui lòng đảm bảo rằng bạn tin tưởng nguồn trước khi cài đặt.", + "author": "Tác giả", + "this_plugin_can_do_following": "Plugin này có thể làm những việc sau", + "install": "Cài đặt", + "install_a_metadata_provider": "Cài đặt một Nhà cung cấp siêu dữ liệu", + "no_tracks_playing": "Hiện không có bản nhạc nào đang phát", + "synced_lyrics_not_available": "Lời bài hát được đồng bộ hóa không có sẵn cho bài hát này. Vui lòng sử dụng", + "plain_lyrics": "Lời bài hát thuần túy", + "tab_instead": "thay thế.", + "disclaimer": "Miễn trừ trách nhiệm", + "third_party_plugin_dmca_notice": "Nhóm Spotube không chịu bất kỳ trách nhiệm nào (bao gồm cả pháp lý) đối với bất kỳ plugin \"Bên thứ ba\" nào.\nVui lòng sử dụng chúng với rủi ro của riêng bạn. Đối với bất kỳ lỗi/vấn đề nào, vui lòng báo cáo chúng cho kho lưu trữ plugin.\n\nNếu bất kỳ plugin \"Bên thứ ba\" nào vi phạm ToS/DMCA của bất kỳ dịch vụ/thực thể pháp lý nào, vui lòng yêu cầu tác giả plugin \"Bên thứ ba\" hoặc nền tảng lưu trữ, ví dụ: GitHub/Codeberg, thực hiện hành động. Tất cả các plugin được liệt kê ở trên (được gắn nhãn \"Bên thứ ba\") đều là các plugin công cộng/do cộng đồng duy trì. Chúng tôi không quản lý chúng, vì vậy chúng tôi không thể thực hiện bất kỳ hành động nào đối với chúng.\n\n", + "input_does_not_match_format": "Đầu vào không khớp với định dạng yêu cầu", + "metadata_provider_plugins": "Plugin Nhà cung cấp siêu dữ liệu", + "paste_plugin_download_url": "Dán url tải xuống hoặc url kho lưu trữ GitHub/Codeberg hoặc liên kết trực tiếp đến tệp .smplug", + "download_and_install_plugin_from_url": "Tải xuống và cài đặt plugin từ url", + "failed_to_add_plugin_error": "Không thể thêm plugin: {error}", + "upload_plugin_from_file": "Tải lên plugin từ tệp", + "installed": "Đã cài đặt", + "available_plugins": "Các plugin có sẵn", + "configure_your_own_metadata_plugin": "Cấu hình nhà cung cấp siêu dữ liệu danh sách phát/album/nghệ sĩ/nguồn cấp dữ liệu của riêng bạn", + "audio_scrobblers": "Bộ scrobbler âm thanh", + "scrobbling": "Scrobbling", + "download_music_format": "Định dạng nhạc tải về", + "streaming_music_format": "Định dạng nhạc phát trực tuyến", + "download_music_quality": "Chất lượng nhạc tải về", + "streaming_music_quality": "Chất lượng nhạc phát trực tuyến", + "default_metadata_source": "Nguồn siêu dữ liệu mặc định", + "set_default_metadata_source": "Đặt nguồn siêu dữ liệu mặc định", + "default_audio_source": "Nguồn âm thanh mặc định", + "set_default_audio_source": "Đặt nguồn âm thanh mặc định", + "plugins": "Tiện ích bổ sung", + "configure_plugins": "Cấu hình nhà cung cấp siêu dữ liệu và tiện ích nguồn âm thanh riêng", + "source": "Nguồn: ", + "uncompressed": "Không nén", + "dab_music_source_description": "Dành cho người yêu âm nhạc chất lượng cao. Cung cấp luồng âm thanh chất lượng cao/không nén. Phù hợp bài hát dựa trên ISRC chính xác." } \ No newline at end of file diff --git a/lib/l10n/app_zh.arb b/lib/l10n/app_zh.arb index dc2920ed..44f7d38c 100644 --- a/lib/l10n/app_zh.arb +++ b/lib/l10n/app_zh.arb @@ -179,8 +179,8 @@ "success_message": "你已经成功使用 Spotify 登录。干得漂亮!", "step_4": "步骤 4", "something_went_wrong": "某些地方出现了问题", - "piped_instance": "管道服务器实例", - "piped_description": "管道服务器实例用于匹配歌曲", + "piped_instance": "Piped 服务器实例", + "piped_description": "Piped 服务器实例用于匹配歌曲", "piped_warning": "它们中的一部分可能并不能正常工作。使用时请自行承担风险", "generate_playlist": "生成歌单", "track_exists": "歌曲 {track} 已存在", @@ -401,5 +401,94 @@ "export_cache_files": "导出缓存文件", "found_n_files": "找到 {count} 个文件", "export_cache_confirmation": "您要导出这些文件到", - "exported_n_out_of_m_files": "导出了 {filesExported} / {files} 个文件" + "exported_n_out_of_m_files": "导出了 {filesExported} / {files} 个文件", + "playlist": "播放列表", + "no_loop": "无循环", + "generate": "生成", + "undo": "撤销", + "download_all": "下载全部", + "add_all_to_playlist": "将全部添加到播放列表", + "add_all_to_queue": "将全部添加到队列", + "play_all_next": "播放全部下一首", + "pause": "暂停", + "view_all": "查看所有", + "no_tracks_added_yet": "看起来你还没有添加任何曲目", + "no_tracks": "看起来这里没有任何曲目", + "no_tracks_listened_yet": "看起来你还没有听任何东西", + "not_following_artists": "你没有关注任何艺术家", + "no_favorite_albums_yet": "看起来你还没有将任何专辑添加到收藏夹", + "no_logs_found": "未找到日志", + "youtube_engine": "YouTube 引擎", + "youtube_engine_not_installed_title": "{engine} 未安装", + "youtube_engine_not_installed_message": "{engine} 未在您的系统中安装。", + "youtube_engine_set_path": "确保它可用在 PATH 变量中,或\n设置 {engine} 可执行文件的绝对路径", + "youtube_engine_unix_issue_message": "在 macOS/Linux/Unix 类操作系统中,在 .zshrc/.bashrc/.bash_profile 等文件中设置路径无效。\n您需要在 shell 配置文件中设置路径", + "download": "下载", + "file_not_found": "文件未找到", + "custom": "自定义", + "add_custom_url": "添加自定义 URL", + "edit_port": "编辑端口", + "port_helper_msg": "默认值为-1,表示随机数。如果您已配置防火墙,建议设置此项。", + "connect_request": "允许 {client} 连接吗?", + "connection_request_denied": "连接被拒绝。用户拒绝访问。", + "hipotetical_calculation": "*这是根据在线音乐流媒体平台每流平均支付0.003美元至0.005美元计算得出的。这是一个假设性的计算,旨在让用户了解如果他们在不同的音乐流媒体平台上收听歌曲,他们将需要向艺人支付多少费用。", + "an_error_occurred": "发生错误", + "copy_to_clipboard": "复制到剪贴板", + "view_logs": "查看日志", + "retry": "重试", + "no_default_metadata_provider_selected": "您未设置默认元数据提供者", + "manage_metadata_providers": "管理元数据提供者", + "open_link_in_browser": "在浏览器中打开链接?", + "do_you_want_to_open_the_following_link": "您想打开以下链接吗", + "unsafe_url_warning": "从不受信任的来源打开链接可能不安全。请谨慎!\n您也可以将链接复制到剪贴板。", + "copy_link": "复制链接", + "building_your_timeline": "正在根据您的收听记录构建您的时间线...", + "official": "官方", + "author_name": "作者:{author}", + "third_party": "第三方", + "plugin_requires_authentication": "插件需要身份验证", + "update_available": "有可用更新", + "supports_scrobbling": "支持 Scrobbling", + "plugin_scrobbling_info": "此插件会 scrobble 您的音乐以生成您的收听历史记录。", + "default_plugin": "默认", + "set_default": "设为默认", + "support": "支持", + "support_plugin_development": "支持插件开发", + "can_access_name_api": "- 可以访问 **{name}** API", + "do_you_want_to_install_this_plugin": "您想安装此插件吗?", + "third_party_plugin_warning": "此插件来自第三方存储库。请在安装前确保您信任此来源。", + "author": "作者", + "this_plugin_can_do_following": "此插件可以执行以下操作", + "install": "安装", + "install_a_metadata_provider": "安装元数据提供者", + "no_tracks_playing": "当前没有播放任何曲目", + "synced_lyrics_not_available": "此歌曲的同步歌词不可用。请使用", + "plain_lyrics": "纯歌词", + "tab_instead": "选项卡。", + "disclaimer": "免责声明", + "third_party_plugin_dmca_notice": "Spotube 团队对任何“第三方”插件不承担任何责任(包括法律责任)。\n请自行承担风险使用。对于任何错误/问题,请向插件存储库报告。\n\n如果任何“第三方”插件违反了任何服务/法律实体的服务条款/DMCA,请要求该“第三方”插件作者或托管平台(例如 GitHub/Codeberg)采取行动。上面列出的(标记为“第三方”)都是公共/社区维护的插件。我们不对此类插件进行管理,因此无法对其采取任何行动。\n\n", + "input_does_not_match_format": "输入与所需格式不匹配", + "metadata_provider_plugins": "元数据提供者插件", + "paste_plugin_download_url": "粘贴下载 URL、GitHub/Codeberg 存储库 URL 或 .smplug 文件的直接链接", + "download_and_install_plugin_from_url": "从 URL 下载并安装插件", + "failed_to_add_plugin_error": "添加插件失败:{error}", + "upload_plugin_from_file": "从文件上传插件", + "installed": "已安装", + "available_plugins": "可用插件", + "configure_your_own_metadata_plugin": "配置您自己的播放列表/专辑/艺人/订阅元数据提供者", + "audio_scrobblers": "音频 Scrobblers", + "scrobbling": "Scrobbling", + "download_music_format": "下载音乐格式", + "streaming_music_format": "流媒体音乐格式", + "download_music_quality": "下载音乐质量", + "streaming_music_quality": "流媒体音乐质量", + "default_metadata_source": "默认元数据源", + "set_default_metadata_source": "设置默认元数据源", + "default_audio_source": "默认音频源", + "set_default_audio_source": "设置默认音频源", + "plugins": "插件", + "configure_plugins": "配置您自己的元数据提供者和音频源插件", + "source": "来源:", + "uncompressed": "无损", + "dab_music_source_description": "适合发烧友。提供高质量/无损音频流。基于 ISRC 的精确曲目匹配。" } \ No newline at end of file diff --git a/lib/l10n/app_zh_TW.arb b/lib/l10n/app_zh_TW.arb new file mode 100644 index 00000000..934006d5 --- /dev/null +++ b/lib/l10n/app_zh_TW.arb @@ -0,0 +1,494 @@ +{ + "guest": "訪客", + "browse": "瀏覽", + "search": "搜尋", + "library": "音樂庫", + "lyrics": "歌詞", + "settings": "設定", + "genre_categories_filter": "過濾分類...", + "genre": "探索歌單", + "personalized": "為你打造", + "featured": "推薦", + "new_releases": "新歌熱播", + "songs": "歌曲", + "playing_track": "播放 {track}", + "queue_clear_alert": "這將清空目前的播放清單。{track_length} 首歌曲將被移除\n你確定要繼續嗎?", + "load_more": "載入更多", + "playlists": "歌單", + "artists": "藝人", + "albums": "專輯", + "tracks": "歌曲", + "downloads": "下載", + "filter_playlists": "過濾歌單...", + "liked_tracks": "已按讚的歌曲", + "liked_tracks_description": "你按過讚的所有歌曲", + "create_playlist": "建立歌單", + "create_a_playlist": "建立一個歌單", + "create": "建立", + "cancel": "取消", + "playlist_name": "歌單名稱", + "name_of_playlist": "歌單的名稱", + "description": "說明", + "public": "公開", + "collaborative": "共享協作", + "search_local_tracks": "搜尋本地歌曲...", + "play": "播放", + "delete": "刪除", + "none": "無", + "sort_a_z": "依字母順序", + "sort_z_a": "依字母倒序", + "sort_artist": "按藝人", + "sort_album": "按專輯", + "sort_tracks": "排序方式", + "currently_downloading": "正在下載 ({tracks_length})", + "cancel_all": "取消全部", + "filter_artist": "過濾藝人...", + "followers": "{followers} 名追蹤者", + "add_artist_to_blacklist": "封鎖該藝人", + "top_tracks": "熱門歌曲", + "fans_also_like": "粉絲也喜歡", + "loading": "載入中...", + "artist": "藝人", + "blacklisted": "已封鎖", + "following": "關注中", + "follow": "關注", + "artist_url_copied": "此名藝人的分享連結已複製至剪貼簿", + "added_to_queue": "已新增 {tracks} 首歌曲到播放清單", + "filter_albums": "過濾專輯...", + "synced": "同步", + "plain": "未同步", + "shuffle": "隨機播放", + "search_tracks": "搜尋歌曲...", + "released": "發表時間", + "error": "發生錯誤: {error}", + "title": "標題", + "time": "時長", + "more_actions": "更多動作", + "download_count": "下載 ({count}) 首歌曲", + "add_count_to_playlist": "將 ({count}) 首歌曲新增到歌單中", + "add_count_to_queue": "新增 ({count}) 首歌曲到播放清單", + "play_count_next": "接下來將播放 ({count}) 首歌曲", + "album": "專輯", + "copied_to_clipboard": "已將 {data} 複製至剪貼簿", + "add_to_following_playlists": "新增 {track} 到以下播放清單", + "add": "新增", + "added_track_to_queue": "新增 {track} 到播放清單", + "add_to_queue": "新增至播放清單", + "track_will_play_next": "{track} 將在下一首播放", + "play_next": "下一首播放", + "removed_track_from_queue": "將 {track} 從播放清單移除", + "remove_from_queue": "從播放清單移除", + "remove_from_favorites": "取消按讚", + "save_as_favorite": "按讚", + "add_to_playlist": "新增到歌單", + "remove_from_playlist": "從歌單移除", + "add_to_blacklist": "新增到已封鎖清單", + "remove_from_blacklist": "從已封鎖清單移除", + "share": "分享", + "mini_player": "小窗模式", + "slide_to_seek": "滑動以前進或後退", + "shuffle_playlist": "隨機播放歌單", + "unshuffle_playlist": "取消隨機播放歌單", + "previous_track": "上一首歌曲", + "next_track": "下一首歌", + "pause_playback": "暫停播放", + "resume_playback": "恢復播放", + "loop_track": "單曲循環", + "repeat_playlist": "歌單循環", + "queue": "播放清單", + "alternative_track_sources": "其它音源", + "download_track": "下載歌曲", + "tracks_in_queue": "{tracks} 首歌曲在播放清單中", + "clear_all": "清除全部", + "show_hide_ui_on_hover": "游標暫留時顯示 / 隱藏控制列", + "always_on_top": "置頂", + "exit_mini_player": "退出小窗模式", + "download_location": "下載路徑", + "account": "帳戶", + "login_with_spotify": "使用 Spotify 登入", + "connect_with_spotify": "與 Spotify 帳號連結", + "logout": "退出", + "logout_of_this_account": "退出該帳戶", + "language_region": "語言與地區", + "language": "語言", + "system_default": "系統預設", + "market_place_region": "市集地區", + "recommendation_country": "請選擇國家與地區以取得對應的音樂推薦", + "appearance": "外觀", + "layout_mode": "佈局類型", + "override_layout_settings": "將覆寫響應式佈局設定", + "adaptive": "響應式", + "compact": "緊湊", + "extended": "寬闊", + "theme": "主題", + "dark": "深色", + "light": "淺色", + "system": "依循系統", + "accent_color": "主色調", + "sync_album_color": "符合封面顏色", + "sync_album_color_description": "選取專輯封面主題色為主色調", + "playback": "播放", + "audio_quality": "音質", + "high": "高", + "low": "低", + "pre_download_play": "下載後播放", + "pre_download_play_description": "先下載歌曲後再播放而非串流播放(建議頻寬較高使用者使用)", + "skip_non_music": "跳過非音樂片段(跳過贊助商廣告)", + "blacklist_description": "已封鎖的歌曲與藝人", + "wait_for_download_to_finish": "請等待目前下載工作完成", + "desktop": "桌面版設定", + "close_behavior": "點選關閉按鈕行為", + "close": "關閉", + "minimize_to_tray": "最小化到工作列", + "show_tray_icon": "顯示工作列圖示", + "about": "關於", + "u_love_spotube": "我們明白你喜歡 Spotube", + "check_for_updates": "檢查更新", + "about_spotube": "關於 Spotube", + "blacklist": "黑名單", + "please_sponsor": "請考慮贊助或捐款", + "spotube_description": "Spotube,一款輕量、跨平台且完全免費的 Spotify 用戶端。", + "version": "版本", + "build_number": "建置編號", + "founder": "發起人", + "repository": "專案儲存庫", + "bug_issues": "缺陷與問題報告", + "made_with": "於孟加拉🇧🇩用 ❤️ 發電", + "kingkor_roy_tirtho": "Kingkor Roy Tirtho", + "copyright": "© 2021-{current_year} Kingkor Roy Tirtho", + "license": "授權", + "add_spotify_credentials": "新增你的 Spotify 登入資訊以開始使用", + "credentials_will_not_be_shared_disclaimer": "您大可放心,軟體不會收集或分享任何個人資料給第三方", + "know_how_to_login": "不知道該怎麼辦?", + "follow_step_by_step_guide": "請依照以下說明進行", + "spotify_cookie": "Spotify {name} Cookie", + "cookie_name_cookie": "{name} Cookie", + "fill_in_all_fields": "請填入所有欄位", + "submit": "提交", + "exit": "退出", + "previous": "上一步", + "next": "下一步", + "done": "完成", + "step_1": "步驟 1", + "first_go_to": "首先,前往", + "login_if_not_logged_in": "如果尚未登入,請登入或註冊帳戶", + "step_2": "步驟 2", + "step_2_steps": "1. 一旦你已經完成登入, 按 F12 鍵或滑鼠右鍵點選網頁空白區域 > 選擇「檢查」以開啟瀏覽器開發者工具(DevTools)\n2. 選擇 \"應用程式(Application)\" 分頁(Chrome, Edge, Brave 等基於 Chromium 記憶體或基於 Choxage, nox Firefox 的瀏覽器))\n3. 選擇 \"Cookies\" 欄位然後選擇 \"https://accounts.spotify.com\" 子選單", + "step_3": "步驟 3", + "success_emoji": "成功🥳", + "success_message": "你已經成功使用 Spotify 登入。幹得漂亮!", + "step_4": "步驟 4", + "something_went_wrong": "某些地方出現了問題", + "piped_instance": "Piped 伺服器實例", + "piped_description": "Piped 伺服器實例用於匹配歌曲", + "piped_warning": "它們之中的一部分可能無法正常運作。使用時請自行承擔風險", + "generate_playlist": "產生歌單", + "track_exists": "曲目 {track} 已存在", + "replace_downloaded_tracks": "替換已下載的歌曲", + "skip_download_tracks": "下載時跳過已下載的歌曲", + "do_you_want_to_replace": "你確定要取代已下載的歌曲嗎??", + "replace": "取代", + "skip": "跳過", + "select_up_to_count_type": "選擇最多 {count} 種的類型 {type}", + "select_genres": "選擇曲風", + "add_genres": "新增曲風", + "country": "國家和地區", + "number_of_tracks_generate": "產生歌曲的數目", + "acousticness": "原聲程度", + "danceability": "律動感", + "energy": "衝擊感", + "instrumentalness": "歌唱部分佔比", + "liveness": "現場感", + "loudness": "響度", + "speechiness": "朗誦比例", + "valence": "心理感受", + "popularity": "流行度", + "key": "曲調", + "duration": "歌曲長度 (s)", + "tempo": "每分鐘拍數 (BPM)", + "mode": "旋律重複度", + "time_signature": "音符時值", + "short": "短", + "medium": "中", + "long": "長", + "min": "最低", + "max": "最高", + "target": "目標", + "moderate": "中", + "deselect_all": "取消全選", + "select_all": "全選", + "are_you_sure": "你確定嗎?", + "generating_playlist": "正在產生你的自訂歌單...", + "selected_count_tracks": "已選取 {count} 首歌曲", + "download_warning": "如果你大量下載這些歌曲,你顯然在侵犯音樂的版權並對音樂創作社區造成了傷害。我希望你能意識到這一點。永遠要尊重並支持藝術家們的辛勤工作", + "download_ip_ban_warning": "小心,如果出現超出正常的下載請求,那你的 IP 可能會被 YouTube 封鎖,這意味著你的裝置將在長達 2-3 個月的時間內無法使用該 IP 訪問 YouTube(即使你沒登入)。Spotube 不會因而承擔任何責任", + "by_clicking_accept_terms": "點擊 '同意' 代表你同意以下的條款", + "download_agreement_1": "我明白侵害音樂版權是一件不好的事", + "download_agreement_2": "我將盡可能支持藝術家的工作。我現在之所以做不到是因為缺乏資金來購買正版", + "download_agreement_3": "我完全了解我的 IP 存在被 YouTube 封鎖的風險。並且我明白 Spotube 的擁有者與貢獻者們無須對我目前的行為所導致的任何後果負責", + "decline": "拒絕", + "accept": "同意", + "details": "詳細資訊", + "youtube": "YouTube", + "channel": "頻道", + "likes": "讚", + "dislikes": "倒讚", + "views": "瀏覽次數", + "streamUrl": "播放串流 URL", + "stop": "停止", + "sort_newest": "依新增日期順序", + "sort_oldest": "依新增日期倒序", + "sleep_timer": "睡眠計時器", + "mins": "{minutes} 分", + "hours": "{hours} 時", + "hour": "{hours} 時", + "custom_hours": "自訂時長", + "logs": "記錄檔(Log)", + "developers": "開發者", + "not_logged_in": "你尚未登入", + "search_mode": "搜尋模式", + "audio_source": "音訊來源", + "ok": "確定", + "failed_to_encrypt": "加密失敗", + "encryption_failed_warning": "Spotube使用加密來安全地儲存您的資料。但是失敗了。因此,它將回退到不安全的儲存空間\n如果您使用Linux,請確保已安裝gnome-keyring、kde-wallet和keepassxc等加密服務", + "querying_info": "正在查詢資訊...", + "piped_api_down": "Piped API 無法使用", + "piped_down_error_instructions": "當前Piped實例 {pipedInstance} 不可用\n\n請更改實例或將'API類型'更改為官方YouTube API\n\n更改後請確保重新啟動應用程式", + "you_are_offline": "您目前處於離線狀態", + "connection_restored": "您的網路連線已恢復", + "use_system_title_bar": "使用作業系統的預設視窗標題列", + "update_playlist": "更新播放清單", + "update": "更新", + "crunching_results": "處理結果中...", + "search_to_get_results": "搜尋以取得結果", + "use_amoled_mode": "使用 AMOLED 模式", + "pitch_dark_theme": "漆黑主題", + "normalize_audio": "標準化音訊", + "change_cover": "更改封面", + "add_cover": "新增封面", + "restore_defaults": "恢復預設值", + "download_music_codec": "下載音樂編解碼器", + "streaming_music_codec": "串流音樂編解碼器", + "login_with_lastfm": "使用 Last.fm 登入", + "connect": "連線", + "disconnect_lastfm": "切斷 Last.fm 連線", + "disconnect": "斷開連線", + "username": "帳號", + "password": "密碼", + "login": "登入", + "login_with_your_lastfm": "使用您的 Last.fm 帳號登入", + "scrobble_to_lastfm": "在 Last.fm 上記錄你的播放", + "go_to_album": "前往專輯", + "discord_rich_presence": "Discord Rick Presence(Discord 狀態)", + "browse_all": "瀏覽全部", + "genres": "音樂類型", + "explore_genres": "探索音樂類型", + "step_3_steps": "複製\"sp_dc\" Cookie的值", + "step_4_steps": "貼上複製的\"sp_dc\"值", + "friends": "好友", + "no_lyrics_available": "抱歉,無法找到這首歌的歌詞", + "sort_duration": "依長度排序", + "start_a_radio": "開始收聽電台", + "how_to_start_radio": "您想如何開始收聽電台?", + "replace_queue_question": "您想要取代目前清單還是追加到清單?", + "endless_playback": "無限播放", + "delete_playlist": "刪除播放清單", + "delete_playlist_confirmation": "您確定要刪除此播放清單嗎?", + "local_tracks": "本地音訊", + "song_link": "歌曲連結", + "skip_this_nonsense": "跳過這個無聊內容", + "freedom_of_music": "“音樂的自由”", + "freedom_of_music_palm": "「音樂的自由掌握在您手中」", + "get_started": "我們開始吧", + "youtube_source_description": "建議且效果最佳。", + "piped_source_description": "感覺自由?與 YouTube 一樣,但更自由。", + "jiosaavn_source_description": "最適合南亞地區。", + "highest_quality": "最高音質:{quality}", + "select_audio_source": "選擇音訊來源", + "endless_playback_description": "自動將新歌曲加入清單的結尾", + "choose_your_region": "選擇您的所在地區", + "choose_your_region_description": "這能幫助 Spotube 為您的所在位置顯示正確的內容。", + "choose_your_language": "選擇您的語言", + "help_project_grow": "幫助這個專案成長", + "help_project_grow_description": "Spotube是一個開源專案。您可以透過為專案做出貢獻、回報錯誤或建議新功能來幫助專案成長。", + "contribute_on_github": "在GitHub上做出貢獻", + "donate_on_open_collective": "在Open Collective上捐款", + "browse_anonymously": "匿名瀏覽", + "enable_connect": "啟用連線", + "enable_connect_description": "從其他裝置控制Spotube", + "devices": "裝置", + "select": "選擇", + "connect_client_alert": "您正在被 {client} 控制", + "this_device": "此裝置", + "remote": "遠端", + "local_library": "本地媒體庫", + "add_library_location": "新增至媒體庫", + "remove_library_location": "從媒體庫移除", + "local_tab": "本地", + "stats": "統計", + "and_n_more": "還有 {count} 個", + "recently_played": "最近播放", + "browse_more": "瀏覽更多", + "no_title": "無標題", + "not_playing": "未播放", + "epic_failure": "史詩級的失敗!", + "added_num_tracks_to_queue": "已將 {tracks_length} 首曲目新增至清單", + "spotube_has_an_update": "Spotube 有更新版本", + "download_now": "立即下載", + "nightly_version": "Spotube Nightly {nightlyBuildNum} 已發佈", + "release_version": "Spotube v{version} 已發布", + "read_the_latest": "閱讀最新", + "release_notes": "版本說明", + "pick_color_scheme": "選擇配色方案", + "save": "儲存", + "choose_the_device": "選擇裝置:", + "multiple_device_connected": "已連接多個裝置。\n選擇您希望執行此操作的裝置", + "nothing_found": "未找到任何內容", + "the_box_is_empty": "箱子為空", + "top_artists": "熱門藝人", + "top_albums": "熱門專輯", + "this_week": "本週", + "this_month": "本月", + "last_6_months": "過去6個月", + "this_year": "今年", + "last_2_years": "過去2年", + "all_time": "所有時間", + "powered_by_provider": "由 {providerName} 提供支援", + "email": "電子郵件", + "profile_followers": "追蹤者", + "birthday": "生日", + "subscription": "訂閱", + "not_born": "尚未建立", + "hacker": "駭客", + "profile": "個人資訊", + "no_name": "沒有名字", + "edit": "編輯", + "user_profile": "使用者資料", + "count_plays": "{count} 次播放", + "streaming_fees_hypothetical": "*基於 Spotify 每次播放的支付金額\n從 $0.003 到 $0.005 計算。這是一個假設性的\n計算,旨在讓用戶了解如果他們在 Spotify 上收聽\n這些歌曲,可能會付給作者的金額。", + "count_mins": "{minutes} 分鐘", + "summary_minutes": "分鐘", + "summary_listened_to_music": "聽音樂", + "summary_songs": "歌曲", + "summary_streamed_overall": "整體串流媒體", + "summary_owed_to_artists": "本月欠藝術家的", + "summary_artists": "藝術家的", + "summary_music_reached_you": "音樂接觸到你", + "summary_full_albums": "完整專輯", + "summary_got_your_love": "獲得了你的愛心", + "summary_playlists": "播放清單", + "summary_were_on_repeat": "已經重複播放", + "total_money": "總計 {money}", + "minutes_listened": "聽的分鐘數", + "streamed_songs": "已串流歌曲", + "count_streams": "{count} 次串流", + "owned_by_you": "由您所有", + "copied_shareurl_to_clipboard": "{shareUrl} 已複製到剪貼簿", + "spotify_hipotetical_calculation": "*根據 Spotify 每次串流媒體的支付金額\n$0.003 到 $0.005 進行計算。這是一個假設性的\n計算,用於給用戶了解他們如果在 Spotify 上\n收聽歌曲會支付給藝術家的金額。", + "webview_not_found": "未找到 Webview 框架", + "webview_not_found_description": "您的裝置中未安裝 Webview Runtime。\n如果已安裝,請確保它的位置在系統環境變數(PATH)中\n\n安裝後,重新啟動應用程式", + "unsupported_platform": "不支援的平台", + "invidious_instance": "Invidious 伺服器實例", + "invidious_description": "用於音軌匹配的 Invidious 伺服器實例", + "invidious_warning": "有些可能無法正常運作。請自行承擔風險", + "invidious_source_description": "類似 Piped,但可用性更高。", + "cache_music": "快取音樂", + "open": "開啟", + "cache_folder": "快取資料夾", + "export": "導出", + "clear_cache": "清除快取", + "clear_cache_confirmation": "您要清除快取嗎?", + "export_cache_files": "匯出快取檔案", + "found_n_files": "找到 {count} 個檔案", + "export_cache_confirmation": "您要匯出這些檔案到", + "exported_n_out_of_m_files": "匯出了 {filesExported} / {files} 個檔案", + "playlist": "播放清單", + "no_loop": "無循環", + "generate": "生成", + "undo": "取消", + "download_all": "下載全部", + "add_all_to_playlist": "全部加入到播放清單", + "add_all_to_queue": "全部加入清單", + "play_all_next": "播放全部下一首", + "pause": "暫停", + "view_all": "檢視全部", + "no_tracks_added_yet": "看起來你還沒有加入任何歌曲", + "no_tracks": "看起來這裡沒有任何歌曲", + "no_tracks_listened_yet": "看起來你還沒聽任何歌曲", + "not_following_artists": "你沒有關注任何藝術家", + "no_favorite_albums_yet": "看起來你還沒有將任何專輯加入到收藏夾", + "no_logs_found": "未找到日誌", + "youtube_engine": "YouTube 引擎", + "youtube_engine_not_installed_title": "{engine} 未安裝", + "youtube_engine_not_installed_message": "{engine} 未在您的系統中安裝。", + "youtube_engine_set_path": "確保它可用在 PATH 變數中,或\n設定 {engine} 執行檔的絕對路徑", + "youtube_engine_unix_issue_message": "在類 Unix 作業系統(如 macOS/Linux/Unix)中,請在 .zshrc/.bashrc/.bash_profile 等檔案中設定路徑無效。\n您需要在 shell 設定檔中設定路徑", + "download": "下載", + "file_not_found": "找不到檔案", + "custom": "自訂", + "add_custom_url": "新增自訂 URL", + "edit_port": "編輯端口", + "port_helper_msg": "預設值為 -1,表示隨機數。如果您已配置防火牆,建議設定此項目。", + "connect_request": "允許 {client} 連線嗎?", + "connection_request_denied": "連線被拒絕。請求被使用者拒絕。", + "hipotetical_calculation": "*此為根據線上音樂串流平台平均每次播放 $0.003 至 $0.005 的收益所計算的假設值。此為一個假設性計算,旨在讓使用者了解若他們在不同的音樂串流平台上收聽同一首歌曲,他們將會支付給藝人多少費用。", + "an_error_occurred": "發生錯誤", + "copy_to_clipboard": "複製到剪貼簿", + "view_logs": "檢視日誌", + "retry": "重試", + "no_default_metadata_provider_selected": "您沒有設定預設的中繼資料供應商", + "manage_metadata_providers": "管理中繼資料供應商", + "open_link_in_browser": "要在瀏覽器中開啟連結嗎?", + "do_you_want_to_open_the_following_link": "您想開啟以下連結嗎", + "unsafe_url_warning": "從不受信任的來源開啟連結可能不安全。請務必小心!\n您也可以將連結複製到剪貼簿。", + "copy_link": "複製連結", + "building_your_timeline": "正在根據您的收聽記錄建立您的時間軸...", + "official": "官方", + "author_name": "作者:{author}", + "third_party": "第三方", + "plugin_requires_authentication": "此外掛程式需要驗證", + "update_available": "有可用的更新", + "supports_scrobbling": "支援 Scrobbling", + "plugin_scrobbling_info": "此外掛程式會 Scrobble 您的音樂以產生您的收聽記錄。", + "default_plugin": "預設", + "set_default": "設為預設", + "support": "支援", + "support_plugin_development": "支援外掛程式開發", + "can_access_name_api": "- 可以存取 **{name}** API", + "do_you_want_to_install_this_plugin": "您想安裝此外掛程式嗎?", + "third_party_plugin_warning": "此外掛程式來自第三方儲存庫。請在安裝前確認您信任該來源。", + "author": "作者", + "this_plugin_can_do_following": "此外掛程式可以執行以下操作", + "install": "安裝", + "install_a_metadata_provider": "安裝中繼資料供應商", + "no_tracks_playing": "目前沒有正在播放的曲目", + "synced_lyrics_not_available": "此歌曲沒有同步歌詞。請改用", + "plain_lyrics": "純歌詞", + "tab_instead": "分頁。", + "disclaimer": "免責聲明", + "third_party_plugin_dmca_notice": "Spotube 團隊對任何「第三方」外掛程式不負任何責任(包括法律責任)。\n請自行承擔使用風險。如有任何錯誤/問題,請向該外掛程式的儲存庫回報。\n\n若有任何「第三方」外掛程式違反任何服務/法律實體的服務條款/DMCA,請向「第三方」外掛程式作者或託管平台(如 GitHub/Codeberg)要求採取行動。以上列出的(標記為「第三方」)外掛程式均為公開/社群維護的外掛程式。我們沒有對其進行審核,因此無法對其採取任何行動。\n\n", + "input_does_not_match_format": "輸入不符合所需格式", + "metadata_provider_plugins": "中繼資料供應商外掛程式", + "paste_plugin_download_url": "貼上下載網址、GitHub/Codeberg 儲存庫網址或 .smplug 檔案的直接連結", + "download_and_install_plugin_from_url": "從網址下載並安裝外掛程式", + "failed_to_add_plugin_error": "新增外掛程式失敗:{error}", + "upload_plugin_from_file": "從檔案上傳外掛程式", + "installed": "已安裝", + "available_plugins": "可用的外掛程式", + "configure_your_own_metadata_plugin": "設定您自己的播放清單/專輯/藝人/動態中繼資料供應商", + "audio_scrobblers": "音訊 Scrobblers", + "scrobbling": "Scrobbling", + "download_music_format": "下載音樂格式", + "streaming_music_format": "串流音樂格式", + "download_music_quality": "下載音樂品質", + "streaming_music_quality": "串流音樂品質", + "default_metadata_source": "預設中繼資料來源", + "set_default_metadata_source": "設定預設中繼資料來源", + "default_audio_source": "預設音訊來源", + "set_default_audio_source": "設定預設音訊來源", + "plugins": "外掛程式", + "configure_plugins": "配置您自己的中繼資料提供者和音訊來源外掛程式", + "source": "來源:", + "uncompressed": "未壓縮", + "dab_music_source_description": "適合音響發燒友。提供高品質/無損音訊串流。精確的 ISRC 曲目比對。" +} \ No newline at end of file diff --git a/lib/l10n/generated/app_localizations.dart b/lib/l10n/generated/app_localizations.dart new file mode 100644 index 00000000..e9d7913d --- /dev/null +++ b/lib/l10n/generated/app_localizations.dart @@ -0,0 +1,3109 @@ +import 'dart:async'; + +import 'package:flutter/foundation.dart'; +import 'package:flutter/widgets.dart'; +import 'package:flutter_localizations/flutter_localizations.dart'; +import 'package:intl/intl.dart' as intl; + +import 'app_localizations_ar.dart'; +import 'app_localizations_bn.dart'; +import 'app_localizations_ca.dart'; +import 'app_localizations_cs.dart'; +import 'app_localizations_de.dart'; +import 'app_localizations_en.dart'; +import 'app_localizations_es.dart'; +import 'app_localizations_eu.dart'; +import 'app_localizations_fa.dart'; +import 'app_localizations_fi.dart'; +import 'app_localizations_fr.dart'; +import 'app_localizations_hi.dart'; +import 'app_localizations_id.dart'; +import 'app_localizations_it.dart'; +import 'app_localizations_ja.dart'; +import 'app_localizations_ka.dart'; +import 'app_localizations_ko.dart'; +import 'app_localizations_ne.dart'; +import 'app_localizations_nl.dart'; +import 'app_localizations_pl.dart'; +import 'app_localizations_pt.dart'; +import 'app_localizations_ru.dart'; +import 'app_localizations_ta.dart'; +import 'app_localizations_th.dart'; +import 'app_localizations_tl.dart'; +import 'app_localizations_tr.dart'; +import 'app_localizations_uk.dart'; +import 'app_localizations_vi.dart'; +import 'app_localizations_zh.dart'; + +// ignore_for_file: type=lint + +/// Callers can lookup localized strings with an instance of AppLocalizations +/// returned by `AppLocalizations.of(context)`. +/// +/// Applications need to include `AppLocalizations.delegate()` in their app's +/// `localizationDelegates` list, and the locales they support in the app's +/// `supportedLocales` list. For example: +/// +/// ```dart +/// import 'generated/app_localizations.dart'; +/// +/// return MaterialApp( +/// localizationsDelegates: AppLocalizations.localizationsDelegates, +/// supportedLocales: AppLocalizations.supportedLocales, +/// home: MyApplicationHome(), +/// ); +/// ``` +/// +/// ## Update pubspec.yaml +/// +/// Please make sure to update your pubspec.yaml to include the following +/// packages: +/// +/// ```yaml +/// dependencies: +/// # Internationalization support. +/// flutter_localizations: +/// sdk: flutter +/// intl: any # Use the pinned version from flutter_localizations +/// +/// # Rest of dependencies +/// ``` +/// +/// ## iOS Applications +/// +/// iOS applications define key application metadata, including supported +/// locales, in an Info.plist file that is built into the application bundle. +/// To configure the locales supported by your app, you’ll need to edit this +/// file. +/// +/// First, open your project’s ios/Runner.xcworkspace Xcode workspace file. +/// Then, in the Project Navigator, open the Info.plist file under the Runner +/// project’s Runner folder. +/// +/// Next, select the Information Property List item, select Add Item from the +/// Editor menu, then select Localizations from the pop-up menu. +/// +/// Select and expand the newly-created Localizations item then, for each +/// locale your application supports, add a new item and select the locale +/// you wish to add from the pop-up menu in the Value field. This list should +/// be consistent with the languages listed in the AppLocalizations.supportedLocales +/// property. +abstract class AppLocalizations { + AppLocalizations(String locale) + : localeName = intl.Intl.canonicalizedLocale(locale.toString()); + + final String localeName; + + static AppLocalizations? of(BuildContext context) { + return Localizations.of(context, AppLocalizations); + } + + static const LocalizationsDelegate delegate = + _AppLocalizationsDelegate(); + + /// A list of this localizations delegate along with the default localizations + /// delegates. + /// + /// Returns a list of localizations delegates containing this delegate along with + /// GlobalMaterialLocalizations.delegate, GlobalCupertinoLocalizations.delegate, + /// and GlobalWidgetsLocalizations.delegate. + /// + /// Additional delegates can be added by appending to this list in + /// MaterialApp. This list does not have to be used at all if a custom list + /// of delegates is preferred or required. + static const List> localizationsDelegates = + >[ + delegate, + GlobalMaterialLocalizations.delegate, + GlobalCupertinoLocalizations.delegate, + GlobalWidgetsLocalizations.delegate, + ]; + + /// A list of this localizations delegate's supported locales. + static const List supportedLocales = [ + Locale('ar'), + Locale('bn'), + Locale('ca'), + Locale('cs'), + Locale('de'), + Locale('en'), + Locale('es'), + Locale('eu'), + Locale('fa'), + Locale('fi'), + Locale('fr'), + Locale('hi'), + Locale('id'), + Locale('it'), + Locale('ja'), + Locale('ka'), + Locale('ko'), + Locale('ne'), + Locale('nl'), + Locale('pl'), + Locale('pt'), + Locale('ru'), + Locale('ta'), + Locale('th'), + Locale('tl'), + Locale('tr'), + Locale('uk'), + Locale('vi'), + Locale('zh'), + Locale('zh', 'TW') + ]; + + /// No description provided for @guest. + /// + /// In en, this message translates to: + /// **'Guest'** + String get guest; + + /// No description provided for @browse. + /// + /// In en, this message translates to: + /// **'Browse'** + String get browse; + + /// No description provided for @search. + /// + /// In en, this message translates to: + /// **'Search'** + String get search; + + /// No description provided for @library. + /// + /// In en, this message translates to: + /// **'Library'** + String get library; + + /// No description provided for @lyrics. + /// + /// In en, this message translates to: + /// **'Lyrics'** + String get lyrics; + + /// No description provided for @settings. + /// + /// In en, this message translates to: + /// **'Settings'** + String get settings; + + /// No description provided for @genre_categories_filter. + /// + /// In en, this message translates to: + /// **'Filter categories or genres...'** + String get genre_categories_filter; + + /// No description provided for @genre. + /// + /// In en, this message translates to: + /// **'Genre'** + String get genre; + + /// No description provided for @personalized. + /// + /// In en, this message translates to: + /// **'Personalized'** + String get personalized; + + /// No description provided for @featured. + /// + /// In en, this message translates to: + /// **'Featured'** + String get featured; + + /// No description provided for @new_releases. + /// + /// In en, this message translates to: + /// **'New Releases'** + String get new_releases; + + /// No description provided for @songs. + /// + /// In en, this message translates to: + /// **'Songs'** + String get songs; + + /// No description provided for @playing_track. + /// + /// In en, this message translates to: + /// **'Playing {track}'** + String playing_track(Object track); + + /// No description provided for @queue_clear_alert. + /// + /// In en, this message translates to: + /// **'This will clear the current queue. {track_length} tracks will be removed\nDo you want to continue?'** + String queue_clear_alert(Object track_length); + + /// No description provided for @load_more. + /// + /// In en, this message translates to: + /// **'Load more'** + String get load_more; + + /// No description provided for @playlists. + /// + /// In en, this message translates to: + /// **'Playlists'** + String get playlists; + + /// No description provided for @artists. + /// + /// In en, this message translates to: + /// **'Artists'** + String get artists; + + /// No description provided for @albums. + /// + /// In en, this message translates to: + /// **'Albums'** + String get albums; + + /// No description provided for @tracks. + /// + /// In en, this message translates to: + /// **'Tracks'** + String get tracks; + + /// No description provided for @downloads. + /// + /// In en, this message translates to: + /// **'Downloads'** + String get downloads; + + /// No description provided for @filter_playlists. + /// + /// In en, this message translates to: + /// **'Filter your playlists...'** + String get filter_playlists; + + /// No description provided for @liked_tracks. + /// + /// In en, this message translates to: + /// **'Liked Tracks'** + String get liked_tracks; + + /// No description provided for @liked_tracks_description. + /// + /// In en, this message translates to: + /// **'All your liked tracks'** + String get liked_tracks_description; + + /// No description provided for @playlist. + /// + /// In en, this message translates to: + /// **'Playlist'** + String get playlist; + + /// No description provided for @create_a_playlist. + /// + /// In en, this message translates to: + /// **'Create a playlist'** + String get create_a_playlist; + + /// No description provided for @update_playlist. + /// + /// In en, this message translates to: + /// **'Update playlist'** + String get update_playlist; + + /// No description provided for @create. + /// + /// In en, this message translates to: + /// **'Create'** + String get create; + + /// No description provided for @cancel. + /// + /// In en, this message translates to: + /// **'Cancel'** + String get cancel; + + /// No description provided for @update. + /// + /// In en, this message translates to: + /// **'Update'** + String get update; + + /// No description provided for @playlist_name. + /// + /// In en, this message translates to: + /// **'Playlist Name'** + String get playlist_name; + + /// No description provided for @name_of_playlist. + /// + /// In en, this message translates to: + /// **'Name of the playlist'** + String get name_of_playlist; + + /// No description provided for @description. + /// + /// In en, this message translates to: + /// **'Description'** + String get description; + + /// No description provided for @public. + /// + /// In en, this message translates to: + /// **'Public'** + String get public; + + /// No description provided for @collaborative. + /// + /// In en, this message translates to: + /// **'Collaborative'** + String get collaborative; + + /// No description provided for @search_local_tracks. + /// + /// In en, this message translates to: + /// **'Search local tracks...'** + String get search_local_tracks; + + /// No description provided for @play. + /// + /// In en, this message translates to: + /// **'Play'** + String get play; + + /// No description provided for @delete. + /// + /// In en, this message translates to: + /// **'Delete'** + String get delete; + + /// No description provided for @none. + /// + /// In en, this message translates to: + /// **'None'** + String get none; + + /// No description provided for @sort_a_z. + /// + /// In en, this message translates to: + /// **'Sort by A-Z'** + String get sort_a_z; + + /// No description provided for @sort_z_a. + /// + /// In en, this message translates to: + /// **'Sort by Z-A'** + String get sort_z_a; + + /// No description provided for @sort_artist. + /// + /// In en, this message translates to: + /// **'Sort by Artist'** + String get sort_artist; + + /// No description provided for @sort_album. + /// + /// In en, this message translates to: + /// **'Sort by Album'** + String get sort_album; + + /// No description provided for @sort_duration. + /// + /// In en, this message translates to: + /// **'Sort by Duration'** + String get sort_duration; + + /// No description provided for @sort_tracks. + /// + /// In en, this message translates to: + /// **'Sort Tracks'** + String get sort_tracks; + + /// No description provided for @currently_downloading. + /// + /// In en, this message translates to: + /// **'Currently Downloading ({tracks_length})'** + String currently_downloading(Object tracks_length); + + /// No description provided for @cancel_all. + /// + /// In en, this message translates to: + /// **'Cancel All'** + String get cancel_all; + + /// No description provided for @filter_artist. + /// + /// In en, this message translates to: + /// **'Filter artists...'** + String get filter_artist; + + /// No description provided for @followers. + /// + /// In en, this message translates to: + /// **'{followers} Followers'** + String followers(Object followers); + + /// No description provided for @add_artist_to_blacklist. + /// + /// In en, this message translates to: + /// **'Add artist to blacklist'** + String get add_artist_to_blacklist; + + /// No description provided for @top_tracks. + /// + /// In en, this message translates to: + /// **'Top Tracks'** + String get top_tracks; + + /// No description provided for @fans_also_like. + /// + /// In en, this message translates to: + /// **'Fans also like'** + String get fans_also_like; + + /// No description provided for @loading. + /// + /// In en, this message translates to: + /// **'Loading...'** + String get loading; + + /// No description provided for @artist. + /// + /// In en, this message translates to: + /// **'Artist'** + String get artist; + + /// No description provided for @blacklisted. + /// + /// In en, this message translates to: + /// **'Blacklisted'** + String get blacklisted; + + /// No description provided for @following. + /// + /// In en, this message translates to: + /// **'Following'** + String get following; + + /// No description provided for @follow. + /// + /// In en, this message translates to: + /// **'Follow'** + String get follow; + + /// No description provided for @artist_url_copied. + /// + /// In en, this message translates to: + /// **'Artist URL copied to clipboard'** + String get artist_url_copied; + + /// No description provided for @added_to_queue. + /// + /// In en, this message translates to: + /// **'Added {tracks} tracks to queue'** + String added_to_queue(Object tracks); + + /// No description provided for @filter_albums. + /// + /// In en, this message translates to: + /// **'Filter albums...'** + String get filter_albums; + + /// No description provided for @synced. + /// + /// In en, this message translates to: + /// **'Synced'** + String get synced; + + /// No description provided for @plain. + /// + /// In en, this message translates to: + /// **'Plain'** + String get plain; + + /// No description provided for @shuffle. + /// + /// In en, this message translates to: + /// **'Shuffle'** + String get shuffle; + + /// No description provided for @search_tracks. + /// + /// In en, this message translates to: + /// **'Search tracks...'** + String get search_tracks; + + /// No description provided for @released. + /// + /// In en, this message translates to: + /// **'Released'** + String get released; + + /// No description provided for @error. + /// + /// In en, this message translates to: + /// **'Error {error}'** + String error(Object error); + + /// No description provided for @title. + /// + /// In en, this message translates to: + /// **'Title'** + String get title; + + /// No description provided for @time. + /// + /// In en, this message translates to: + /// **'Time'** + String get time; + + /// No description provided for @more_actions. + /// + /// In en, this message translates to: + /// **'More actions'** + String get more_actions; + + /// No description provided for @download_count. + /// + /// In en, this message translates to: + /// **'Download ({count})'** + String download_count(Object count); + + /// No description provided for @add_count_to_playlist. + /// + /// In en, this message translates to: + /// **'Add ({count}) to Playlist'** + String add_count_to_playlist(Object count); + + /// No description provided for @add_count_to_queue. + /// + /// In en, this message translates to: + /// **'Add ({count}) to Queue'** + String add_count_to_queue(Object count); + + /// No description provided for @play_count_next. + /// + /// In en, this message translates to: + /// **'Play ({count}) next'** + String play_count_next(Object count); + + /// No description provided for @album. + /// + /// In en, this message translates to: + /// **'Album'** + String get album; + + /// No description provided for @copied_to_clipboard. + /// + /// In en, this message translates to: + /// **'Copied {data} to clipboard'** + String copied_to_clipboard(Object data); + + /// No description provided for @add_to_following_playlists. + /// + /// In en, this message translates to: + /// **'Add {track} to following Playlists'** + String add_to_following_playlists(Object track); + + /// No description provided for @add. + /// + /// In en, this message translates to: + /// **'Add'** + String get add; + + /// No description provided for @added_track_to_queue. + /// + /// In en, this message translates to: + /// **'Added {track} to queue'** + String added_track_to_queue(Object track); + + /// No description provided for @add_to_queue. + /// + /// In en, this message translates to: + /// **'Add to queue'** + String get add_to_queue; + + /// No description provided for @track_will_play_next. + /// + /// In en, this message translates to: + /// **'{track} will play next'** + String track_will_play_next(Object track); + + /// No description provided for @play_next. + /// + /// In en, this message translates to: + /// **'Play next'** + String get play_next; + + /// No description provided for @removed_track_from_queue. + /// + /// In en, this message translates to: + /// **'Removed {track} from queue'** + String removed_track_from_queue(Object track); + + /// No description provided for @remove_from_queue. + /// + /// In en, this message translates to: + /// **'Remove from queue'** + String get remove_from_queue; + + /// No description provided for @remove_from_favorites. + /// + /// In en, this message translates to: + /// **'Remove from favorites'** + String get remove_from_favorites; + + /// No description provided for @save_as_favorite. + /// + /// In en, this message translates to: + /// **'Save as favorite'** + String get save_as_favorite; + + /// No description provided for @add_to_playlist. + /// + /// In en, this message translates to: + /// **'Add to playlist'** + String get add_to_playlist; + + /// No description provided for @remove_from_playlist. + /// + /// In en, this message translates to: + /// **'Remove from playlist'** + String get remove_from_playlist; + + /// No description provided for @add_to_blacklist. + /// + /// In en, this message translates to: + /// **'Add to blacklist'** + String get add_to_blacklist; + + /// No description provided for @remove_from_blacklist. + /// + /// In en, this message translates to: + /// **'Remove from blacklist'** + String get remove_from_blacklist; + + /// No description provided for @share. + /// + /// In en, this message translates to: + /// **'Share'** + String get share; + + /// No description provided for @mini_player. + /// + /// In en, this message translates to: + /// **'Mini Player'** + String get mini_player; + + /// No description provided for @slide_to_seek. + /// + /// In en, this message translates to: + /// **'Slide to seek forward or backward'** + String get slide_to_seek; + + /// No description provided for @shuffle_playlist. + /// + /// In en, this message translates to: + /// **'Shuffle playlist'** + String get shuffle_playlist; + + /// No description provided for @unshuffle_playlist. + /// + /// In en, this message translates to: + /// **'Unshuffle playlist'** + String get unshuffle_playlist; + + /// No description provided for @previous_track. + /// + /// In en, this message translates to: + /// **'Previous track'** + String get previous_track; + + /// No description provided for @next_track. + /// + /// In en, this message translates to: + /// **'Next track'** + String get next_track; + + /// No description provided for @pause_playback. + /// + /// In en, this message translates to: + /// **'Pause Playback'** + String get pause_playback; + + /// No description provided for @resume_playback. + /// + /// In en, this message translates to: + /// **'Resume Playback'** + String get resume_playback; + + /// No description provided for @loop_track. + /// + /// In en, this message translates to: + /// **'Loop track'** + String get loop_track; + + /// No description provided for @no_loop. + /// + /// In en, this message translates to: + /// **'No loop'** + String get no_loop; + + /// No description provided for @repeat_playlist. + /// + /// In en, this message translates to: + /// **'Repeat playlist'** + String get repeat_playlist; + + /// No description provided for @queue. + /// + /// In en, this message translates to: + /// **'Queue'** + String get queue; + + /// No description provided for @alternative_track_sources. + /// + /// In en, this message translates to: + /// **'Alternative track sources'** + String get alternative_track_sources; + + /// No description provided for @download_track. + /// + /// In en, this message translates to: + /// **'Download track'** + String get download_track; + + /// No description provided for @tracks_in_queue. + /// + /// In en, this message translates to: + /// **'{tracks} tracks in queue'** + String tracks_in_queue(Object tracks); + + /// No description provided for @clear_all. + /// + /// In en, this message translates to: + /// **'Clear all'** + String get clear_all; + + /// No description provided for @show_hide_ui_on_hover. + /// + /// In en, this message translates to: + /// **'Show/Hide UI on hover'** + String get show_hide_ui_on_hover; + + /// No description provided for @always_on_top. + /// + /// In en, this message translates to: + /// **'Always on top'** + String get always_on_top; + + /// No description provided for @exit_mini_player. + /// + /// In en, this message translates to: + /// **'Exit Mini player'** + String get exit_mini_player; + + /// No description provided for @download_location. + /// + /// In en, this message translates to: + /// **'Download location'** + String get download_location; + + /// No description provided for @local_library. + /// + /// In en, this message translates to: + /// **'Local library'** + String get local_library; + + /// No description provided for @add_library_location. + /// + /// In en, this message translates to: + /// **'Add to library'** + String get add_library_location; + + /// No description provided for @remove_library_location. + /// + /// In en, this message translates to: + /// **'Remove from library'** + String get remove_library_location; + + /// No description provided for @account. + /// + /// In en, this message translates to: + /// **'Account'** + String get account; + + /// No description provided for @logout. + /// + /// In en, this message translates to: + /// **'Logout'** + String get logout; + + /// No description provided for @logout_of_this_account. + /// + /// In en, this message translates to: + /// **'Logout of this account'** + String get logout_of_this_account; + + /// No description provided for @language_region. + /// + /// In en, this message translates to: + /// **'Language & Region'** + String get language_region; + + /// No description provided for @language. + /// + /// In en, this message translates to: + /// **'Language'** + String get language; + + /// No description provided for @system_default. + /// + /// In en, this message translates to: + /// **'System Default'** + String get system_default; + + /// No description provided for @market_place_region. + /// + /// In en, this message translates to: + /// **'Marketplace Region'** + String get market_place_region; + + /// No description provided for @recommendation_country. + /// + /// In en, this message translates to: + /// **'Recommendation Country'** + String get recommendation_country; + + /// No description provided for @appearance. + /// + /// In en, this message translates to: + /// **'Appearance'** + String get appearance; + + /// No description provided for @layout_mode. + /// + /// In en, this message translates to: + /// **'Layout Mode'** + String get layout_mode; + + /// No description provided for @override_layout_settings. + /// + /// In en, this message translates to: + /// **'Override responsive layout mode settings'** + String get override_layout_settings; + + /// No description provided for @adaptive. + /// + /// In en, this message translates to: + /// **'Adaptive'** + String get adaptive; + + /// No description provided for @compact. + /// + /// In en, this message translates to: + /// **'Compact'** + String get compact; + + /// No description provided for @extended. + /// + /// In en, this message translates to: + /// **'Extended'** + String get extended; + + /// No description provided for @theme. + /// + /// In en, this message translates to: + /// **'Theme'** + String get theme; + + /// No description provided for @dark. + /// + /// In en, this message translates to: + /// **'Dark'** + String get dark; + + /// No description provided for @light. + /// + /// In en, this message translates to: + /// **'Light'** + String get light; + + /// No description provided for @system. + /// + /// In en, this message translates to: + /// **'System'** + String get system; + + /// No description provided for @accent_color. + /// + /// In en, this message translates to: + /// **'Accent Color'** + String get accent_color; + + /// No description provided for @sync_album_color. + /// + /// In en, this message translates to: + /// **'Sync album color'** + String get sync_album_color; + + /// No description provided for @sync_album_color_description. + /// + /// In en, this message translates to: + /// **'Uses the dominant color of the album art as the accent color'** + String get sync_album_color_description; + + /// No description provided for @playback. + /// + /// In en, this message translates to: + /// **'Playback'** + String get playback; + + /// No description provided for @audio_quality. + /// + /// In en, this message translates to: + /// **'Audio Quality'** + String get audio_quality; + + /// No description provided for @high. + /// + /// In en, this message translates to: + /// **'High'** + String get high; + + /// No description provided for @low. + /// + /// In en, this message translates to: + /// **'Low'** + String get low; + + /// No description provided for @pre_download_play. + /// + /// In en, this message translates to: + /// **'Pre-download and play'** + String get pre_download_play; + + /// No description provided for @pre_download_play_description. + /// + /// In en, this message translates to: + /// **'Instead of streaming audio, download bytes and play instead (Recommended for higher bandwidth users)'** + String get pre_download_play_description; + + /// No description provided for @skip_non_music. + /// + /// In en, this message translates to: + /// **'Skip non-music segments (SponsorBlock)'** + String get skip_non_music; + + /// No description provided for @blacklist_description. + /// + /// In en, this message translates to: + /// **'Blacklisted tracks and artists'** + String get blacklist_description; + + /// No description provided for @wait_for_download_to_finish. + /// + /// In en, this message translates to: + /// **'Please wait for the current download to finish'** + String get wait_for_download_to_finish; + + /// No description provided for @desktop. + /// + /// In en, this message translates to: + /// **'Desktop'** + String get desktop; + + /// No description provided for @close_behavior. + /// + /// In en, this message translates to: + /// **'Close Behavior'** + String get close_behavior; + + /// No description provided for @close. + /// + /// In en, this message translates to: + /// **'Close'** + String get close; + + /// No description provided for @minimize_to_tray. + /// + /// In en, this message translates to: + /// **'Minimize to tray'** + String get minimize_to_tray; + + /// No description provided for @show_tray_icon. + /// + /// In en, this message translates to: + /// **'Show System tray icon'** + String get show_tray_icon; + + /// No description provided for @about. + /// + /// In en, this message translates to: + /// **'About'** + String get about; + + /// No description provided for @u_love_spotube. + /// + /// In en, this message translates to: + /// **'We know you love Spotube'** + String get u_love_spotube; + + /// No description provided for @check_for_updates. + /// + /// In en, this message translates to: + /// **'Check for updates'** + String get check_for_updates; + + /// No description provided for @about_spotube. + /// + /// In en, this message translates to: + /// **'About Spotube'** + String get about_spotube; + + /// No description provided for @blacklist. + /// + /// In en, this message translates to: + /// **'Blacklist'** + String get blacklist; + + /// No description provided for @please_sponsor. + /// + /// In en, this message translates to: + /// **'Please Sponsor/Donate'** + String get please_sponsor; + + /// No description provided for @spotube_description. + /// + /// In en, this message translates to: + /// **'Open source extensible music streaming platform and app, based on BYOMM (Bring your own music metadata) concept'** + String get spotube_description; + + /// No description provided for @version. + /// + /// In en, this message translates to: + /// **'Version'** + String get version; + + /// No description provided for @build_number. + /// + /// In en, this message translates to: + /// **'Build Number'** + String get build_number; + + /// No description provided for @founder. + /// + /// In en, this message translates to: + /// **'Founder'** + String get founder; + + /// No description provided for @repository. + /// + /// In en, this message translates to: + /// **'Repository'** + String get repository; + + /// No description provided for @bug_issues. + /// + /// In en, this message translates to: + /// **'Bug+Issues'** + String get bug_issues; + + /// No description provided for @made_with. + /// + /// In en, this message translates to: + /// **'Made with ❤️ in Bangladesh🇧🇩'** + String get made_with; + + /// No description provided for @kingkor_roy_tirtho. + /// + /// In en, this message translates to: + /// **'Kingkor Roy Tirtho'** + String get kingkor_roy_tirtho; + + /// No description provided for @copyright. + /// + /// In en, this message translates to: + /// **'© 2021-{current_year} Kingkor Roy Tirtho'** + String copyright(Object current_year); + + /// No description provided for @license. + /// + /// In en, this message translates to: + /// **'License'** + String get license; + + /// No description provided for @credentials_will_not_be_shared_disclaimer. + /// + /// In en, this message translates to: + /// **'Don\'t worry, any of your credentials won\'t be collected or shared with anyone'** + String get credentials_will_not_be_shared_disclaimer; + + /// No description provided for @know_how_to_login. + /// + /// In en, this message translates to: + /// **'Don\'t know how to do this?'** + String get know_how_to_login; + + /// No description provided for @follow_step_by_step_guide. + /// + /// In en, this message translates to: + /// **'Follow along the Step by Step guide'** + String get follow_step_by_step_guide; + + /// No description provided for @cookie_name_cookie. + /// + /// In en, this message translates to: + /// **'{name} Cookie'** + String cookie_name_cookie(Object name); + + /// No description provided for @fill_in_all_fields. + /// + /// In en, this message translates to: + /// **'Please fill in all the fields'** + String get fill_in_all_fields; + + /// No description provided for @submit. + /// + /// In en, this message translates to: + /// **'Submit'** + String get submit; + + /// No description provided for @exit. + /// + /// In en, this message translates to: + /// **'Exit'** + String get exit; + + /// No description provided for @previous. + /// + /// In en, this message translates to: + /// **'Previous'** + String get previous; + + /// No description provided for @next. + /// + /// In en, this message translates to: + /// **'Next'** + String get next; + + /// No description provided for @done. + /// + /// In en, this message translates to: + /// **'Done'** + String get done; + + /// No description provided for @step_1. + /// + /// In en, this message translates to: + /// **'Step 1'** + String get step_1; + + /// No description provided for @first_go_to. + /// + /// In en, this message translates to: + /// **'First, Go to'** + String get first_go_to; + + /// No description provided for @something_went_wrong. + /// + /// In en, this message translates to: + /// **'Something went wrong'** + String get something_went_wrong; + + /// No description provided for @piped_instance. + /// + /// In en, this message translates to: + /// **'Piped Server Instance'** + String get piped_instance; + + /// No description provided for @piped_description. + /// + /// In en, this message translates to: + /// **'The Piped server instance to use for track matching'** + String get piped_description; + + /// No description provided for @piped_warning. + /// + /// In en, this message translates to: + /// **'Some of them might not work well. So use at your own risk'** + String get piped_warning; + + /// No description provided for @invidious_instance. + /// + /// In en, this message translates to: + /// **'Invidious Server Instance'** + String get invidious_instance; + + /// No description provided for @invidious_description. + /// + /// In en, this message translates to: + /// **'The Invidious server instance to use for track matching'** + String get invidious_description; + + /// No description provided for @invidious_warning. + /// + /// In en, this message translates to: + /// **'Some of them might not work well. So use at your own risk'** + String get invidious_warning; + + /// No description provided for @generate. + /// + /// In en, this message translates to: + /// **'Generate'** + String get generate; + + /// No description provided for @track_exists. + /// + /// In en, this message translates to: + /// **'Track {track} already exists'** + String track_exists(Object track); + + /// No description provided for @replace_downloaded_tracks. + /// + /// In en, this message translates to: + /// **'Replace all downloaded tracks'** + String get replace_downloaded_tracks; + + /// No description provided for @skip_download_tracks. + /// + /// In en, this message translates to: + /// **'Skip downloading all downloaded tracks'** + String get skip_download_tracks; + + /// No description provided for @do_you_want_to_replace. + /// + /// In en, this message translates to: + /// **'Do you want to replace the existing track??'** + String get do_you_want_to_replace; + + /// No description provided for @replace. + /// + /// In en, this message translates to: + /// **'Replace'** + String get replace; + + /// No description provided for @skip. + /// + /// In en, this message translates to: + /// **'Skip'** + String get skip; + + /// No description provided for @select_up_to_count_type. + /// + /// In en, this message translates to: + /// **'Select up to {count} {type}'** + String select_up_to_count_type(Object count, Object type); + + /// No description provided for @select_genres. + /// + /// In en, this message translates to: + /// **'Select Genres'** + String get select_genres; + + /// No description provided for @add_genres. + /// + /// In en, this message translates to: + /// **'Add Genres'** + String get add_genres; + + /// No description provided for @country. + /// + /// In en, this message translates to: + /// **'Country'** + String get country; + + /// No description provided for @number_of_tracks_generate. + /// + /// In en, this message translates to: + /// **'Number of tracks to generate'** + String get number_of_tracks_generate; + + /// No description provided for @acousticness. + /// + /// In en, this message translates to: + /// **'Acousticness'** + String get acousticness; + + /// No description provided for @danceability. + /// + /// In en, this message translates to: + /// **'Danceability'** + String get danceability; + + /// No description provided for @energy. + /// + /// In en, this message translates to: + /// **'Energy'** + String get energy; + + /// No description provided for @instrumentalness. + /// + /// In en, this message translates to: + /// **'Instrumentalness'** + String get instrumentalness; + + /// No description provided for @liveness. + /// + /// In en, this message translates to: + /// **'Liveness'** + String get liveness; + + /// No description provided for @loudness. + /// + /// In en, this message translates to: + /// **'Loudness'** + String get loudness; + + /// No description provided for @speechiness. + /// + /// In en, this message translates to: + /// **'Speechiness'** + String get speechiness; + + /// No description provided for @valence. + /// + /// In en, this message translates to: + /// **'Valence'** + String get valence; + + /// No description provided for @popularity. + /// + /// In en, this message translates to: + /// **'Popularity'** + String get popularity; + + /// No description provided for @key. + /// + /// In en, this message translates to: + /// **'Key'** + String get key; + + /// No description provided for @duration. + /// + /// In en, this message translates to: + /// **'Duration (s)'** + String get duration; + + /// No description provided for @tempo. + /// + /// In en, this message translates to: + /// **'Tempo (BPM)'** + String get tempo; + + /// No description provided for @mode. + /// + /// In en, this message translates to: + /// **'Mode'** + String get mode; + + /// No description provided for @time_signature. + /// + /// In en, this message translates to: + /// **'Time Signature'** + String get time_signature; + + /// No description provided for @short. + /// + /// In en, this message translates to: + /// **'Short'** + String get short; + + /// No description provided for @medium. + /// + /// In en, this message translates to: + /// **'Medium'** + String get medium; + + /// No description provided for @long. + /// + /// In en, this message translates to: + /// **'Long'** + String get long; + + /// No description provided for @min. + /// + /// In en, this message translates to: + /// **'Min'** + String get min; + + /// No description provided for @max. + /// + /// In en, this message translates to: + /// **'Max'** + String get max; + + /// No description provided for @target. + /// + /// In en, this message translates to: + /// **'Target'** + String get target; + + /// No description provided for @moderate. + /// + /// In en, this message translates to: + /// **'Moderate'** + String get moderate; + + /// No description provided for @deselect_all. + /// + /// In en, this message translates to: + /// **'Deselect All'** + String get deselect_all; + + /// No description provided for @select_all. + /// + /// In en, this message translates to: + /// **'Select All'** + String get select_all; + + /// No description provided for @are_you_sure. + /// + /// In en, this message translates to: + /// **'Are you sure?'** + String get are_you_sure; + + /// No description provided for @generating_playlist. + /// + /// In en, this message translates to: + /// **'Generating your custom playlist...'** + String get generating_playlist; + + /// No description provided for @selected_count_tracks. + /// + /// In en, this message translates to: + /// **'Selected {count} tracks'** + String selected_count_tracks(Object count); + + /// No description provided for @download_warning. + /// + /// In en, this message translates to: + /// **'If you download all Tracks at bulk you\'re clearly pirating Music & causing damage to the creative society of Music. I hope you are aware of this. Always, try respecting & supporting Artist\'s hard work'** + String get download_warning; + + /// No description provided for @download_ip_ban_warning. + /// + /// In en, this message translates to: + /// **'BTW, your IP can get blocked on YouTube due excessive download requests than usual. IP block means you can\'t use YouTube (even if you\'re logged in) for at least 2-3 months from that IP device. And Spotube doesn\'t hold any responsibility if this ever happens'** + String get download_ip_ban_warning; + + /// No description provided for @by_clicking_accept_terms. + /// + /// In en, this message translates to: + /// **'By clicking \'accept\' you agree to following terms:'** + String get by_clicking_accept_terms; + + /// No description provided for @download_agreement_1. + /// + /// In en, this message translates to: + /// **'I know I\'m pirating Music. I\'m bad'** + String get download_agreement_1; + + /// No description provided for @download_agreement_2. + /// + /// In en, this message translates to: + /// **'I\'ll support the Artist wherever I can and I\'m only doing this because I don\'t have money to buy their art'** + String get download_agreement_2; + + /// No description provided for @download_agreement_3. + /// + /// In en, this message translates to: + /// **'I\'m completely aware that my IP can get blocked on YouTube & I don\'t hold Spotube or his owners/contributors responsible for any accidents caused by my current action'** + String get download_agreement_3; + + /// No description provided for @decline. + /// + /// In en, this message translates to: + /// **'Decline'** + String get decline; + + /// No description provided for @accept. + /// + /// In en, this message translates to: + /// **'Accept'** + String get accept; + + /// No description provided for @details. + /// + /// In en, this message translates to: + /// **'Details'** + String get details; + + /// No description provided for @youtube. + /// + /// In en, this message translates to: + /// **'YouTube'** + String get youtube; + + /// No description provided for @channel. + /// + /// In en, this message translates to: + /// **'Channel'** + String get channel; + + /// No description provided for @likes. + /// + /// In en, this message translates to: + /// **'Likes'** + String get likes; + + /// No description provided for @dislikes. + /// + /// In en, this message translates to: + /// **'Dislikes'** + String get dislikes; + + /// No description provided for @views. + /// + /// In en, this message translates to: + /// **'Views'** + String get views; + + /// No description provided for @streamUrl. + /// + /// In en, this message translates to: + /// **'Stream URL'** + String get streamUrl; + + /// No description provided for @stop. + /// + /// In en, this message translates to: + /// **'Stop'** + String get stop; + + /// No description provided for @sort_newest. + /// + /// In en, this message translates to: + /// **'Sort by newest added'** + String get sort_newest; + + /// No description provided for @sort_oldest. + /// + /// In en, this message translates to: + /// **'Sort by oldest added'** + String get sort_oldest; + + /// No description provided for @sleep_timer. + /// + /// In en, this message translates to: + /// **'Sleep Timer'** + String get sleep_timer; + + /// No description provided for @mins. + /// + /// In en, this message translates to: + /// **'{minutes} Minutes'** + String mins(Object minutes); + + /// No description provided for @hours. + /// + /// In en, this message translates to: + /// **'{hours} Hours'** + String hours(Object hours); + + /// No description provided for @hour. + /// + /// In en, this message translates to: + /// **'{hours} Hour'** + String hour(Object hours); + + /// No description provided for @custom_hours. + /// + /// In en, this message translates to: + /// **'Custom Hours'** + String get custom_hours; + + /// No description provided for @logs. + /// + /// In en, this message translates to: + /// **'Logs'** + String get logs; + + /// No description provided for @developers. + /// + /// In en, this message translates to: + /// **'Developers'** + String get developers; + + /// No description provided for @not_logged_in. + /// + /// In en, this message translates to: + /// **'You\'re not logged in'** + String get not_logged_in; + + /// No description provided for @search_mode. + /// + /// In en, this message translates to: + /// **'Search Mode'** + String get search_mode; + + /// No description provided for @audio_source. + /// + /// In en, this message translates to: + /// **'Audio Source'** + String get audio_source; + + /// No description provided for @ok. + /// + /// In en, this message translates to: + /// **'Ok'** + String get ok; + + /// No description provided for @failed_to_encrypt. + /// + /// In en, this message translates to: + /// **'Failed to encrypt'** + String get failed_to_encrypt; + + /// No description provided for @encryption_failed_warning. + /// + /// In en, this message translates to: + /// **'Spotube uses encryption to securely store your data. But failed to do so. So it\'ll fallback to insecure storage\nIf you\'re using linux, please make sure you\'ve any secret-service (gnome-keyring, kde-wallet, keepassxc etc) installed'** + String get encryption_failed_warning; + + /// No description provided for @querying_info. + /// + /// In en, this message translates to: + /// **'Querying info...'** + String get querying_info; + + /// No description provided for @piped_api_down. + /// + /// In en, this message translates to: + /// **'Piped API is down'** + String get piped_api_down; + + /// No description provided for @piped_down_error_instructions. + /// + /// In en, this message translates to: + /// **'The Piped instance {pipedInstance} is currently down\n\nEither change the instance or change the \'API type\' to official YouTube API\n\nMake sure to restart the app after change'** + String piped_down_error_instructions(Object pipedInstance); + + /// No description provided for @you_are_offline. + /// + /// In en, this message translates to: + /// **'You are currently offline'** + String get you_are_offline; + + /// No description provided for @connection_restored. + /// + /// In en, this message translates to: + /// **'Your internet connection was restored'** + String get connection_restored; + + /// No description provided for @use_system_title_bar. + /// + /// In en, this message translates to: + /// **'Use system title bar'** + String get use_system_title_bar; + + /// No description provided for @crunching_results. + /// + /// In en, this message translates to: + /// **'Crunching results...'** + String get crunching_results; + + /// No description provided for @search_to_get_results. + /// + /// In en, this message translates to: + /// **'Search to get results'** + String get search_to_get_results; + + /// No description provided for @use_amoled_mode. + /// + /// In en, this message translates to: + /// **'Pitch black dark theme'** + String get use_amoled_mode; + + /// No description provided for @pitch_dark_theme. + /// + /// In en, this message translates to: + /// **'AMOLED Mode'** + String get pitch_dark_theme; + + /// No description provided for @normalize_audio. + /// + /// In en, this message translates to: + /// **'Normalize audio'** + String get normalize_audio; + + /// No description provided for @change_cover. + /// + /// In en, this message translates to: + /// **'Change cover'** + String get change_cover; + + /// No description provided for @add_cover. + /// + /// In en, this message translates to: + /// **'Add cover'** + String get add_cover; + + /// No description provided for @restore_defaults. + /// + /// In en, this message translates to: + /// **'Restore defaults'** + String get restore_defaults; + + /// No description provided for @download_music_format. + /// + /// In en, this message translates to: + /// **'Download music format'** + String get download_music_format; + + /// No description provided for @streaming_music_format. + /// + /// In en, this message translates to: + /// **'Streaming music format'** + String get streaming_music_format; + + /// No description provided for @download_music_quality. + /// + /// In en, this message translates to: + /// **'Download music quality'** + String get download_music_quality; + + /// No description provided for @streaming_music_quality. + /// + /// In en, this message translates to: + /// **'Streaming music quality'** + String get streaming_music_quality; + + /// No description provided for @login_with_lastfm. + /// + /// In en, this message translates to: + /// **'Login with Last.fm'** + String get login_with_lastfm; + + /// No description provided for @connect. + /// + /// In en, this message translates to: + /// **'Connect'** + String get connect; + + /// No description provided for @disconnect_lastfm. + /// + /// In en, this message translates to: + /// **'Disconnect Last.fm'** + String get disconnect_lastfm; + + /// No description provided for @disconnect. + /// + /// In en, this message translates to: + /// **'Disconnect'** + String get disconnect; + + /// No description provided for @username. + /// + /// In en, this message translates to: + /// **'Username'** + String get username; + + /// No description provided for @password. + /// + /// In en, this message translates to: + /// **'Password'** + String get password; + + /// No description provided for @login. + /// + /// In en, this message translates to: + /// **'Login'** + String get login; + + /// No description provided for @login_with_your_lastfm. + /// + /// In en, this message translates to: + /// **'Login with your Last.fm account'** + String get login_with_your_lastfm; + + /// No description provided for @scrobble_to_lastfm. + /// + /// In en, this message translates to: + /// **'Scrobble to Last.fm'** + String get scrobble_to_lastfm; + + /// No description provided for @go_to_album. + /// + /// In en, this message translates to: + /// **'Go to Album'** + String get go_to_album; + + /// No description provided for @discord_rich_presence. + /// + /// In en, this message translates to: + /// **'Discord Rich Presence'** + String get discord_rich_presence; + + /// No description provided for @browse_all. + /// + /// In en, this message translates to: + /// **'Browse All'** + String get browse_all; + + /// No description provided for @genres. + /// + /// In en, this message translates to: + /// **'Genres'** + String get genres; + + /// No description provided for @explore_genres. + /// + /// In en, this message translates to: + /// **'Explore Genres'** + String get explore_genres; + + /// No description provided for @friends. + /// + /// In en, this message translates to: + /// **'Friends'** + String get friends; + + /// No description provided for @no_lyrics_available. + /// + /// In en, this message translates to: + /// **'Sorry, unable find lyrics for this track'** + String get no_lyrics_available; + + /// No description provided for @start_a_radio. + /// + /// In en, this message translates to: + /// **'Start a Radio'** + String get start_a_radio; + + /// No description provided for @how_to_start_radio. + /// + /// In en, this message translates to: + /// **'How do you want to start the radio?'** + String get how_to_start_radio; + + /// No description provided for @replace_queue_question. + /// + /// In en, this message translates to: + /// **'Do you want to replace the current queue or append to it?'** + String get replace_queue_question; + + /// No description provided for @endless_playback. + /// + /// In en, this message translates to: + /// **'Endless Playback'** + String get endless_playback; + + /// No description provided for @delete_playlist. + /// + /// In en, this message translates to: + /// **'Delete Playlist'** + String get delete_playlist; + + /// No description provided for @delete_playlist_confirmation. + /// + /// In en, this message translates to: + /// **'Are you sure you want to delete this playlist?'** + String get delete_playlist_confirmation; + + /// No description provided for @local_tracks. + /// + /// In en, this message translates to: + /// **'Local Tracks'** + String get local_tracks; + + /// No description provided for @local_tab. + /// + /// In en, this message translates to: + /// **'Local'** + String get local_tab; + + /// No description provided for @song_link. + /// + /// In en, this message translates to: + /// **'Song Link'** + String get song_link; + + /// No description provided for @skip_this_nonsense. + /// + /// In en, this message translates to: + /// **'Skip this nonsense'** + String get skip_this_nonsense; + + /// No description provided for @freedom_of_music. + /// + /// In en, this message translates to: + /// **'“Freedom of Music”'** + String get freedom_of_music; + + /// No description provided for @freedom_of_music_palm. + /// + /// In en, this message translates to: + /// **'“Freedom of Music in the palm of your hand”'** + String get freedom_of_music_palm; + + /// No description provided for @get_started. + /// + /// In en, this message translates to: + /// **'Let\'s get started'** + String get get_started; + + /// No description provided for @youtube_source_description. + /// + /// In en, this message translates to: + /// **'Recommended and works best.'** + String get youtube_source_description; + + /// No description provided for @piped_source_description. + /// + /// In en, this message translates to: + /// **'Feeling free? Same as YouTube but a lot free.'** + String get piped_source_description; + + /// No description provided for @jiosaavn_source_description. + /// + /// In en, this message translates to: + /// **'Best for South Asian region.'** + String get jiosaavn_source_description; + + /// No description provided for @invidious_source_description. + /// + /// In en, this message translates to: + /// **'Similar to Piped but with higher availability.'** + String get invidious_source_description; + + /// No description provided for @highest_quality. + /// + /// In en, this message translates to: + /// **'Highest Quality: {quality}'** + String highest_quality(Object quality); + + /// No description provided for @select_audio_source. + /// + /// In en, this message translates to: + /// **'Select Audio Source'** + String get select_audio_source; + + /// No description provided for @endless_playback_description. + /// + /// In en, this message translates to: + /// **'Automatically append new songs\nto the end of the queue'** + String get endless_playback_description; + + /// No description provided for @choose_your_region. + /// + /// In en, this message translates to: + /// **'Choose your region'** + String get choose_your_region; + + /// No description provided for @choose_your_region_description. + /// + /// In en, this message translates to: + /// **'This will help Spotube show you the right content\nfor your location.'** + String get choose_your_region_description; + + /// No description provided for @choose_your_language. + /// + /// In en, this message translates to: + /// **'Choose your language'** + String get choose_your_language; + + /// No description provided for @help_project_grow. + /// + /// In en, this message translates to: + /// **'Help this project grow'** + String get help_project_grow; + + /// No description provided for @help_project_grow_description. + /// + /// In en, this message translates to: + /// **'Spotube is an open-source project. You can help this project grow by contributing to the project, reporting bugs, or suggesting new features.'** + String get help_project_grow_description; + + /// No description provided for @contribute_on_github. + /// + /// In en, this message translates to: + /// **'Contribute on GitHub'** + String get contribute_on_github; + + /// No description provided for @donate_on_open_collective. + /// + /// In en, this message translates to: + /// **'Donate on Open Collective'** + String get donate_on_open_collective; + + /// No description provided for @browse_anonymously. + /// + /// In en, this message translates to: + /// **'Browse Anonymously'** + String get browse_anonymously; + + /// No description provided for @enable_connect. + /// + /// In en, this message translates to: + /// **'Enable Connect'** + String get enable_connect; + + /// No description provided for @enable_connect_description. + /// + /// In en, this message translates to: + /// **'Control Spotube from other devices'** + String get enable_connect_description; + + /// No description provided for @devices. + /// + /// In en, this message translates to: + /// **'Devices'** + String get devices; + + /// No description provided for @select. + /// + /// In en, this message translates to: + /// **'Select'** + String get select; + + /// No description provided for @connect_client_alert. + /// + /// In en, this message translates to: + /// **'You\'re being controlled by {client}'** + String connect_client_alert(Object client); + + /// No description provided for @this_device. + /// + /// In en, this message translates to: + /// **'This Device'** + String get this_device; + + /// No description provided for @remote. + /// + /// In en, this message translates to: + /// **'Remote'** + String get remote; + + /// No description provided for @stats. + /// + /// In en, this message translates to: + /// **'Stats'** + String get stats; + + /// No description provided for @and_n_more. + /// + /// In en, this message translates to: + /// **'and {count} more'** + String and_n_more(Object count); + + /// No description provided for @recently_played. + /// + /// In en, this message translates to: + /// **'Recently Played'** + String get recently_played; + + /// No description provided for @browse_more. + /// + /// In en, this message translates to: + /// **'Browse More'** + String get browse_more; + + /// No description provided for @no_title. + /// + /// In en, this message translates to: + /// **'No Title'** + String get no_title; + + /// No description provided for @not_playing. + /// + /// In en, this message translates to: + /// **'Not playing'** + String get not_playing; + + /// No description provided for @epic_failure. + /// + /// In en, this message translates to: + /// **'Epic failure!'** + String get epic_failure; + + /// No description provided for @added_num_tracks_to_queue. + /// + /// In en, this message translates to: + /// **'Added {tracks_length} tracks to queue'** + String added_num_tracks_to_queue(Object tracks_length); + + /// No description provided for @spotube_has_an_update. + /// + /// In en, this message translates to: + /// **'Spotube has an update'** + String get spotube_has_an_update; + + /// No description provided for @download_now. + /// + /// In en, this message translates to: + /// **'Download Now'** + String get download_now; + + /// No description provided for @nightly_version. + /// + /// In en, this message translates to: + /// **'Spotube Nightly {nightlyBuildNum} has been released'** + String nightly_version(Object nightlyBuildNum); + + /// No description provided for @release_version. + /// + /// In en, this message translates to: + /// **'Spotube v{version} has been released'** + String release_version(Object version); + + /// No description provided for @read_the_latest. + /// + /// In en, this message translates to: + /// **'Read the latest '** + String get read_the_latest; + + /// No description provided for @release_notes. + /// + /// In en, this message translates to: + /// **'release notes'** + String get release_notes; + + /// No description provided for @pick_color_scheme. + /// + /// In en, this message translates to: + /// **'Pick color scheme'** + String get pick_color_scheme; + + /// No description provided for @save. + /// + /// In en, this message translates to: + /// **'Save'** + String get save; + + /// No description provided for @choose_the_device. + /// + /// In en, this message translates to: + /// **'Choose the device:'** + String get choose_the_device; + + /// No description provided for @multiple_device_connected. + /// + /// In en, this message translates to: + /// **'There are multiple device connected.\nChoose the device you want this action to take place'** + String get multiple_device_connected; + + /// No description provided for @nothing_found. + /// + /// In en, this message translates to: + /// **'Nothing found'** + String get nothing_found; + + /// No description provided for @the_box_is_empty. + /// + /// In en, this message translates to: + /// **'The box is empty'** + String get the_box_is_empty; + + /// No description provided for @top_artists. + /// + /// In en, this message translates to: + /// **'Top Artists'** + String get top_artists; + + /// No description provided for @top_albums. + /// + /// In en, this message translates to: + /// **'Top Albums'** + String get top_albums; + + /// No description provided for @this_week. + /// + /// In en, this message translates to: + /// **'This week'** + String get this_week; + + /// No description provided for @this_month. + /// + /// In en, this message translates to: + /// **'This month'** + String get this_month; + + /// No description provided for @last_6_months. + /// + /// In en, this message translates to: + /// **'Last 6 months'** + String get last_6_months; + + /// No description provided for @this_year. + /// + /// In en, this message translates to: + /// **'This year'** + String get this_year; + + /// No description provided for @last_2_years. + /// + /// In en, this message translates to: + /// **'Last 2 years'** + String get last_2_years; + + /// No description provided for @all_time. + /// + /// In en, this message translates to: + /// **'All time'** + String get all_time; + + /// No description provided for @powered_by_provider. + /// + /// In en, this message translates to: + /// **'Powered by {providerName}'** + String powered_by_provider(Object providerName); + + /// No description provided for @email. + /// + /// In en, this message translates to: + /// **'Email'** + String get email; + + /// No description provided for @profile_followers. + /// + /// In en, this message translates to: + /// **'Followers'** + String get profile_followers; + + /// No description provided for @birthday. + /// + /// In en, this message translates to: + /// **'Birthday'** + String get birthday; + + /// No description provided for @subscription. + /// + /// In en, this message translates to: + /// **'Subscription'** + String get subscription; + + /// No description provided for @not_born. + /// + /// In en, this message translates to: + /// **'Not born'** + String get not_born; + + /// No description provided for @hacker. + /// + /// In en, this message translates to: + /// **'Hacker'** + String get hacker; + + /// No description provided for @profile. + /// + /// In en, this message translates to: + /// **'Profile'** + String get profile; + + /// No description provided for @no_name. + /// + /// In en, this message translates to: + /// **'No Name'** + String get no_name; + + /// No description provided for @edit. + /// + /// In en, this message translates to: + /// **'Edit'** + String get edit; + + /// No description provided for @user_profile. + /// + /// In en, this message translates to: + /// **'User Profile'** + String get user_profile; + + /// No description provided for @count_plays. + /// + /// In en, this message translates to: + /// **'{count} plays'** + String count_plays(Object count); + + /// No description provided for @streaming_fees_hypothetical. + /// + /// In en, this message translates to: + /// **'Streaming fees (hypothetical)'** + String get streaming_fees_hypothetical; + + /// No description provided for @minutes_listened. + /// + /// In en, this message translates to: + /// **'Minutes listened'** + String get minutes_listened; + + /// No description provided for @streamed_songs. + /// + /// In en, this message translates to: + /// **'Streamed songs'** + String get streamed_songs; + + /// No description provided for @count_streams. + /// + /// In en, this message translates to: + /// **'{count} streams'** + String count_streams(Object count); + + /// No description provided for @owned_by_you. + /// + /// In en, this message translates to: + /// **'Owned by you'** + String get owned_by_you; + + /// No description provided for @copied_shareurl_to_clipboard. + /// + /// In en, this message translates to: + /// **'Copied {shareUrl} to clipboard'** + String copied_shareurl_to_clipboard(Object shareUrl); + + /// No description provided for @hipotetical_calculation. + /// + /// In en, this message translates to: + /// **'*This is calculated based on average online music streaming platform\'s per stream\npayout of \$0.003 to \$0.005. This is a hypothetical\ncalculation to give user insight about how much they\nwould have paid to the artists if they were to listen\ntheir song in different music streaming platform.'** + String get hipotetical_calculation; + + /// No description provided for @count_mins. + /// + /// In en, this message translates to: + /// **'{minutes} mins'** + String count_mins(Object minutes); + + /// No description provided for @summary_minutes. + /// + /// In en, this message translates to: + /// **'minutes'** + String get summary_minutes; + + /// No description provided for @summary_listened_to_music. + /// + /// In en, this message translates to: + /// **'Listened to music'** + String get summary_listened_to_music; + + /// No description provided for @summary_songs. + /// + /// In en, this message translates to: + /// **'songs'** + String get summary_songs; + + /// No description provided for @summary_streamed_overall. + /// + /// In en, this message translates to: + /// **'Streamed overall'** + String get summary_streamed_overall; + + /// No description provided for @summary_owed_to_artists. + /// + /// In en, this message translates to: + /// **'Owed to artists\nthis month'** + String get summary_owed_to_artists; + + /// No description provided for @summary_artists. + /// + /// In en, this message translates to: + /// **'artist\'s'** + String get summary_artists; + + /// No description provided for @summary_music_reached_you. + /// + /// In en, this message translates to: + /// **'Music reached you'** + String get summary_music_reached_you; + + /// No description provided for @summary_full_albums. + /// + /// In en, this message translates to: + /// **'full albums'** + String get summary_full_albums; + + /// No description provided for @summary_got_your_love. + /// + /// In en, this message translates to: + /// **'Got your love'** + String get summary_got_your_love; + + /// No description provided for @summary_playlists. + /// + /// In en, this message translates to: + /// **'playlists'** + String get summary_playlists; + + /// No description provided for @summary_were_on_repeat. + /// + /// In en, this message translates to: + /// **'Were on repeat'** + String get summary_were_on_repeat; + + /// No description provided for @total_money. + /// + /// In en, this message translates to: + /// **'Total {money}'** + String total_money(Object money); + + /// No description provided for @webview_not_found. + /// + /// In en, this message translates to: + /// **'Webview not found'** + String get webview_not_found; + + /// No description provided for @webview_not_found_description. + /// + /// In en, this message translates to: + /// **'No webview runtime is installed in your device.\nIf it\'s installed make sure it\'s in the Environment PATH\n\nAfter installing, restart the app'** + String get webview_not_found_description; + + /// No description provided for @unsupported_platform. + /// + /// In en, this message translates to: + /// **'Unsupported platform'** + String get unsupported_platform; + + /// No description provided for @cache_music. + /// + /// In en, this message translates to: + /// **'Cache music'** + String get cache_music; + + /// No description provided for @open. + /// + /// In en, this message translates to: + /// **'Open'** + String get open; + + /// No description provided for @cache_folder. + /// + /// In en, this message translates to: + /// **'Cache folder'** + String get cache_folder; + + /// No description provided for @export. + /// + /// In en, this message translates to: + /// **'Export'** + String get export; + + /// No description provided for @clear_cache. + /// + /// In en, this message translates to: + /// **'Clear cache'** + String get clear_cache; + + /// No description provided for @clear_cache_confirmation. + /// + /// In en, this message translates to: + /// **'Do you want to clear the cache?'** + String get clear_cache_confirmation; + + /// No description provided for @export_cache_files. + /// + /// In en, this message translates to: + /// **'Export Cached Files'** + String get export_cache_files; + + /// No description provided for @found_n_files. + /// + /// In en, this message translates to: + /// **'Found {count} files'** + String found_n_files(Object count); + + /// No description provided for @export_cache_confirmation. + /// + /// In en, this message translates to: + /// **'Do you want to export these files to'** + String get export_cache_confirmation; + + /// No description provided for @exported_n_out_of_m_files. + /// + /// In en, this message translates to: + /// **'Exported {filesExported} out of {files} files'** + String exported_n_out_of_m_files(Object files, Object filesExported); + + /// No description provided for @undo. + /// + /// In en, this message translates to: + /// **'Undo'** + String get undo; + + /// No description provided for @download_all. + /// + /// In en, this message translates to: + /// **'Download all'** + String get download_all; + + /// No description provided for @add_all_to_playlist. + /// + /// In en, this message translates to: + /// **'Add all to playlist'** + String get add_all_to_playlist; + + /// No description provided for @add_all_to_queue. + /// + /// In en, this message translates to: + /// **'Add all to queue'** + String get add_all_to_queue; + + /// No description provided for @play_all_next. + /// + /// In en, this message translates to: + /// **'Play all next'** + String get play_all_next; + + /// No description provided for @pause. + /// + /// In en, this message translates to: + /// **'Pause'** + String get pause; + + /// No description provided for @view_all. + /// + /// In en, this message translates to: + /// **'View all'** + String get view_all; + + /// No description provided for @no_tracks_added_yet. + /// + /// In en, this message translates to: + /// **'Looks like you haven\'t added any tracks yet'** + String get no_tracks_added_yet; + + /// No description provided for @no_tracks. + /// + /// In en, this message translates to: + /// **'Looks like there are no tracks here'** + String get no_tracks; + + /// No description provided for @no_tracks_listened_yet. + /// + /// In en, this message translates to: + /// **'Looks like you haven\'t listened to anything yet'** + String get no_tracks_listened_yet; + + /// No description provided for @not_following_artists. + /// + /// In en, this message translates to: + /// **'You\'re not following any artists'** + String get not_following_artists; + + /// No description provided for @no_favorite_albums_yet. + /// + /// In en, this message translates to: + /// **'Looks like you haven\'t added any albums to your favorites yet'** + String get no_favorite_albums_yet; + + /// No description provided for @no_logs_found. + /// + /// In en, this message translates to: + /// **'No logs found'** + String get no_logs_found; + + /// No description provided for @youtube_engine. + /// + /// In en, this message translates to: + /// **'YouTube Engine'** + String get youtube_engine; + + /// No description provided for @youtube_engine_not_installed_title. + /// + /// In en, this message translates to: + /// **'{engine} is not installed'** + String youtube_engine_not_installed_title(Object engine); + + /// No description provided for @youtube_engine_not_installed_message. + /// + /// In en, this message translates to: + /// **'{engine} is not installed in your system.'** + String youtube_engine_not_installed_message(Object engine); + + /// No description provided for @youtube_engine_set_path. + /// + /// In en, this message translates to: + /// **'Make sure it\'s available in the PATH variable or\nset the absolute path to the {engine} executable below'** + String youtube_engine_set_path(Object engine); + + /// No description provided for @youtube_engine_unix_issue_message. + /// + /// In en, this message translates to: + /// **'In macOS/Linux/unix like OS\'s, setting path on .zshrc/.bashrc/.bash_profile etc. won\'t work.\nYou need to set the path in the shell configuration file'** + String get youtube_engine_unix_issue_message; + + /// No description provided for @download. + /// + /// In en, this message translates to: + /// **'Download'** + String get download; + + /// No description provided for @file_not_found. + /// + /// In en, this message translates to: + /// **'File not found'** + String get file_not_found; + + /// No description provided for @custom. + /// + /// In en, this message translates to: + /// **'Custom'** + String get custom; + + /// No description provided for @add_custom_url. + /// + /// In en, this message translates to: + /// **'Add custom URL'** + String get add_custom_url; + + /// No description provided for @edit_port. + /// + /// In en, this message translates to: + /// **'Edit port'** + String get edit_port; + + /// No description provided for @port_helper_msg. + /// + /// In en, this message translates to: + /// **'Default is -1 which indicates random number. If you\'ve firewall configured, setting this is recommended.'** + String get port_helper_msg; + + /// No description provided for @connect_request. + /// + /// In en, this message translates to: + /// **'Allow {client} to connect?'** + String connect_request(Object client); + + /// No description provided for @connection_request_denied. + /// + /// In en, this message translates to: + /// **'Connection denied. User denied access.'** + String get connection_request_denied; + + /// No description provided for @an_error_occurred. + /// + /// In en, this message translates to: + /// **'An error occurred'** + String get an_error_occurred; + + /// No description provided for @copy_to_clipboard. + /// + /// In en, this message translates to: + /// **'Copy to clipboard'** + String get copy_to_clipboard; + + /// No description provided for @view_logs. + /// + /// In en, this message translates to: + /// **'View logs'** + String get view_logs; + + /// No description provided for @retry. + /// + /// In en, this message translates to: + /// **'Retry'** + String get retry; + + /// No description provided for @no_default_metadata_provider_selected. + /// + /// In en, this message translates to: + /// **'You\'ve no default metadata provider set'** + String get no_default_metadata_provider_selected; + + /// No description provided for @manage_metadata_providers. + /// + /// In en, this message translates to: + /// **'Manage metadata providers'** + String get manage_metadata_providers; + + /// No description provided for @open_link_in_browser. + /// + /// In en, this message translates to: + /// **'Open Link in Browser?'** + String get open_link_in_browser; + + /// No description provided for @do_you_want_to_open_the_following_link. + /// + /// In en, this message translates to: + /// **'Do you want to open the following link'** + String get do_you_want_to_open_the_following_link; + + /// No description provided for @unsafe_url_warning. + /// + /// In en, this message translates to: + /// **'It can be unsafe to open links from untrusted sources. Be cautious!\nYou can also copy the link to your clipboard.'** + String get unsafe_url_warning; + + /// No description provided for @copy_link. + /// + /// In en, this message translates to: + /// **'Copy Link'** + String get copy_link; + + /// No description provided for @building_your_timeline. + /// + /// In en, this message translates to: + /// **'Building your timeline based on your listenings...'** + String get building_your_timeline; + + /// No description provided for @official. + /// + /// In en, this message translates to: + /// **'Official'** + String get official; + + /// No description provided for @author_name. + /// + /// In en, this message translates to: + /// **'Author: {author}'** + String author_name(Object author); + + /// No description provided for @third_party. + /// + /// In en, this message translates to: + /// **'Third-party'** + String get third_party; + + /// No description provided for @plugin_requires_authentication. + /// + /// In en, this message translates to: + /// **'Plugin requires authentication'** + String get plugin_requires_authentication; + + /// No description provided for @update_available. + /// + /// In en, this message translates to: + /// **'Update available'** + String get update_available; + + /// No description provided for @supports_scrobbling. + /// + /// In en, this message translates to: + /// **'Supports scrobbling'** + String get supports_scrobbling; + + /// No description provided for @plugin_scrobbling_info. + /// + /// In en, this message translates to: + /// **'This plugin scrobbles your music to generate your listening history.'** + String get plugin_scrobbling_info; + + /// No description provided for @default_metadata_source. + /// + /// In en, this message translates to: + /// **'Default metadata source'** + String get default_metadata_source; + + /// No description provided for @set_default_metadata_source. + /// + /// In en, this message translates to: + /// **'Set default metadata source'** + String get set_default_metadata_source; + + /// No description provided for @default_audio_source. + /// + /// In en, this message translates to: + /// **'Default audio source'** + String get default_audio_source; + + /// No description provided for @set_default_audio_source. + /// + /// In en, this message translates to: + /// **'Set default audio source'** + String get set_default_audio_source; + + /// No description provided for @set_default. + /// + /// In en, this message translates to: + /// **'Set default'** + String get set_default; + + /// No description provided for @support. + /// + /// In en, this message translates to: + /// **'Support'** + String get support; + + /// No description provided for @support_plugin_development. + /// + /// In en, this message translates to: + /// **'Support plugin development'** + String get support_plugin_development; + + /// No description provided for @can_access_name_api. + /// + /// In en, this message translates to: + /// **'- Can access **{name}** API'** + String can_access_name_api(Object name); + + /// No description provided for @do_you_want_to_install_this_plugin. + /// + /// In en, this message translates to: + /// **'Do you want to install this plugin?'** + String get do_you_want_to_install_this_plugin; + + /// No description provided for @third_party_plugin_warning. + /// + /// In en, this message translates to: + /// **'This plugin is from a third-party repository. Please ensure you trust the source before installing.'** + String get third_party_plugin_warning; + + /// No description provided for @author. + /// + /// In en, this message translates to: + /// **'Author'** + String get author; + + /// No description provided for @this_plugin_can_do_following. + /// + /// In en, this message translates to: + /// **'This plugin can do following'** + String get this_plugin_can_do_following; + + /// No description provided for @install. + /// + /// In en, this message translates to: + /// **'Install'** + String get install; + + /// No description provided for @install_a_metadata_provider. + /// + /// In en, this message translates to: + /// **'Install a Metadata Provider'** + String get install_a_metadata_provider; + + /// No description provided for @no_tracks_playing. + /// + /// In en, this message translates to: + /// **'No Track being played currently'** + String get no_tracks_playing; + + /// No description provided for @synced_lyrics_not_available. + /// + /// In en, this message translates to: + /// **'Synced lyrics are not available for this song. Please use the'** + String get synced_lyrics_not_available; + + /// No description provided for @plain_lyrics. + /// + /// In en, this message translates to: + /// **'Plain Lyrics'** + String get plain_lyrics; + + /// No description provided for @tab_instead. + /// + /// In en, this message translates to: + /// **'tab instead.'** + String get tab_instead; + + /// No description provided for @disclaimer. + /// + /// In en, this message translates to: + /// **'Disclaimer'** + String get disclaimer; + + /// No description provided for @third_party_plugin_dmca_notice. + /// + /// In en, this message translates to: + /// **'The Spotube team does not hold any responsibility (including legal) for any \"Third-party\" plugins.\nPlease use them at your own risk. For any bugs/issues, please report them to the plugin repository.\n\nIf any \"Third-party\" plugin is breaking ToS/DMCA of any service/legal entity, please ask the \"Third-party\" plugin author or the hosting platform .e.g GitHub/Codeberg to take action. Above listed (\"Third-party\" labelled) are all public/community maintained plugins. We\'re not curating them, so we cannot take any action on them.\n\n'** + String get third_party_plugin_dmca_notice; + + /// No description provided for @input_does_not_match_format. + /// + /// In en, this message translates to: + /// **'Input doesn\'t match the required format'** + String get input_does_not_match_format; + + /// No description provided for @plugins. + /// + /// In en, this message translates to: + /// **'Plugins'** + String get plugins; + + /// No description provided for @paste_plugin_download_url. + /// + /// In en, this message translates to: + /// **'Paste download url or GitHub/Codeberg repo url or direct link to .smplug file'** + String get paste_plugin_download_url; + + /// No description provided for @download_and_install_plugin_from_url. + /// + /// In en, this message translates to: + /// **'Download and install plugin from url'** + String get download_and_install_plugin_from_url; + + /// No description provided for @failed_to_add_plugin_error. + /// + /// In en, this message translates to: + /// **'Failed to add plugin: {error}'** + String failed_to_add_plugin_error(Object error); + + /// No description provided for @upload_plugin_from_file. + /// + /// In en, this message translates to: + /// **'Upload plugin from file'** + String get upload_plugin_from_file; + + /// No description provided for @installed. + /// + /// In en, this message translates to: + /// **'Installed'** + String get installed; + + /// No description provided for @available_plugins. + /// + /// In en, this message translates to: + /// **'Available plugins'** + String get available_plugins; + + /// No description provided for @configure_plugins. + /// + /// In en, this message translates to: + /// **'Configure your own metadata provider and audio source plugins'** + String get configure_plugins; + + /// No description provided for @audio_scrobblers. + /// + /// In en, this message translates to: + /// **'Audio Scrobblers'** + String get audio_scrobblers; + + /// No description provided for @scrobbling. + /// + /// In en, this message translates to: + /// **'Scrobbling'** + String get scrobbling; + + /// No description provided for @source. + /// + /// In en, this message translates to: + /// **'Source: '** + String get source; + + /// No description provided for @uncompressed. + /// + /// In en, this message translates to: + /// **'Uncompressed'** + String get uncompressed; + + /// No description provided for @dab_music_source_description. + /// + /// In en, this message translates to: + /// **'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.'** + String get dab_music_source_description; +} + +class _AppLocalizationsDelegate + extends LocalizationsDelegate { + const _AppLocalizationsDelegate(); + + @override + Future load(Locale locale) { + return SynchronousFuture(lookupAppLocalizations(locale)); + } + + @override + bool isSupported(Locale locale) => [ + 'ar', + 'bn', + 'ca', + 'cs', + 'de', + 'en', + 'es', + 'eu', + 'fa', + 'fi', + 'fr', + 'hi', + 'id', + 'it', + 'ja', + 'ka', + 'ko', + 'ne', + 'nl', + 'pl', + 'pt', + 'ru', + 'ta', + 'th', + 'tl', + 'tr', + 'uk', + 'vi', + 'zh' + ].contains(locale.languageCode); + + @override + bool shouldReload(_AppLocalizationsDelegate old) => false; +} + +AppLocalizations lookupAppLocalizations(Locale locale) { + // Lookup logic when language+country codes are specified. + switch (locale.languageCode) { + case 'zh': + { + switch (locale.countryCode) { + case 'TW': + return AppLocalizationsZhTw(); + } + break; + } + } + + // Lookup logic when only language code is specified. + switch (locale.languageCode) { + case 'ar': + return AppLocalizationsAr(); + case 'bn': + return AppLocalizationsBn(); + case 'ca': + return AppLocalizationsCa(); + case 'cs': + return AppLocalizationsCs(); + case 'de': + return AppLocalizationsDe(); + case 'en': + return AppLocalizationsEn(); + case 'es': + return AppLocalizationsEs(); + case 'eu': + return AppLocalizationsEu(); + case 'fa': + return AppLocalizationsFa(); + case 'fi': + return AppLocalizationsFi(); + case 'fr': + return AppLocalizationsFr(); + case 'hi': + return AppLocalizationsHi(); + case 'id': + return AppLocalizationsId(); + case 'it': + return AppLocalizationsIt(); + case 'ja': + return AppLocalizationsJa(); + case 'ka': + return AppLocalizationsKa(); + case 'ko': + return AppLocalizationsKo(); + case 'ne': + return AppLocalizationsNe(); + case 'nl': + return AppLocalizationsNl(); + case 'pl': + return AppLocalizationsPl(); + case 'pt': + return AppLocalizationsPt(); + case 'ru': + return AppLocalizationsRu(); + case 'ta': + return AppLocalizationsTa(); + case 'th': + return AppLocalizationsTh(); + case 'tl': + return AppLocalizationsTl(); + case 'tr': + return AppLocalizationsTr(); + case 'uk': + return AppLocalizationsUk(); + case 'vi': + return AppLocalizationsVi(); + case 'zh': + return AppLocalizationsZh(); + } + + throw FlutterError( + 'AppLocalizations.delegate failed to load unsupported locale "$locale". This is likely ' + 'an issue with the localizations generation tool. Please file an issue ' + 'on GitHub with a reproducible sample app and the gen-l10n configuration ' + 'that was used.'); +} diff --git a/lib/l10n/generated/app_localizations_ar.dart b/lib/l10n/generated/app_localizations_ar.dart new file mode 100644 index 00000000..8fd50ffa --- /dev/null +++ b/lib/l10n/generated/app_localizations_ar.dart @@ -0,0 +1,1566 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Arabic (`ar`). +class AppLocalizationsAr extends AppLocalizations { + AppLocalizationsAr([String locale = 'ar']) : super(locale); + + @override + String get guest => 'ضيف'; + + @override + String get browse => 'تصفح'; + + @override + String get search => 'بحث'; + + @override + String get library => 'مكتبة'; + + @override + String get lyrics => 'كلمات'; + + @override + String get settings => 'إعدادات'; + + @override + String get genre_categories_filter => 'تصفية الفئات أو الأنواع...'; + + @override + String get genre => 'النوع'; + + @override + String get personalized => 'شخصية'; + + @override + String get featured => 'متميز'; + + @override + String get new_releases => 'الإصدارات الجديدة'; + + @override + String get songs => 'أغاني'; + + @override + String playing_track(Object track) { + return 'تشغيل $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'سيؤدي هذا إلى مسح قائمة الانتظار الحالية. $track_length ستتم إزالة المقطوعات\nهل تريد الإستمرار؟'; + } + + @override + String get load_more => 'تحميل المزيد'; + + @override + String get playlists => 'قوائم التشغيل'; + + @override + String get artists => 'فنانون'; + + @override + String get albums => 'ألبومات'; + + @override + String get tracks => 'مقطوعات'; + + @override + String get downloads => 'تنزيلات'; + + @override + String get filter_playlists => 'تصفية قوائم التشغيل الخاصة بك...'; + + @override + String get liked_tracks => 'المقطوعات التي أعجبتك'; + + @override + String get liked_tracks_description => 'جميع المقطوعات التي أعجبتك'; + + @override + String get playlist => 'قائمة التشغيل'; + + @override + String get create_a_playlist => 'إنشاء قائمة تشغيل'; + + @override + String get update_playlist => 'تحديث قائمة التشغيل'; + + @override + String get create => 'إنشاء'; + + @override + String get cancel => 'إلغاء'; + + @override + String get update => 'تحديث'; + + @override + String get playlist_name => 'اسم قائمة التشغيل'; + + @override + String get name_of_playlist => 'اسم قائمة التشغيل'; + + @override + String get description => 'وصف'; + + @override + String get public => 'عام'; + + @override + String get collaborative => 'تعاوني'; + + @override + String get search_local_tracks => 'بحث عن مقطوعات محلية'; + + @override + String get play => 'تشغيل'; + + @override + String get delete => 'حذف'; + + @override + String get none => 'لا شيء'; + + @override + String get sort_a_z => 'الترتيب من A-Z'; + + @override + String get sort_z_a => 'الترتيب من Z-A'; + + @override + String get sort_artist => 'الترتيب حسب الفنان'; + + @override + String get sort_album => 'فرز حسب الألبوم'; + + @override + String get sort_duration => 'ترتيب حسب المدة'; + + @override + String get sort_tracks => 'ترتيب المقطوعات'; + + @override + String currently_downloading(Object tracks_length) { + return 'يتم التنزيل ($tracks_length)'; + } + + @override + String get cancel_all => 'إلغاء الكل'; + + @override + String get filter_artist => 'تصفية الفنانين...'; + + @override + String followers(Object followers) { + return '$followers متابعون'; + } + + @override + String get add_artist_to_blacklist => 'إضافة فنان إلى القائمة السوداء'; + + @override + String get top_tracks => 'أهم المقطوعات الصوتية'; + + @override + String get fans_also_like => 'المعجبون يحبون أيضاً'; + + @override + String get loading => 'جارٍ التحميل'; + + @override + String get artist => 'فنان'; + + @override + String get blacklisted => 'في القائمة السوداء'; + + @override + String get following => 'يتابع'; + + @override + String get follow => 'تابع'; + + @override + String get artist_url_copied => 'تم نسخ عنوان URL للفنان إلى الحافظة'; + + @override + String added_to_queue(Object tracks) { + return 'تم إضافة المقطوعات إلى قائمة الإنتظار $tracks'; + } + + @override + String get filter_albums => 'تصفية الألبومات...'; + + @override + String get synced => 'تم المزامنة'; + + @override + String get plain => 'سهل'; + + @override + String get shuffle => 'خلط'; + + @override + String get search_tracks => 'يحث عن مقطوعات'; + + @override + String get released => 'تم الإصدار'; + + @override + String error(Object error) { + return 'خطأ $error'; + } + + @override + String get title => 'عنوان'; + + @override + String get time => 'وقت'; + + @override + String get more_actions => 'المزيد من الإجراءات'; + + @override + String download_count(Object count) { + return 'تنزيل ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'إضافة ($count) إلى قائمة التشغيل'; + } + + @override + String add_count_to_queue(Object count) { + return 'إضافة ($count) إلى قائمة الإنتظار'; + } + + @override + String play_count_next(Object count) { + return 'تشغيل ($count) التالي'; + } + + @override + String get album => 'ألبوم'; + + @override + String copied_to_clipboard(Object data) { + return 'تم النسخ $data إلى الحافظة'; + } + + @override + String add_to_following_playlists(Object track) { + return 'إضافة $track إلى قوائم التشغيل التالية'; + } + + @override + String get add => 'إضافة'; + + @override + String added_track_to_queue(Object track) { + return 'تم الإضافة $track إلى قائمة الإنتظار'; + } + + @override + String get add_to_queue => 'إضافة إلى قائمة التشغيل'; + + @override + String track_will_play_next(Object track) { + return '$track سيتم تشغيل التالي'; + } + + @override + String get play_next => 'تشغيل التالي'; + + @override + String removed_track_from_queue(Object track) { + return 'تم الإزالة $track من قائمة الإنتظار'; + } + + @override + String get remove_from_queue => 'إزالة من قائمة الإنتظار'; + + @override + String get remove_from_favorites => 'إزالة من المفضلة'; + + @override + String get save_as_favorite => 'حفظ كمفضل'; + + @override + String get add_to_playlist => 'إضافة إلى قائمة التشغيل'; + + @override + String get remove_from_playlist => 'إزالة من قائمة التشغيل'; + + @override + String get add_to_blacklist => 'إضافة إلى القائمة السوداء'; + + @override + String get remove_from_blacklist => 'إزالة من القائمة السوداء'; + + @override + String get share => 'مشاكرة'; + + @override + String get mini_player => 'مشغل مصغر'; + + @override + String get slide_to_seek => 'قم بالتمرير للبحث للأمام أو للخلف'; + + @override + String get shuffle_playlist => 'قائمة تشغيل عشوائية'; + + @override + String get unshuffle_playlist => 'إلغاء ترتيب قائمة التشغيل'; + + @override + String get previous_track => 'المقطوعة السابقة'; + + @override + String get next_track => 'مقطوعة جديدة'; + + @override + String get pause_playback => 'إيقاف التشغيل مؤقتًا'; + + @override + String get resume_playback => 'استئناف التشغيل'; + + @override + String get loop_track => 'تشغيل المقطوعة بشكل لا نهائي'; + + @override + String get no_loop => 'بدون تكرار'; + + @override + String get repeat_playlist => 'تكرار قائمة التشغيل'; + + @override + String get queue => 'قائمة الإنتظار'; + + @override + String get alternative_track_sources => 'مصادر مقطوعات بديلة'; + + @override + String get download_track => 'تنزيل المقطوعة'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks المقطوعات في قائمة الإنتظار'; + } + + @override + String get clear_all => 'مسح الكل'; + + @override + String get show_hide_ui_on_hover => 'إظهار/إخفاء واجهة المستخدم عند التمرير'; + + @override + String get always_on_top => 'دائما في القمة'; + + @override + String get exit_mini_player => 'خروج من المشغل المصغر'; + + @override + String get download_location => 'تنزيل الموقع'; + + @override + String get local_library => 'المكتبة المحلية'; + + @override + String get add_library_location => 'أضف إلى المكتبة'; + + @override + String get remove_library_location => 'إزالة من المكتبة'; + + @override + String get account => 'حساب'; + + @override + String get logout => 'تسجيل الخروج'; + + @override + String get logout_of_this_account => 'تسجيل الخروج من هذا الحساب'; + + @override + String get language_region => 'اللغة والمنطقة'; + + @override + String get language => 'لغة'; + + @override + String get system_default => 'لغة النظام الإفتراضية'; + + @override + String get market_place_region => 'منطقة السوق'; + + @override + String get recommendation_country => 'بلد التوصية'; + + @override + String get appearance => 'مظهر'; + + @override + String get layout_mode => 'وضع التخطيط'; + + @override + String get override_layout_settings => + 'تجاوز إعدادات وضع التخطيط سريع الاستجابة'; + + @override + String get adaptive => 'متكيف'; + + @override + String get compact => 'مدمج'; + + @override + String get extended => 'ممتد'; + + @override + String get theme => 'مظهر'; + + @override + String get dark => 'داكن'; + + @override + String get light => 'ساطعt'; + + @override + String get system => 'حسب النظام'; + + @override + String get accent_color => 'لون تمييز'; + + @override + String get sync_album_color => 'مزامنة لون الألبوم'; + + @override + String get sync_album_color_description => + 'يستخدم اللون السائد لصورة الألبوم باعتباره لون التمييز'; + + @override + String get playback => 'التشغيل'; + + @override + String get audio_quality => 'جودة الصوت'; + + @override + String get high => 'مرتفعة'; + + @override + String get low => 'منخفضة'; + + @override + String get pre_download_play => 'التحميل المسبق والتشغيل'; + + @override + String get pre_download_play_description => + 'بدلاً من دفق الصوت، قم بتنزيل وحدات البايت وتشغيلها بدلاً من ذلك (موصى به لمستخدمي Bandwidth)'; + + @override + String get skip_non_music => 'تخطي المقاطع غير الموسيقية (SponsorBlock)'; + + @override + String get blacklist_description => + 'المقطوعات والفنانون المدرجون في القائمة السوداء'; + + @override + String get wait_for_download_to_finish => + 'يرجى الانتظار حتى انتهاء التنزيل الحالي'; + + @override + String get desktop => 'سطح المكتب'; + + @override + String get close_behavior => 'إغلاق التصرف'; + + @override + String get close => 'إغلاق'; + + @override + String get minimize_to_tray => 'تصغير إلى الدرج'; + + @override + String get show_tray_icon => 'إظهار أيقونات درج النظام'; + + @override + String get about => 'حول'; + + @override + String get u_love_spotube => 'نحن نعلم أنك تحب Spotube'; + + @override + String get check_for_updates => 'تحقق من وجود تحديثات'; + + @override + String get about_spotube => 'حول Spotube'; + + @override + String get blacklist => 'قائمة سوداء'; + + @override + String get please_sponsor => 'يرجى دعم/التبرع'; + + @override + String get spotube_description => + 'Spotube، عميل Spotify خفيف الوزن ومتعدد المنصات ومجاني للجميع'; + + @override + String get version => 'إصدار'; + + @override + String get build_number => 'رقم البنية'; + + @override + String get founder => 'الموئسس'; + + @override + String get repository => 'المستودع'; + + @override + String get bug_issues => 'أخطاء+مشاكل'; + + @override + String get made_with => 'صُنع باستخدام ❤️ في بنغلاديش🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'الترخيص'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'لا تقلق، لن يتم جمع أي من بيانات الخاصة بك أو مشاركتها مع أي شخص'; + + @override + String get know_how_to_login => 'لا تعرف كيف تفعل هذا؟'; + + @override + String get follow_step_by_step_guide => 'اتبع الدليل خطوة بخطوة'; + + @override + String cookie_name_cookie(Object name) { + return '$name كوكيز'; + } + + @override + String get fill_in_all_fields => 'يرجى تعبئة جميع الحقول'; + + @override + String get submit => 'إرسال'; + + @override + String get exit => 'خروج'; + + @override + String get previous => 'السابق'; + + @override + String get next => 'التالي'; + + @override + String get done => 'تم'; + + @override + String get step_1 => 'الخطوة 1'; + + @override + String get first_go_to => 'أولا، اذهب إلى'; + + @override + String get something_went_wrong => 'هناك خطأ ما'; + + @override + String get piped_instance => 'مثيل خادم Piped'; + + @override + String get piped_description => + 'مثيل خادم Piped الذي سيتم استخدامه لمطابقة المقطوعة'; + + @override + String get piped_warning => + 'البعض منهم قد لا يعمل بشكل جيد. لذلك استخدمه على مسؤوليتك'; + + @override + String get invidious_instance => 'مثيل خادم Invidious'; + + @override + String get invidious_description => + 'مثيل خادم Invidious المستخدم لمطابقة المسارات'; + + @override + String get invidious_warning => + 'قد لا تعمل بعض الخوادم بشكل جيد. استخدمها على مسؤوليتك الخاصة'; + + @override + String get generate => 'إنشاء'; + + @override + String track_exists(Object track) { + return 'المقطوعة $track بالفعل موجودة'; + } + + @override + String get replace_downloaded_tracks => + 'استبدل جميع المقطوعات التي تم تنزيلها'; + + @override + String get skip_download_tracks => + 'تخطي تنزيل كافة المقطوعات التي تم تنزيلها'; + + @override + String get do_you_want_to_replace => 'هل تريد استبدال المقطوعة الحالية؟'; + + @override + String get replace => 'إستبدال'; + + @override + String get skip => 'تخطي'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'إختر ما يصل إلى $count $type'; + } + + @override + String get select_genres => 'حدد الأنواع'; + + @override + String get add_genres => 'أضف الأنواع'; + + @override + String get country => 'دولة'; + + @override + String get number_of_tracks_generate => + 'عدد المسارات المقطوعات المراد توليدها'; + + @override + String get acousticness => 'صوتية'; + + @override + String get danceability => 'قدرة على الرقص'; + + @override + String get energy => 'طاقة'; + + @override + String get instrumentalness => 'نفعية'; + + @override + String get liveness => 'حيوية'; + + @override + String get loudness => 'بريق'; + + @override + String get speechiness => 'كلام'; + + @override + String get valence => 'تكافؤ'; + + @override + String get popularity => 'شعبية'; + + @override + String get key => 'مفتاح'; + + @override + String get duration => 'مدة (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'توقيع الوقت'; + + @override + String get short => 'قصير'; + + @override + String get medium => 'متوسط'; + + @override + String get long => 'طويل'; + + @override + String get min => 'أدنى'; + + @override + String get max => 'أقصى'; + + @override + String get target => 'هدف'; + + @override + String get moderate => 'معتدل'; + + @override + String get deselect_all => 'الغاء تحديد الكل'; + + @override + String get select_all => 'اختر الكل'; + + @override + String get are_you_sure => 'هل أنت متأكد؟'; + + @override + String get generating_playlist => 'جارٍ إنشاء قائمة التشغيل المخصصة...'; + + @override + String selected_count_tracks(Object count) { + return 'مقطوعات $count مختارة'; + } + + @override + String get download_warning => + 'إذا قمت بتنزيل جميع المقاطع الصوتية بكميات كبيرة، فمن الواضح أنك تقوم بقرصنة الموسيقى وتسبب الضرر للمجتمع الإبداعي للموسيقى. أتمنى أن تكون على علم بهذا. حاول دائمًا احترام ودعم العمل الجاد للفنان'; + + @override + String get download_ip_ban_warning => + 'بالمناسبة، يمكن أن يتم حظر عنوان IP الخاص بك على YouTube بسبب طلبات التنزيل الزائدة عن المعتاد. يعني حظر IP أنه لا يمكنك استخدام YouTube (حتى إذا قمت بتسجيل الدخول) لمدة تتراوح بين شهرين إلى ثلاثة أشهر على الأقل من جهاز IP هذا. ولا يتحمل Spotube أي مسؤولية إذا حدث هذا على الإطلاق'; + + @override + String get by_clicking_accept_terms => + 'بالنقر على \"قبول\"، فإنك توافق على الشروط التالية:'; + + @override + String get download_agreement_1 => 'أعلم أنني أقوم بقرصنة الموسيقى. انا سيئ'; + + @override + String get download_agreement_2 => + 'سأدعم الفنان أينما أستطيع، وأنا أفعل هذا فقط لأنني لا أملك المال لشراء أعمالهم الفنية'; + + @override + String get download_agreement_3 => + 'أدرك تمامًا أنه يمكن حظر عنوان IP الخاص بي على YouTube ولا أحمل Spotube أو مالكيه/مساهميه المسؤولية عن أي حوادث ناجمة عن الإجراء الحالي الخاص بي'; + + @override + String get decline => 'رفض'; + + @override + String get accept => 'قبول'; + + @override + String get details => 'تفاصيل'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'قناة'; + + @override + String get likes => 'إعجابات'; + + @override + String get dislikes => 'عدم الإعجابات'; + + @override + String get views => 'مشاهدات'; + + @override + String get streamUrl => 'عنوان URL البث'; + + @override + String get stop => 'إيقاف'; + + @override + String get sort_newest => 'الترتيب حسب الأقدم'; + + @override + String get sort_oldest => 'الترتيب حسب الأقدم'; + + @override + String get sleep_timer => 'مؤقت النوم'; + + @override + String mins(Object minutes) { + return '$minutes دقائق'; + } + + @override + String hours(Object hours) { + return '$hours ساعات'; + } + + @override + String hour(Object hours) { + return '$hours ساعة'; + } + + @override + String get custom_hours => 'ساعات مخصصة'; + + @override + String get logs => 'سجلات'; + + @override + String get developers => 'المطورون'; + + @override + String get not_logged_in => 'لم تقم بتسجيل الدخول'; + + @override + String get search_mode => 'وضع البحث'; + + @override + String get audio_source => 'مصدر الصوت'; + + @override + String get ok => 'حسسناً'; + + @override + String get failed_to_encrypt => 'فشل في التشفير'; + + @override + String get encryption_failed_warning => + 'يستخدم Spotube التشفير لتخزين بياناتك بشكل آمن. لكنها فشلت في القيام بذلك. لذلك سيعود الأمر إلى التخزين غير الآمن\nإذا كنت تستخدم Linux، فيرجى التأكد من تثبيت أي خدمة سرية (gnome-keyring، kde-wallet، keepassxc، إلخ)'; + + @override + String get querying_info => 'جارٍ الاستعلام عن معلومات...'; + + @override + String get piped_api_down => 'Piped API معطلة'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'المثيل الموجه $pipedInstance معطل حاليًا\n\nيمكنك إما تغيير المثيل أو تغيير \'نوع API\' إلى YouTube API الرسمي\n\nتأكد من إعادة تشغيل التطبيق بعد التغيير'; + } + + @override + String get you_are_offline => 'أنت غير متصل حالياً'; + + @override + String get connection_restored => 'تمت استعادة اتصالك بالإنترنت'; + + @override + String get use_system_title_bar => 'استخدم شريط عنوان النظام'; + + @override + String get crunching_results => 'تدمير النتائج'; + + @override + String get search_to_get_results => 'إبحث للحصول على النتائج'; + + @override + String get use_amoled_mode => 'استخدم وضع AMOLED'; + + @override + String get pitch_dark_theme => 'موضوع دارت الأسود الفحمي'; + + @override + String get normalize_audio => 'تطبيع الصوت'; + + @override + String get change_cover => 'تغيير الغلاف'; + + @override + String get add_cover => 'إضافة غلاف'; + + @override + String get restore_defaults => 'استعادة الإعدادات الافتراضية'; + + @override + String get download_music_format => 'تنسيق تنزيل الموسيقى'; + + @override + String get streaming_music_format => 'تنسيق بث الموسيقى'; + + @override + String get download_music_quality => 'جودة تنزيل الموسيقى'; + + @override + String get streaming_music_quality => 'جودة بث الموسيقى'; + + @override + String get login_with_lastfm => 'تسجيل الدخول باستخدام Last.fm'; + + @override + String get connect => 'اتصال'; + + @override + String get disconnect_lastfm => 'قطع الاتصال بـ Last.fm'; + + @override + String get disconnect => 'قطع الاتصال'; + + @override + String get username => 'اسم المستخدم'; + + @override + String get password => 'كلمة المرور'; + + @override + String get login => 'تسجيل الدخول'; + + @override + String get login_with_your_lastfm => + 'تسجيل الدخول باستخدام حساب Last.fm الخاص بك'; + + @override + String get scrobble_to_lastfm => 'تسجيل الاستماع على Last.fm'; + + @override + String get go_to_album => 'الانتقال إلى الألبوم'; + + @override + String get discord_rich_presence => 'وجود ديسكورد الغني'; + + @override + String get browse_all => 'تصفح الكل'; + + @override + String get genres => 'الأنواع الموسيقية'; + + @override + String get explore_genres => 'استكشاف الأنواع'; + + @override + String get friends => 'أصدقاء'; + + @override + String get no_lyrics_available => + 'عذرًا، تعذر العثور على كلمات الأغنية لهذه العنصر'; + + @override + String get start_a_radio => 'بدء راديو'; + + @override + String get how_to_start_radio => 'كيف تريد بدء الراديو؟'; + + @override + String get replace_queue_question => + 'هل تريد استبدال قائمة التشغيل الحالية أم إضافة إليها؟'; + + @override + String get endless_playback => 'تشغيل بلا نهاية'; + + @override + String get delete_playlist => 'حذف قائمة التشغيل'; + + @override + String get delete_playlist_confirmation => + 'هل أنت متأكد أنك تريد حذف هذه قائمة التشغيل؟'; + + @override + String get local_tracks => 'المسارات المحلية'; + + @override + String get local_tab => 'محلي'; + + @override + String get song_link => 'رابط الأغنية'; + + @override + String get skip_this_nonsense => 'تخطي هذه الهراء'; + + @override + String get freedom_of_music => '“حرية الموسيقى”'; + + @override + String get freedom_of_music_palm => '“حرية الموسيقى في متناول يدك”'; + + @override + String get get_started => 'لنبدأ'; + + @override + String get youtube_source_description => 'موصى به ويعمل بشكل أفضل.'; + + @override + String get piped_source_description => + 'تشعر بالحرية؟ نفس يوتيوب ولكن أكثر حرية.'; + + @override + String get jiosaavn_source_description => 'الأفضل لمنطقة جنوب آسيا.'; + + @override + String get invidious_source_description => 'مشابه لـ Piped ولكن بتوافر أعلى'; + + @override + String highest_quality(Object quality) { + return 'أعلى جودة: $quality'; + } + + @override + String get select_audio_source => 'اختر مصدر الصوت'; + + @override + String get endless_playback_description => + 'إلحاق الأغاني الجديدة تلقائيًا\nإلى نهاية قائمة التشغيل'; + + @override + String get choose_your_region => 'اختر منطقتك'; + + @override + String get choose_your_region_description => + 'سيساعدك هذا في عرض المحتوى المناسب\nلموقعك.'; + + @override + String get choose_your_language => 'اختر لغتك'; + + @override + String get help_project_grow => 'ساعد في نمو هذا المشروع'; + + @override + String get help_project_grow_description => + 'Spotube هو مشروع مفتوح المصدر. يمكنك مساعدة هذا المشروع في النمو عن طريق المساهمة في المشروع، أو الإبلاغ عن الأخطاء، أو اقتراح ميزات جديدة.'; + + @override + String get contribute_on_github => 'المساهمة على GitHub'; + + @override + String get donate_on_open_collective => 'التبرع على Open Collective'; + + @override + String get browse_anonymously => 'تصفح بشكل مجهول'; + + @override + String get enable_connect => 'تمكين الاتصال'; + + @override + String get enable_connect_description => + 'التحكم في Spotube من الأجهزة الأخرى'; + + @override + String get devices => 'الأجهزة'; + + @override + String get select => 'اختر'; + + @override + String connect_client_alert(Object client) { + return 'أنت تتم التحكم بواسطة $client'; + } + + @override + String get this_device => 'هذا الجهاز'; + + @override + String get remote => 'بعيد'; + + @override + String get stats => 'إحصائيات'; + + @override + String and_n_more(Object count) { + return 'و $count أكثر'; + } + + @override + String get recently_played => 'تم تشغيله مؤخرًا'; + + @override + String get browse_more => 'تصفح المزيد'; + + @override + String get no_title => 'بدون عنوان'; + + @override + String get not_playing => 'غير مشغل'; + + @override + String get epic_failure => 'فشل كبير!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'تمت إضافة $tracks_length مسارات إلى قائمة الانتظار'; + } + + @override + String get spotube_has_an_update => 'يوجد تحديث لسبوتيوب'; + + @override + String get download_now => 'تحميل الآن'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'تم إصدار سبوتيوب الليلي $nightlyBuildNum'; + } + + @override + String release_version(Object version) { + return 'تم إصدار سبوتيوب v$version'; + } + + @override + String get read_the_latest => 'اقرأ الأحدث'; + + @override + String get release_notes => 'ملاحظات الإصدار'; + + @override + String get pick_color_scheme => 'اختر نظام الألوان'; + + @override + String get save => 'حفظ'; + + @override + String get choose_the_device => 'اختر الجهاز:'; + + @override + String get multiple_device_connected => + 'تم توصيل أجهزة متعددة.\nاختر الجهاز الذي تريد إجراء هذه العملية عليه'; + + @override + String get nothing_found => 'لم يتم العثور على شيء'; + + @override + String get the_box_is_empty => 'الصندوق فارغ'; + + @override + String get top_artists => 'أفضل الفنانين'; + + @override + String get top_albums => 'أفضل الألبومات'; + + @override + String get this_week => 'هذا الأسبوع'; + + @override + String get this_month => 'هذا الشهر'; + + @override + String get last_6_months => 'آخر 6 أشهر'; + + @override + String get this_year => 'هذا العام'; + + @override + String get last_2_years => 'آخر سنتين'; + + @override + String get all_time => 'كل الوقت'; + + @override + String powered_by_provider(Object providerName) { + return 'مدعوم من $providerName'; + } + + @override + String get email => 'البريد الإلكتروني'; + + @override + String get profile_followers => 'المتابعين'; + + @override + String get birthday => 'عيد الميلاد'; + + @override + String get subscription => 'اشتراك'; + + @override + String get not_born => 'لم يولد'; + + @override + String get hacker => 'هاكر'; + + @override + String get profile => 'الملف الشخصي'; + + @override + String get no_name => 'بدون اسم'; + + @override + String get edit => 'تعديل'; + + @override + String get user_profile => 'ملف المستخدم'; + + @override + String count_plays(Object count) { + return '$count تشغيلات'; + } + + @override + String get streaming_fees_hypothetical => 'رسوم البث (افتراضية)'; + + @override + String get minutes_listened => 'الدقائق المستمعة'; + + @override + String get streamed_songs => 'الأغاني المذاعة'; + + @override + String count_streams(Object count) { + return '$count بث'; + } + + @override + String get owned_by_you => 'مملوك لك'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'تم نسخ $shareUrl إلى الحافظة'; + } + + @override + String get hipotetical_calculation => + '*تمّ الحساب بمعدّل دفعة تتراوح بين 0.003–0.005 دولار أمريكي لكل تشغيل على منصات الموسيقى عبر الإنترنت. هذا حساب افتراضي لتوضيح للمستخدم مقدار ما كان سيدفعه للفنانين لو استمع إلى أغنيتهم على منصات مختلفة.'; + + @override + String count_mins(Object minutes) { + return '$minutes دقيقة'; + } + + @override + String get summary_minutes => 'الدقائق'; + + @override + String get summary_listened_to_music => 'استمعت إلى الموسيقى'; + + @override + String get summary_songs => 'أغاني'; + + @override + String get summary_streamed_overall => 'بث بشكل عام'; + + @override + String get summary_owed_to_artists => 'مدين للفنانين\nهذا الشهر'; + + @override + String get summary_artists => 'الفنانين'; + + @override + String get summary_music_reached_you => 'وصلت إليك الموسيقى'; + + @override + String get summary_full_albums => 'ألبومات كاملة'; + + @override + String get summary_got_your_love => 'حصلت على حبك'; + + @override + String get summary_playlists => 'قوائم التشغيل'; + + @override + String get summary_were_on_repeat => 'كانت على التكرار'; + + @override + String total_money(Object money) { + return 'المجموع $money'; + } + + @override + String get webview_not_found => 'لم يتم العثور على Webview'; + + @override + String get webview_not_found_description => + 'لم يتم تثبيت بيئة تشغيل Webview على جهازك.\nإذا كانت مثبتة، تأكد من وجودها في environment PATH\n\nبعد التثبيت، أعد تشغيل التطبيق'; + + @override + String get unsupported_platform => 'المنصة غير مدعومة'; + + @override + String get cache_music => 'تخزين الموسيقى مؤقتًا'; + + @override + String get open => 'فتح'; + + @override + String get cache_folder => 'مجلد التخزين المؤقت'; + + @override + String get export => 'تصدير'; + + @override + String get clear_cache => 'مسح التخزين المؤقت'; + + @override + String get clear_cache_confirmation => 'هل تريد مسح التخزين المؤقت؟'; + + @override + String get export_cache_files => 'تصدير الملفات المخزنة مؤقتًا'; + + @override + String found_n_files(Object count) { + return 'تم العثور على $count ملف'; + } + + @override + String get export_cache_confirmation => 'هل تريد تصدير هذه الملفات إلى'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'تم تصدير $filesExported من أصل $files ملفات'; + } + + @override + String get undo => 'تراجع'; + + @override + String get download_all => 'تنزيل الكل'; + + @override + String get add_all_to_playlist => 'إضافة الكل إلى قائمة التشغيل'; + + @override + String get add_all_to_queue => 'إضافة الكل إلى القائمة'; + + @override + String get play_all_next => 'تشغيل الكل بعد ذلك'; + + @override + String get pause => 'إيقاف مؤقت'; + + @override + String get view_all => 'عرض الكل'; + + @override + String get no_tracks_added_yet => 'يبدو أنك لم تضف أي مسارات بعد'; + + @override + String get no_tracks => 'يبدو أنه لا يوجد أي مسارات هنا'; + + @override + String get no_tracks_listened_yet => 'يبدو أنك لم تستمع إلى أي شيء بعد'; + + @override + String get not_following_artists => 'أنت لا تتابع أي فنانين'; + + @override + String get no_favorite_albums_yet => + 'يبدو أنك لم تضف أي ألبومات إلى المفضلة بعد'; + + @override + String get no_logs_found => 'لم يتم العثور على سجلات'; + + @override + String get youtube_engine => 'محرك يوتيوب'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine غير مثبت'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine غير مثبت في نظامك.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'تأكد من أنه متاح في متغير PATH أو\nحدد المسار الكامل للملف القابل للتنفيذ $engine أدناه'; + } + + @override + String get youtube_engine_unix_issue_message => + 'في أنظمة macOS/Linux/Unix مثل الأنظمة، لن يعمل تعيين المسار في .zshrc/.bashrc/.bash_profile وما إلى ذلك.\nيجب تعيين المسار في ملف تكوين الصدفة'; + + @override + String get download => 'تنزيل'; + + @override + String get file_not_found => 'الملف غير موجود'; + + @override + String get custom => 'مخصص'; + + @override + String get add_custom_url => 'إضافة URL مخصص'; + + @override + String get edit_port => 'تعديل المنفذ'; + + @override + String get port_helper_msg => + 'القيمة الافتراضية هي -1 والتي تشير إلى رقم عشوائي. إذا كان لديك جدار ناري مُعد، يُوصى بتعيين هذا.'; + + @override + String connect_request(Object client) { + return 'السماح لـ $client بالاتصال؟'; + } + + @override + String get connection_request_denied => + 'تم رفض الاتصال. المستخدم رفض الوصول.'; + + @override + String get an_error_occurred => 'حدث خطأ'; + + @override + String get copy_to_clipboard => 'نسخ إلى الحافظة'; + + @override + String get view_logs => 'عرض السجلات'; + + @override + String get retry => 'إعادة المحاولة'; + + @override + String get no_default_metadata_provider_selected => + 'لم تقُم بتعيين مزود بيانات افتراضي'; + + @override + String get manage_metadata_providers => 'إدارة مزوّدي البيانات'; + + @override + String get open_link_in_browser => 'فتح الرابط في المتصفح؟'; + + @override + String get do_you_want_to_open_the_following_link => + 'هل ترغب في فتح الرابط التالي؟'; + + @override + String get unsafe_url_warning => + 'قد يكون فتح الروابط من مصادر غير موثوقة غير آمن. تحرّ الحذر!\nيمكنك أيضًا نسخ الرابط إلى الحافظة.'; + + @override + String get copy_link => 'نسخ الرابط'; + + @override + String get building_your_timeline => + 'جاري بناء المخطط الزمني استنادًا إلى استماعاتك...'; + + @override + String get official => 'رسمي'; + + @override + String author_name(Object author) { + return 'المؤلّف: $author'; + } + + @override + String get third_party => 'طرف ثالث'; + + @override + String get plugin_requires_authentication => 'تتطلّب الإضافة تسجيل الدخول'; + + @override + String get update_available => 'تحديث متوفر'; + + @override + String get supports_scrobbling => 'يدعم التتبع (scrobbling)'; + + @override + String get plugin_scrobbling_info => + 'تقوم هذه الإضافة بتتبع مقاطعك الموسيقية لإنشاء سجل الاستماع الخاص بك.'; + + @override + String get default_metadata_source => 'مصدر البيانات الوصفية الافتراضي'; + + @override + String get set_default_metadata_source => + 'تعيين مصدر البيانات الوصفية الافتراضي'; + + @override + String get default_audio_source => 'مصدر الصوت الافتراضي'; + + @override + String get set_default_audio_source => 'تعيين مصدر الصوت الافتراضي'; + + @override + String get set_default => 'تعيين كافتراضي'; + + @override + String get support => 'الدعم'; + + @override + String get support_plugin_development => 'دعم تطوير الإضافات'; + + @override + String can_access_name_api(Object name) { + return '- يمكن الوصول إلى واجهة برمجة التطبيقات **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'هل ترغب في تثبيت هذه الإضافة؟'; + + @override + String get third_party_plugin_warning => + 'هذه الإضافة من مستودع طرف ثالث. تأكد من موثوقية المصدر قبل التثبيت.'; + + @override + String get author => 'المؤلف'; + + @override + String get this_plugin_can_do_following => 'يمكن لهذه الإضافة القيام بما يلي'; + + @override + String get install => 'تثبيت'; + + @override + String get install_a_metadata_provider => 'تثبيت مزوّد بيانات'; + + @override + String get no_tracks_playing => 'لا توجد مقاطع تعمل حاليًا'; + + @override + String get synced_lyrics_not_available => + 'الكلمات المتزامنة غير متوفرة لهذه الأغنية. يُرجى استخدام'; + + @override + String get plain_lyrics => 'الكلمات العادية'; + + @override + String get tab_instead => 'بدلاً من ذلك، استخدم التبويب.'; + + @override + String get disclaimer => 'إخلاء المسؤولية'; + + @override + String get third_party_plugin_dmca_notice => + 'لا تتحمّل فريق Spotube أي مسؤولية (بما في ذلك القانونية) عن أي من الإضافات “لطرف ثالث”.\nاستخدمها على مسؤوليتك الخاصّة. لأيّة أخطاء/مشكلات، يُرجى الإبلاغ عنها في مستودع الإضافة.\n\nإذا كانت أي إضافة “لطرف ثالث” تنتهك شروط الخدمة أو قانون DMCA الخاص بأي خدمة أو كيان قانوني، فيُرجى طلب اتخاذ إجراء من مؤلف الإضافة أو منصة الاستضافة مثل GitHub/Codeberg. الإضافات المدرجة كـ “لطرف ثالث” هي مفعّلة ومُدارة من المجتمع، وليس لدينا صلاحية إدارتها أو التدخل فيها.\n\n'; + + @override + String get input_does_not_match_format => + 'المدخل لا يتوافق مع التنسيق المطلوب'; + + @override + String get plugins => 'الإضافات'; + + @override + String get paste_plugin_download_url => + 'الصق رابط التنزيل أو GitHub/Codeberg أو رابط مباشر لملف .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'تنزيل وتثبيت الإضافة من رابط'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'فشل في إضافة الإضافة: $error'; + } + + @override + String get upload_plugin_from_file => 'رفع الإضافة من ملف'; + + @override + String get installed => 'تم التثبيت'; + + @override + String get available_plugins => 'الإضافات المتوفّرة'; + + @override + String get configure_plugins => + 'قم بتكوين مزود البيانات الوصفية ومكونات مصدر الصوت الخاصة بك'; + + @override + String get audio_scrobblers => 'أجهزة تتبع الصوت'; + + @override + String get scrobbling => 'التتبع'; + + @override + String get source => 'المصدر: '; + + @override + String get uncompressed => 'غير مضغوط'; + + @override + String get dab_music_source_description => + 'لمحبي الصوتيات. يوفر تدفقات صوتية عالية الجودة/بدون فقدان. مطابقة دقيقة للمسارات بناءً على ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_bn.dart b/lib/l10n/generated/app_localizations_bn.dart new file mode 100644 index 00000000..7dc1e07f --- /dev/null +++ b/lib/l10n/generated/app_localizations_bn.dart @@ -0,0 +1,1566 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Bengali Bangla (`bn`). +class AppLocalizationsBn extends AppLocalizations { + AppLocalizationsBn([String locale = 'bn']) : super(locale); + + @override + String get guest => 'অতিথি'; + + @override + String get browse => 'ব্রাউজ করুন'; + + @override + String get search => 'অনুসন্ধান করুন'; + + @override + String get library => 'লাইব্রেরী'; + + @override + String get lyrics => 'গানের কথা'; + + @override + String get settings => 'সেটিংস'; + + @override + String get genre_categories_filter => 'গানের ধরণ বা শ্রেণি খুঁজুন'; + + @override + String get genre => 'গানের ধরণ'; + + @override + String get personalized => 'আপনার জন্য'; + + @override + String get featured => 'বৈশিষ্ট্যযুক্ত'; + + @override + String get new_releases => 'সাম্প্রতিক মুক্তি প্রাপ্ত'; + + @override + String get songs => 'গান'; + + @override + String playing_track(Object track) { + return '$track চালানো হচ্ছে'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'এটি বর্তমান প্লেলিষ্ট সাফ করে দিবে। $track_lengthটি গান বাদ দেওয়া হবে\nআপনি কি চালিয়ে যেতে চান?'; + } + + @override + String get load_more => 'আরো লোড করুন'; + + @override + String get playlists => 'প্লেলিস্ট'; + + @override + String get artists => 'শিল্পী'; + + @override + String get albums => 'অ্যালবাম'; + + @override + String get tracks => 'গানের ট্র্যাক'; + + @override + String get downloads => 'ডাউনলোড'; + + @override + String get filter_playlists => 'প্লেলিস্ট অনুসন্ধান করুন...'; + + @override + String get liked_tracks => 'পছন্দের গান'; + + @override + String get liked_tracks_description => 'আপনার পছন্দের গান সমূহ'; + + @override + String get playlist => 'প্লেলিস্ট'; + + @override + String get create_a_playlist => 'একটি প্লেলিস্ট তৈরি করুন'; + + @override + String get update_playlist => 'প্লেলিস্ট আপডেট করুন'; + + @override + String get create => 'তৈরি করুন'; + + @override + String get cancel => 'বাতিল করুন'; + + @override + String get update => 'আপডেট'; + + @override + String get playlist_name => 'প্লেলিস্টের নাম'; + + @override + String get name_of_playlist => 'প্লেলিস্টের নাম'; + + @override + String get description => 'বিবরণ'; + + @override + String get public => 'পাবলিক'; + + @override + String get collaborative => 'সহযোগিতামূলক'; + + @override + String get search_local_tracks => 'ডাউনলোডকৃত গান অনুসন্ধান করুন...'; + + @override + String get play => 'চালান'; + + @override + String get delete => 'মুছে ফেলুন'; + + @override + String get none => 'কোনটিই না'; + + @override + String get sort_a_z => 'A-Z ক্রমে সাজান'; + + @override + String get sort_z_a => 'Z-A ক্রমে সাজান'; + + @override + String get sort_artist => 'শিল্পীর ক্রমে সাজান'; + + @override + String get sort_album => 'অ্যালবামের ক্রমে সাজান'; + + @override + String get sort_duration => 'দৈর্ঘ্য অনুযায়ী বাছাই করুন'; + + @override + String get sort_tracks => 'গানের ক্রম'; + + @override + String currently_downloading(Object tracks_length) { + return 'ডাউনলোড করা হচ্ছে ($tracks_length)'; + } + + @override + String get cancel_all => 'সব বাতিল করুন'; + + @override + String get filter_artist => 'শিল্পীর অনুসন্ধান করুন...'; + + @override + String followers(Object followers) { + return '$followers অনুসরণকারী'; + } + + @override + String get add_artist_to_blacklist => 'শিল্পীকে ব্ল্যাকলিস্টে যোগ করুন'; + + @override + String get top_tracks => 'শীর্ষ গানের ট্র্যাক'; + + @override + String get fans_also_like => 'অনুসরণকারীদের পছন্দ'; + + @override + String get loading => 'লোড হচ্ছে...'; + + @override + String get artist => 'শিল্পী'; + + @override + String get blacklisted => 'ব্ল্যাকলিস্টে আছে'; + + @override + String get following => 'অনুসরণ করছেন'; + + @override + String get follow => 'অনুসরণ করুন'; + + @override + String get artist_url_copied => 'শিল্পীর URL কপি করা হয়েছে'; + + @override + String added_to_queue(Object tracks) { + return '$tracksটি গানের ট্র্যাক কিউতে যোগ করা হয়েছে'; + } + + @override + String get filter_albums => 'অ্যালবাম অনুসন্ধান করুন...'; + + @override + String get synced => 'সময় সুসংগত'; + + @override + String get plain => 'অসুসংগত'; + + @override + String get shuffle => 'অদলবদল'; + + @override + String get search_tracks => 'গান অনুসন্ধান করুন...'; + + @override + String get released => 'প্রকাশিত হয়েছে'; + + @override + String error(Object error) { + return 'ত্রুটি $error'; + } + + @override + String get title => 'শিরোনাম'; + + @override + String get time => 'সময়'; + + @override + String get more_actions => 'আরও অপশন'; + + @override + String download_count(Object count) { + return 'ডাউনলোড ($countটি)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'প্লেলিস্টে যোগ করুন ($countটি)'; + } + + @override + String add_count_to_queue(Object count) { + return 'কিউতে যোগ করুন ($countটি)'; + } + + @override + String play_count_next(Object count) { + return 'পরবর্তীতে চালান ($countটি)'; + } + + @override + String get album => 'অ্যালবাম'; + + @override + String copied_to_clipboard(Object data) { + return '$data ক্লিপবোর্ডে কপি করা হয়েছে'; + } + + @override + String add_to_following_playlists(Object track) { + return 'নিম্নলিখিত প্লেলিস্টে $track যোগ করুন'; + } + + @override + String get add => 'যোগ করুন'; + + @override + String added_track_to_queue(Object track) { + return 'কিউতে $track যোগ করা হয়েছে'; + } + + @override + String get add_to_queue => 'কিউতে যোগ করুন'; + + @override + String track_will_play_next(Object track) { + return '$track পরবর্তীতে চালানো হবে'; + } + + @override + String get play_next => 'পরবর্তীতে চালান'; + + @override + String removed_track_from_queue(Object track) { + return 'কিউ থেকে $track সরিয়ে নেওয়া হয়েছে'; + } + + @override + String get remove_from_queue => 'কিউ থেকে সরান'; + + @override + String get remove_from_favorites => 'পছন্দের তালিকা থেকে অপসারণ করুন'; + + @override + String get save_as_favorite => 'পছন্দের তালিকায় সংরক্ষণ করুন'; + + @override + String get add_to_playlist => 'প্লেলিস্টে যোগ করুন'; + + @override + String get remove_from_playlist => 'প্লেলিস্ট থেকে সরান'; + + @override + String get add_to_blacklist => 'ব্ল্যাকলিস্টে যোগ করুন'; + + @override + String get remove_from_blacklist => 'ব্ল্যাকলিস্ট থেকে সরান'; + + @override + String get share => 'শেয়ার করুন'; + + @override + String get mini_player => 'মিনি প্লেয়ার'; + + @override + String get slide_to_seek => 'গান সামনে বা পিছনে নিতে স্লাইড করুন'; + + @override + String get shuffle_playlist => 'প্লেলিস্ট এলোমেলো করুন'; + + @override + String get unshuffle_playlist => 'প্লেলিস্ট আগের মতো করুন'; + + @override + String get previous_track => 'আগের গানের ট্র্যাক'; + + @override + String get next_track => 'পরের গানের ট্র্যাক'; + + @override + String get pause_playback => 'গান বন্ধ করুন'; + + @override + String get resume_playback => 'গান চালু করুন'; + + @override + String get loop_track => 'গান শেষে পুনরায় চালান'; + + @override + String get no_loop => 'কোনো লুপ নেই'; + + @override + String get repeat_playlist => 'প্লেলিস্ট শেষে পুনরায় চালান'; + + @override + String get queue => 'গানের কিউ'; + + @override + String get alternative_track_sources => 'বিকল্প গানের উৎস'; + + @override + String get download_track => 'গান ডাউনলোড করুন'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracksটি গান কিউতে রয়েছে'; + } + + @override + String get clear_all => 'সব মুছে ফেলুন'; + + @override + String get show_hide_ui_on_hover => 'হভার করলে UI দেখান/লুকান'; + + @override + String get always_on_top => 'সর্বদা উপরে'; + + @override + String get exit_mini_player => 'মিনি প্লেয়ার থেকে বের হয়ে যান'; + + @override + String get download_location => 'ডাউনলোড স্থান'; + + @override + String get local_library => 'স্থানীয় লাইব্রেরি'; + + @override + String get add_library_location => 'লাইব্রেরিতে যোগ করুন'; + + @override + String get remove_library_location => 'লাইব্রেরি থেকে সরান'; + + @override + String get account => 'অ্যাকাউন্ট'; + + @override + String get logout => 'লগআউট করুন'; + + @override + String get logout_of_this_account => 'অ্যাকাউন্ট থেকে লগআউট করুন'; + + @override + String get language_region => 'ভাষা ও অঞ্চল'; + + @override + String get language => 'ভাষা'; + + @override + String get system_default => 'সিস্টেম ডিফল্ট'; + + @override + String get market_place_region => 'মার্কেটপ্লেস অঞ্চল'; + + @override + String get recommendation_country => 'দেশভিত্তিক সঙ্গীত পরামর্শের জন্য দেশ'; + + @override + String get appearance => 'রুপ'; + + @override + String get layout_mode => 'UI বিন্যাস রূপ'; + + @override + String get override_layout_settings => + 'প্রতিক্রিয়াশীল UI বিন্যাস রূপের সেটিংস পরিবর্তন করুন'; + + @override + String get adaptive => 'অভিযোজিত'; + + @override + String get compact => 'আঁটসাঁট UI'; + + @override + String get extended => 'বিস্তৃত UI'; + + @override + String get theme => 'থিম'; + + @override + String get dark => 'অন্ধকার'; + + @override + String get light => 'উজ্জল'; + + @override + String get system => 'সিস্টেম থিম'; + + @override + String get accent_color => 'প্রভাবশালী রং'; + + @override + String get sync_album_color => 'অ্যালবাম সুসংগত UI এর রং'; + + @override + String get sync_album_color_description => + 'অ্যালবাম কভারের প্রভাবশালী রঙ UI অ্যাকসেন্ট রঙ হিসাবে ব্যবহার করে'; + + @override + String get playback => 'সংগীতের প্লেব্যাক'; + + @override + String get audio_quality => 'শব্দের গুণমান'; + + @override + String get high => 'উচ্চ'; + + @override + String get low => 'নিম্ন'; + + @override + String get pre_download_play => 'আগে গান ডাউনলোড করে পরে চালান '; + + @override + String get pre_download_play_description => + 'গান স্ট্রিম করার পরিবর্তে, ডাউনলোড করুন এবং প্লে করুন (উচ্চ ব্যান্ডউইথ ব্যবহারকারীদের জন্য প্রস্তাবিত)'; + + @override + String get skip_non_music => + 'গানের নন-মিউজিক সেগমেন্ট এড়িয়ে যান (SponsorBlock)'; + + @override + String get blacklist_description => + 'কালো তালিকাভুক্ত গানের ট্র্যাক এবং শিল্পী'; + + @override + String get wait_for_download_to_finish => + 'ডাউনলোড শেষ হওয়ার জন্য অপেক্ষা করুন'; + + @override + String get desktop => 'ডেস্কটপ'; + + @override + String get close_behavior => 'বন্ধ করার প্রক্রিয়া'; + + @override + String get close => 'বন্ধ করুন'; + + @override + String get minimize_to_tray => 'সিস্টেম ট্রেতে রাখুন'; + + @override + String get show_tray_icon => 'সিস্টেম ট্রে আইকন দেখান'; + + @override + String get about => 'বিস্তারিত'; + + @override + String get u_love_spotube => 'আমরা জানি আপনি Spotube কে ভালবাসেন'; + + @override + String get check_for_updates => 'আপডেট চেক করুন'; + + @override + String get about_spotube => 'Spotube সম্পর্কে বিস্তারিত'; + + @override + String get blacklist => 'কালো তালিকা'; + + @override + String get please_sponsor => 'স্পনসর/সহায়তা করুন'; + + @override + String get spotube_description => + 'Spotube, একটি কর্মদক্ষ, ক্রস-প্ল্যাটফর্ম, বিনামূল্যের জন্য Spotify ক্লায়েন্ট'; + + @override + String get version => 'সংস্করণ'; + + @override + String get build_number => 'বিল্ড নম্বর'; + + @override + String get founder => 'প্রতিষ্ঠাতা'; + + @override + String get repository => 'সংগ্রহস্থল'; + + @override + String get bug_issues => 'বাগ/সমস্যা'; + + @override + String get made_with => '❤️ দিয়ে বাংলাদেশে🇧🇩 তৈরি'; + + @override + String get kingkor_roy_tirtho => 'কিংকর রায় তীর্থ'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year কিংকর রায় তীর্থ'; + } + + @override + String get license => 'লাইসেন্স'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'চিন্তা করবেন না, আপনার কোনো লগইন তথ্য সংগ্রহ করা হবে না বা কারো সাথে শেয়ার করা হবে না'; + + @override + String get know_how_to_login => 'আপনি কিভাবে লগইন করবেন তা জানেন না?'; + + @override + String get follow_step_by_step_guide => 'ধাপে ধাপে নির্দেশিকা অনুসরণ করুন'; + + @override + String cookie_name_cookie(Object name) { + return '$name কুকি'; + } + + @override + String get fill_in_all_fields => 'সমস্ত ফর্ম ক্ষেত্র পূরণ করুন'; + + @override + String get submit => 'জমা দিন'; + + @override + String get exit => 'প্রস্থান'; + + @override + String get previous => 'পূর্ববর্তী'; + + @override + String get next => 'পরবর্তী'; + + @override + String get done => 'সম্পন্ন'; + + @override + String get step_1 => 'ধাপ 1'; + + @override + String get first_go_to => 'প্রথমে যান'; + + @override + String get something_went_wrong => 'কিছু ভুল হয়েছে'; + + @override + String get piped_instance => 'Piped সার্ভার এড্রেস'; + + @override + String get piped_description => 'গান ম্যাচ করার জন্য ব্যবহৃত পাইপড সার্ভার'; + + @override + String get piped_warning => + 'এগুলোর মধ্যে কিছু ভাল কাজ নাও করতে পারে৷ তাই নিজ দায়িত্বে ব্যবহার করুন'; + + @override + String get invidious_instance => 'ইনভিডিয়াস সার্ভার ইন্সটেন্স'; + + @override + String get invidious_description => + 'ট্রাক মিলানোর জন্য ব্যবহৃত ইনভিডিয়াস সার্ভার'; + + @override + String get invidious_warning => + 'কিছু সার্ভার ভাল কাজ নাও করতে পারে। নিজের ঝুঁকিতে ব্যবহার করুন'; + + @override + String get generate => 'উৎপন্ন করুন'; + + @override + String track_exists(Object track) { + return 'ট্র্যাক $track ইতিমধ্যে বিদ্যমান'; + } + + @override + String get replace_downloaded_tracks => + 'সমস্ত ডাউনলোড করা ট্র্যাক প্রতিস্থাপন করুন'; + + @override + String get skip_download_tracks => 'সমস্ত ডাউনলোড করা ট্র্যাক এ স্কিপ করুন'; + + @override + String get do_you_want_to_replace => + 'আপনি কি বিদ্যমান ট্র্যাকটি প্রতিস্থাপন করতে চান?'; + + @override + String get replace => 'প্রতিস্থাপন করুন'; + + @override + String get skip => 'স্কিপ করুন'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '$count $type পর্যন্ত নির্বাচন করুন'; + } + + @override + String get select_genres => 'গানের ধরণ নির্বাচন করুন'; + + @override + String get add_genres => 'গানের ধরণ যুক্ত করুন'; + + @override + String get country => 'দেশ'; + + @override + String get number_of_tracks_generate => 'উত্পাদিত ট্র্যাকের সংখ্যা'; + + @override + String get acousticness => 'অধ্যাত্মিকতা'; + + @override + String get danceability => 'নৃত্যমূলকতা'; + + @override + String get energy => 'শক্তি'; + + @override + String get instrumentalness => 'সাধারণতা'; + + @override + String get liveness => 'জীবনমুক্ততা'; + + @override + String get loudness => 'স্বরের উচ্চতা'; + + @override + String get speechiness => 'বক্তব্যমূলকতা'; + + @override + String get valence => 'সন্তোষমূলকতা'; + + @override + String get popularity => 'জনপ্রিয়তা'; + + @override + String get key => 'কী'; + + @override + String get duration => 'সময়কাল (সেকেন্ড)'; + + @override + String get tempo => 'গতি (বিপিএম)'; + + @override + String get mode => 'মোড'; + + @override + String get time_signature => 'সময়ের স্বাক্ষর'; + + @override + String get short => 'সংক্ষিপ্ত'; + + @override + String get medium => 'মাঝারি'; + + @override + String get long => 'দীর্ঘ'; + + @override + String get min => 'সর্বনিম্ন'; + + @override + String get max => 'সর্বাধিক'; + + @override + String get target => 'লক্ষ্য'; + + @override + String get moderate => 'মাঝারি'; + + @override + String get deselect_all => 'সমস্ত অপচুন করুন'; + + @override + String get select_all => 'সমস্ত নির্বাচন করুন'; + + @override + String get are_you_sure => 'আপনি কি নিশ্চিত?'; + + @override + String get generating_playlist => 'আপনার কাস্টম প্লেলিস্ট তৈরি হচ্ছে...'; + + @override + String selected_count_tracks(Object count) { + return '$count ট্র্যাক নির্বাচিত'; + } + + @override + String get download_warning => + 'যদি আপনি সমস্ত ট্র্যাকগুলি একসঙ্গে ডাউনলোড করেন, তবে আপনি নিশ্চিতভাবে সঙ্গীত চুরি করছেন এবং সৃষ্টিশীল সমাজে ক্ষতি দিচ্ছেন। আমি আশা করি আপনি এটা সম্পর্কে জানেন। সর্বদা, শিল্পীদের কঠিন পরিশ্রমকে সম্মান করতে চেষ্টা করুন এবং সমর্থন করুন'; + + @override + String get download_ip_ban_warning => + 'তথ্যবিশ্বস্ত করে নেওয়া যায় যে, আপনার IP ঠিকানাটি YouTube দ্বারা স্থানান্তরিত করা হতে পারে যখন সাধারন থেকে বেশি ডাউনলোড অনুরোধ হয়। IP ব্লকের মাধ্যমে আপনি কমপক্ষে ২-৩ মাস ধরে (ঐ IP ডিভাইস থেকে) YouTube ব্যবহার করতে পারবেন না। এবং Spotube কোনও দায়িত্ব সম্পর্কে দায়িত্ব বহন করে না যদি এটি ঘটে।'; + + @override + String get by_clicking_accept_terms => + '\'গ্রহণ\' ক্লিক করে আপনি নিম্নলিখিত শর্তাদি স্বীকার করছেন:'; + + @override + String get download_agreement_1 => 'আমি জানি আমি সঙ্গীত চুরি করছি। আমি খারাপ'; + + @override + String get download_agreement_2 => + 'আমি কেবলমাত্র তাদের কাজ কেনার জন্য অর্থ নেই কিন্তু যেখানে প্রয়োজন সেখানে আমি শিল্পীদের সমর্থন করব।'; + + @override + String get download_agreement_3 => + 'আমি সম্পূর্ণরূপে জানি যে আমার IP YouTube-তে ব্লক হতে পারে এবং আমি Spotube বা তার মালিকানাধীন কোনও দায়িত্ব পেতে পারিনি আমার বর্তমান ক্রিয়াটি দ্বারা সৃষ্ট দুর্ঘটনা করার জন্য'; + + @override + String get decline => 'অগ্রায়ন করুন'; + + @override + String get accept => 'গ্রহণ করুন'; + + @override + String get details => 'বিস্তারিত'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'চ্যানেল'; + + @override + String get likes => 'লাইক'; + + @override + String get dislikes => 'অপছন্দ'; + + @override + String get views => 'দর্শনার্থী'; + + @override + String get streamUrl => 'স্ট্রিম URL'; + + @override + String get stop => 'বন্ধ করুন'; + + @override + String get sort_newest => 'নতুনতম অনুসারে সাজান'; + + @override + String get sort_oldest => 'পুরানোতম অনুসারে সাজান'; + + @override + String get sleep_timer => 'স্লীপ টাইমার'; + + @override + String mins(Object minutes) { + return '$minutes মিনিট'; + } + + @override + String hours(Object hours) { + return '$hours ঘন্টা'; + } + + @override + String hour(Object hours) { + return '$hours ঘন্টা'; + } + + @override + String get custom_hours => 'কাস্টম ঘন্টা'; + + @override + String get logs => 'লগ'; + + @override + String get developers => 'ডেভেলপার'; + + @override + String get not_logged_in => 'আপনি লগইন করা নেই'; + + @override + String get search_mode => 'অনুসন্ধান মোড'; + + @override + String get audio_source => 'অডিও উৎস'; + + @override + String get ok => 'ঠিক আছে'; + + @override + String get failed_to_encrypt => 'এনক্রিপ্ট করা ব্যর্থ হয়েছে'; + + @override + String get encryption_failed_warning => + 'Spotube আপনার তথ্যগুলি নিরাপদভাবে স্টোর করতে এনক্রিপশন ব্যবহার করে। কিন্তু এটি ব্যর্থ হয়েছে। তাই এটি অনিরাপদ স্টোরে ফলফল হবে\nযদি আপনি Linux ব্যবহার করেন, তবে দয়া করে নিশ্চিত হউন যে আপনার কোনও সিক্রেট-সার্ভিস gnome-keyring, kde-wallet, keepassxc ইত্যাদি ইনস্টল করা আছে'; + + @override + String get querying_info => 'তথ্য অনুসন্ধান করা হচ্ছে'; + + @override + String get piped_api_down => 'পাইপড API ডাউন আছে'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'বর্তমানে পাইপড ইনস্ট্যান্স $pipedInstance ডাউন আছে\n\nইনস্ট্যান্স পরিবর্তন করুন অথবা \'API টাইপ\' পরিবর্তন করুন অফিসিয়াল ইউটিউব API হতে\n\nপরিবর্তনের পরে অ্যাপটি পুনরায় চালানোর নিশ্চিত করুন'; + } + + @override + String get you_are_offline => 'আপনি বর্তমানে অফলাইন'; + + @override + String get connection_restored => 'আপনার ইন্টারনেট সংযোগ পুনরুদ্ধার হয়েছে'; + + @override + String get use_system_title_bar => 'সিস্টেম শিরোনাম বার ব্যবহার করুন'; + + @override + String get crunching_results => 'ফলাফল বিশ্লেষণ করা হচ্ছে...'; + + @override + String get search_to_get_results => 'ফলাফল পেতে খোঁজ করুন'; + + @override + String get use_amoled_mode => 'AMOLED মোড ব্যবহার করুন'; + + @override + String get pitch_dark_theme => 'পিচ ব্ল্যাক ডার্ট থিম'; + + @override + String get normalize_audio => 'অডিও স্তরমান করুন'; + + @override + String get change_cover => 'কভার পরিবর্তন করুন'; + + @override + String get add_cover => 'কভার যোগ করুন'; + + @override + String get restore_defaults => 'ডিফল্ট সেটিংস পুনরুদ্ধার করুন'; + + @override + String get download_music_format => 'গান ডাউনলোডের বিন্যাস'; + + @override + String get streaming_music_format => 'গান স্ট্রিমিং এর বিন্যাস'; + + @override + String get download_music_quality => 'গান ডাউনলোডের মান'; + + @override + String get streaming_music_quality => 'গান স্ট্রিমিং এর মান'; + + @override + String get login_with_lastfm => 'Last.fm দিয়ে লগইন করুন'; + + @override + String get connect => 'সংযোগ করুন'; + + @override + String get disconnect_lastfm => 'Last.fm সংযোগ বিচ্ছিন্ন করুন'; + + @override + String get disconnect => 'সংযোগ বিচ্ছিন্ন করুন'; + + @override + String get username => 'ব্যবহারকারীর নাম'; + + @override + String get password => 'পাসওয়ার্ড'; + + @override + String get login => 'লগইন'; + + @override + String get login_with_your_lastfm => + 'আপনার Last.fm অ্যাকাউন্ট দিয়ে লগইন করুন'; + + @override + String get scrobble_to_lastfm => 'Last.fm এ স্ক্রবল করুন'; + + @override + String get go_to_album => 'الانتقال إلى الألبوم'; + + @override + String get discord_rich_presence => 'وجود ديسكورد الغني'; + + @override + String get browse_all => 'تصفح الكل'; + + @override + String get genres => 'الأنواع الموسيقية'; + + @override + String get explore_genres => 'استكشاف الأنواع'; + + @override + String get friends => 'বন্ধু'; + + @override + String get no_lyrics_available => + 'দুঃখিত, এই ট্র্যাকের জন্য কথা খুঁজে পাওয়া গেলনা'; + + @override + String get start_a_radio => 'রেডিও শুরু করুন'; + + @override + String get how_to_start_radio => 'রেডিও কিভাবে শুরু করতে চান?'; + + @override + String get replace_queue_question => + 'আপনি বর্তমান কিউটি প্রতিস্থাপন করতে চান কিনা বা এর সাথে যুক্ত করতে চান?'; + + @override + String get endless_playback => 'অবিরাম প্রচার'; + + @override + String get delete_playlist => 'প্লেলিস্ট মুছুন'; + + @override + String get delete_playlist_confirmation => + 'আপনি কি নিশ্চিত যে আপনি এই প্লেলিস্টটি মুছতে চান?'; + + @override + String get local_tracks => 'স্থানীয় ট্র্যাক'; + + @override + String get local_tab => 'স্থানীয়'; + + @override + String get song_link => 'গানের লিংক'; + + @override + String get skip_this_nonsense => 'এই বাকবাস পালান'; + + @override + String get freedom_of_music => '“সংগীতের স্বাধীনতা”'; + + @override + String get freedom_of_music_palm => '“তোমার হাতের কাছে সংগীতের স্বাধীনতা”'; + + @override + String get get_started => 'শুরু করা যাক'; + + @override + String get youtube_source_description => 'প্রস্তাবিত এবং সেরা কাজ করে।'; + + @override + String get piped_source_description => 'মন খারাপ? ইউটিউবের মতো আবার ফ্রি।'; + + @override + String get jiosaavn_source_description => 'দক্ষিণ এশিয়ান অঞ্চলের জন্য সেরা।'; + + @override + String get invidious_source_description => + 'পাইপের মতো কিন্তু আরও বেশি উপলব্ধতা সহ'; + + @override + String highest_quality(Object quality) { + return 'সর্বোচ্চ গুণগতি: $quality'; + } + + @override + String get select_audio_source => 'অডিও উৎস নির্বাচন করুন'; + + @override + String get endless_playback_description => + 'নতুন গান নিজে নিজে প্লেলিস্টের শেষে\nসংযুক্ত করুন'; + + @override + String get choose_your_region => 'আপনার অঞ্চল নির্বাচন করুন'; + + @override + String get choose_your_region_description => + 'এটি স্পটুবে আপনাকে আপনার অবস্থানের জন্য ঠিক কন্টেন্ট দেখানোর সাহায্য করবে।'; + + @override + String get choose_your_language => 'আপনার ভাষা নির্বাচন করুন'; + + @override + String get help_project_grow => 'এই প্রকল্পের বৃদ্ধি করুন'; + + @override + String get help_project_grow_description => + 'স্পটুব একটি ওপেন সোর্স প্রকল্প। আপনি প্রকল্পে অবদান রাখেন, বাগ রিপোর্ট করেন, বা নতুন বৈশিষ্ট্যগুলি সুপারিশ করেন।'; + + @override + String get contribute_on_github => 'গিটহাবে অবদান রাখুন'; + + @override + String get donate_on_open_collective => 'ওপেন কলেক্টিভে অনুদান করুন'; + + @override + String get browse_anonymously => 'অজানে ব্রাউজ করুন'; + + @override + String get enable_connect => 'সংযোগ সক্রিয় করুন'; + + @override + String get enable_connect_description => + 'অন্যান্য ডিভাইস থেকে Spotube নিয়ন্ত্রণ করুন'; + + @override + String get devices => 'ডিভাইস'; + + @override + String get select => 'নির্বাচন করুন'; + + @override + String connect_client_alert(Object client) { + return 'আপনি $client দ্বারা নিয়ন্ত্রিত হচ্ছেন'; + } + + @override + String get this_device => 'এই ডিভাইস'; + + @override + String get remote => 'রিমোট'; + + @override + String get stats => 'পরিসংখ্যান'; + + @override + String and_n_more(Object count) { + return 'এবং $count আরও'; + } + + @override + String get recently_played => 'সম্প্রতি বাজানো'; + + @override + String get browse_more => 'আরও ব্রাউজ করুন'; + + @override + String get no_title => 'কোনো শিরোনাম নেই'; + + @override + String get not_playing => 'চালানো হচ্ছে না'; + + @override + String get epic_failure => 'বিরাট ব্যর্থতা!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length ট্র্যাক সারিতে যোগ করা হয়েছে'; + } + + @override + String get spotube_has_an_update => 'স্পটিউবে একটি আপডেট আছে'; + + @override + String get download_now => 'এখনই ডাউনলোড করুন'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'স্পটিউব নাইটলি $nightlyBuildNum প্রকাশিত হয়েছে'; + } + + @override + String release_version(Object version) { + return 'স্পটিউব v$version প্রকাশিত হয়েছে'; + } + + @override + String get read_the_latest => 'সর্বশেষ পড়ুন'; + + @override + String get release_notes => 'রিলিজ নোট'; + + @override + String get pick_color_scheme => 'রঙের থিম নির্বাচন করুন'; + + @override + String get save => 'সংরক্ষণ করুন'; + + @override + String get choose_the_device => 'ডিভাইস নির্বাচন করুন:'; + + @override + String get multiple_device_connected => + 'একাধিক ডিভাইস সংযুক্ত রয়েছে।\nযে ডিভাইসে আপনি এই ক্রিয়াটি চালাতে চান সেটি নির্বাচন করুন'; + + @override + String get nothing_found => 'কিছুই পাওয়া যায়নি'; + + @override + String get the_box_is_empty => 'বাক্সটি খালি'; + + @override + String get top_artists => 'শীর্ষ শিল্পী'; + + @override + String get top_albums => 'শীর্ষ অ্যালবাম'; + + @override + String get this_week => 'এই সপ্তাহ'; + + @override + String get this_month => 'এই মাস'; + + @override + String get last_6_months => 'গত ৬ মাস'; + + @override + String get this_year => 'এই বছর'; + + @override + String get last_2_years => 'গত ২ বছর'; + + @override + String get all_time => 'সব সময়'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName দ্বারা চালিত'; + } + + @override + String get email => 'ইমেইল'; + + @override + String get profile_followers => 'অনুসারী'; + + @override + String get birthday => 'জন্মদিন'; + + @override + String get subscription => 'সাবস্ক্রিপশন'; + + @override + String get not_born => 'জন্মগ্রহণ করেনি'; + + @override + String get hacker => 'হ্যাকার'; + + @override + String get profile => 'প্রোফাইল'; + + @override + String get no_name => 'কোন নাম নেই'; + + @override + String get edit => 'সম্পাদনা করুন'; + + @override + String get user_profile => 'ব্যবহারকারীর প্রোফাইল'; + + @override + String count_plays(Object count) { + return '$count বার প্লে হয়েছে'; + } + + @override + String get streaming_fees_hypothetical => 'স্ট্রিমিং ফি (ধারণাগত)'; + + @override + String get minutes_listened => 'শুনেছেন মিনিট'; + + @override + String get streamed_songs => 'স্ট্রিম করা গান'; + + @override + String count_streams(Object count) { + return '$count বার স্ট্রিম'; + } + + @override + String get owned_by_you => 'আপনার মালিকানাধীন'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl ক্লিপবোর্ডে কপি করা হয়েছে'; + } + + @override + String get hipotetical_calculation => + '*এটি নিরূপণ করা হয়েছে গড় অনলাইন মিউজিক স্ট্রিমিং প্ল্যাটফর্মের প্রতি স্ট্রিম 0.003–0.005 USD পেআউটের ভিত্তিতে। এটি একটি কাল্পনিক হিসাব যা ব্যবহারকারীকে ধারণা দিতে পারে তারা অন্যান্য স্ট্রিমিং প্ল্যাটফর্মে একই গান শোনার জন্য শিল্পীদের কত টাকা দিয়েছেন হোক।'; + + @override + String count_mins(Object minutes) { + return '$minutes মিনিট'; + } + + @override + String get summary_minutes => 'মিনিট'; + + @override + String get summary_listened_to_music => 'সঙ্গীত শুনেছেন'; + + @override + String get summary_songs => 'গান'; + + @override + String get summary_streamed_overall => 'মোট স্ট্রিম'; + + @override + String get summary_owed_to_artists => 'এই মাসে\nশিল্পীদেরকে ঋণী'; + + @override + String get summary_artists => 'শিল্পীর'; + + @override + String get summary_music_reached_you => 'আপনার কাছে পৌঁছেছে সঙ্গীত'; + + @override + String get summary_full_albums => 'সম্পূর্ণ অ্যালবাম'; + + @override + String get summary_got_your_love => 'আপনার ভালোবাসা পেয়েছে'; + + @override + String get summary_playlists => 'প্লেলিস্ট'; + + @override + String get summary_were_on_repeat => 'পুনরাবৃত্তিতে ছিল'; + + @override + String total_money(Object money) { + return 'মোট $money'; + } + + @override + String get webview_not_found => 'ওয়েবভিউ পাওয়া যায়নি'; + + @override + String get webview_not_found_description => + 'আপনার ডিভাইসে কোনো ওয়েবভিউ রানটাইম ইনস্টল করা নেই।\nযদি ইনস্টল থাকে, তা নিশ্চিত করুন যে এটি environment PATH এ রয়েছে\n\nইনস্টল করার পর, অ্যাপটি পুনরায় চালু করুন'; + + @override + String get unsupported_platform => 'সমর্থিত প্ল্যাটফর্ম নয়'; + + @override + String get cache_music => 'ক্যাশে সংগীত'; + + @override + String get open => 'খুলুন'; + + @override + String get cache_folder => 'ক্যাশে ফোল্ডার'; + + @override + String get export => 'রপ্তানি'; + + @override + String get clear_cache => 'ক্যাশে পরিষ্কার'; + + @override + String get clear_cache_confirmation => 'আপনি কি ক্যাশে পরিষ্কার করতে চান?'; + + @override + String get export_cache_files => 'ক্যাশে ফাইল রপ্তানি'; + + @override + String found_n_files(Object count) { + return '$count টি ফাইল পাওয়া গেছে'; + } + + @override + String get export_cache_confirmation => + 'আপনি কি এই ফাইলগুলি রপ্তানি করতে চান'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported টি ফাইল রপ্তানি করা হয়েছে $files এর মধ্যে'; + } + + @override + String get undo => 'পূর্বাবস্থায় ফিরুন'; + + @override + String get download_all => 'সব ডাউনলোড করুন'; + + @override + String get add_all_to_playlist => 'সব প্লেলিস্টে যোগ করুন'; + + @override + String get add_all_to_queue => 'সব কিউতে যোগ করুন'; + + @override + String get play_all_next => 'সব পরবর্তী খেলুন'; + + @override + String get pause => 'বিরতি'; + + @override + String get view_all => 'সব দেখুন'; + + @override + String get no_tracks_added_yet => 'এখনও কোনো ট্র্যাক যোগ করা হয়নি মনে হচ্ছে'; + + @override + String get no_tracks => 'এখানে কোনো ট্র্যাক নেই মনে হচ্ছে'; + + @override + String get no_tracks_listened_yet => 'এখনও কিছু শোনা হয়নি মনে হচ্ছে'; + + @override + String get not_following_artists => 'আপনি কোনো শিল্পীকে অনুসরণ করছেন না'; + + @override + String get no_favorite_albums_yet => + 'এখনও কোনো অ্যালবাম প্রিয় তালিকায় যোগ করা হয়নি মনে হচ্ছে'; + + @override + String get no_logs_found => 'কোনো লগ পাওয়া যায়নি'; + + @override + String get youtube_engine => 'ইউটিউব ইঞ্জিন'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine ইনস্টল করা নেই'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine আপনার সিস্টেমে ইনস্টল করা নেই।'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'এটি PATH ভেরিয়েবলে উপলব্ধ কিনা নিশ্চিত করুন অথবা\nনীচে $engine এক্সিকিউটেবল এর পূর্ণপথ সেট করুন'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/Unix-এর মতো অপারেটিং সিস্টেমে, .zshrc/.bashrc/.bash_profile ইত্যাদিতে পাথ সেট করা কাজ করবে না।\nআপনাকে শেল কনফিগারেশন ফাইলে পাথ সেট করতে হবে'; + + @override + String get download => 'ডাউনলোড'; + + @override + String get file_not_found => 'ফাইল পাওয়া যায়নি'; + + @override + String get custom => 'কাস্টম'; + + @override + String get add_custom_url => 'কাস্টম URL যোগ করুন'; + + @override + String get edit_port => 'পোর্ট সম্পাদনা করুন'; + + @override + String get port_helper_msg => + 'ডিফল্ট হল -1 যা এলোমেলো সংখ্যা নির্দেশ করে। যদি আপনার ফায়ারওয়াল কনফিগার করা থাকে, তবে এটি সেট করা সুপারিশ করা হয়।'; + + @override + String connect_request(Object client) { + return '$client কে সংযোগ করতে অনুমতি দেবেন?'; + } + + @override + String get connection_request_denied => + 'সংযোগ অস্বীকৃত। ব্যবহারকারী প্রবেশাধিকার অস্বীকার করেছে।'; + + @override + String get an_error_occurred => 'একটি ত্রুটি ঘটেছে'; + + @override + String get copy_to_clipboard => 'ক্লিপবোর্ডে কপি করুন'; + + @override + String get view_logs => 'লগ দেখুন'; + + @override + String get retry => 'পুনরায় চেষ্টা করুন'; + + @override + String get no_default_metadata_provider_selected => + 'আপনি কোনো ডিফল্ট মেটাডেটা প্রদানকারী সেট করেননি'; + + @override + String get manage_metadata_providers => 'মেটাডেটা প্রদানকারীগণ পরিচালনা করুন'; + + @override + String get open_link_in_browser => 'লিংকটি ব্রাউজারে খুলবেন?'; + + @override + String get do_you_want_to_open_the_following_link => + 'নিচের লিংকটি খুলতে চান?'; + + @override + String get unsafe_url_warning => + 'অবিশ্বাসযোগ্য উৎস থেকে লিংক খোলা নিরাপদ নাও হতে পারে। সতর্ক থাকুন!\nআপনি এটি ক্লিপবোর্ডে কপি করতে পারেন।'; + + @override + String get copy_link => 'লিংক কপি করুন'; + + @override + String get building_your_timeline => + 'আপনার শোনার ধারা অনুযায়ী টাইমলাইন তৈরি করা হচ্ছে...'; + + @override + String get official => 'সরকারি'; + + @override + String author_name(Object author) { + return 'লেখক: $author'; + } + + @override + String get third_party => 'তৃতীয় পক্ষ'; + + @override + String get plugin_requires_authentication => 'প্লাগইনটি প্রমাণীকরণ প্রয়োজন'; + + @override + String get update_available => 'হালনাগাদ উপলব্ধ'; + + @override + String get supports_scrobbling => 'স্ক্রোব্বলিং সমর্থিত'; + + @override + String get plugin_scrobbling_info => + 'এই প্লাগইনটি আপনার সঙ্গীত স্ক্রোব্বল করে আপনার শোনা ইতিহাস তৈরি করে।'; + + @override + String get default_metadata_source => 'ডিফল্ট মেটাডেটা উৎস'; + + @override + String get set_default_metadata_source => 'ডিফল্ট মেটাডেটা উৎস সেট করুন'; + + @override + String get default_audio_source => 'ডিফল্ট অডিও উৎস'; + + @override + String get set_default_audio_source => 'ডিফল্ট অডিও উৎস সেট করুন'; + + @override + String get set_default => 'ডিফল্ট হিসাবে নির্ধারণ করুন'; + + @override + String get support => 'সমর্থন'; + + @override + String get support_plugin_development => 'প্লাগইন উন্নয়নকে সমর্থন করুন'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API-তে অ্যাক্সেস করতে পারে'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'আপনি কি এই প্লাগইন ইনস্টল করতে চান?'; + + @override + String get third_party_plugin_warning => + 'এই প্লাগইন একটি তৃতীয় পক্ষের রেপোজিটরির। ইনস্টল করার আগে উৎস বিশ্বস্ত কিনা নিশ্চিত করুন।'; + + @override + String get author => 'লেখক'; + + @override + String get this_plugin_can_do_following => 'এই প্লাগইন নিচের কাজ করতে পারে'; + + @override + String get install => 'ইনস্টল করুন'; + + @override + String get install_a_metadata_provider => + 'একটি মেটাডেটা প্রদানকারী ইনস্টল করুন'; + + @override + String get no_tracks_playing => 'বর্তমানে কোনো ট্র্যাক শোনা হচ্ছে না'; + + @override + String get synced_lyrics_not_available => + 'এই গানের জন্য সিঙ্ক্রোনাইজড লিরিক্স পাওয়া যায় না। অনুগ্রহ করে ব্যবহার করুন'; + + @override + String get plain_lyrics => 'সহজ লিরিক্স'; + + @override + String get tab_instead => 'তার পরিবর্তে ট্যাব ব্যবহার করুন।'; + + @override + String get disclaimer => 'অস্বীকৃতি'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube দল কোনো “তৃতীয় পক্ষ” প্লাগইনের জন্য কোনো (আইনগত সহ) দায়িত্ব নেয় না। নিজের বিপদে ব্যবহার করুন। কোনো বাগ/সমস্যা হলে প্লাগইন রেপোজিটরিতে জানাতে অনুরোধ করা হচ্ছে।\n\nযদি কোনো “তৃতীয় পক্ষ” প্লাগইন কোনো পরিষেবা/আইনগত সংস্থার ToS/DMCA ভূঙ্গ করে, অনুগ্রহ করে “তৃতীয় পক্ষ” প্লাগইনের লেখক বা হোস্টিং প্ল্যাটফর্মে (যেমন GitHub/Codeberg) পদক্ষেপ নিতে বলুন। “তৃতীয় পক্ষ” লেবেলযুক্ত যুক্তিগুলি সকলই পাবলিক/কমিউনিটি দ্বারা রক্ষণাবেক্ষণ করা হয়; আমরা সেগুলি কিউরেট করি না, তাই আমরা কোনো পদক্ষেপ নিতে পারি না।\n\n'; + + @override + String get input_does_not_match_format => + 'ইনপুট প্রয়োজনীয় ফরম্যাটের সাথে মেলে না'; + + @override + String get plugins => 'প্লাগইন'; + + @override + String get paste_plugin_download_url => + 'ডাউনলোড URL বা GitHub/Codeberg রিপো URL বা .smplug ফাইলের সরাসরি লিঙ্ক পেস্ট করুন'; + + @override + String get download_and_install_plugin_from_url => + 'URL থেকে প্লাগইন ডাউনলোড এবং ইনস্টল করুন'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'প্লাগইন যোগ করতে ব্যর্থ: $error'; + } + + @override + String get upload_plugin_from_file => 'ফাইল থেকে প্লাগইন আপলোড করুন'; + + @override + String get installed => 'ইনস্টল করা হয়েছে'; + + @override + String get available_plugins => 'উপলব্ধ প্লাগইনগুলো'; + + @override + String get configure_plugins => + 'আপনার নিজের মেটাডেটা প্রদানকারী এবং অডিও উৎস প্লাগইন কনফিগার করুন'; + + @override + String get audio_scrobblers => 'অডিও স্ক্রোব্বলার্স'; + + @override + String get scrobbling => 'স্ক্রোব্বলিং'; + + @override + String get source => 'উৎস: '; + + @override + String get uncompressed => 'অ-সংকুচিত'; + + @override + String get dab_music_source_description => + 'অডিওফাইলদের জন্য। উচ্চ-মানের/লসলেস অডিও স্ট্রিম প্রদান করে। সঠিক ISRC ভিত্তিক ট্র্যাক ম্যাচিং।'; +} diff --git a/lib/l10n/generated/app_localizations_ca.dart b/lib/l10n/generated/app_localizations_ca.dart new file mode 100644 index 00000000..a4f587c9 --- /dev/null +++ b/lib/l10n/generated/app_localizations_ca.dart @@ -0,0 +1,1579 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Catalan Valencian (`ca`). +class AppLocalizationsCa extends AppLocalizations { + AppLocalizationsCa([String locale = 'ca']) : super(locale); + + @override + String get guest => 'Convidat'; + + @override + String get browse => 'Explorar'; + + @override + String get search => 'Cercar'; + + @override + String get library => 'Biblioteca'; + + @override + String get lyrics => 'Lletres'; + + @override + String get settings => 'Configuració'; + + @override + String get genre_categories_filter => 'Filtrar categories o gèneres...'; + + @override + String get genre => 'Gènere'; + + @override + String get personalized => 'Personalizat'; + + @override + String get featured => 'Destacat'; + + @override + String get new_releases => 'Nous Llançaments'; + + @override + String get songs => 'Cançons'; + + @override + String playing_track(Object track) { + return 'Reproduint $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Això eliminarà la llista actual. S\'eliminaran $track_length cançons.\n¿Vol continuar?'; + } + + @override + String get load_more => 'Carregar més'; + + @override + String get playlists => 'Llistes de reproducció'; + + @override + String get artists => 'Artistes'; + + @override + String get albums => 'Àlbums'; + + @override + String get tracks => 'Cançons'; + + @override + String get downloads => 'Descàrregues'; + + @override + String get filter_playlists => 'Filtrar les seves llistes de reproducció...'; + + @override + String get liked_tracks => 'Cançons Preferides'; + + @override + String get liked_tracks_description => 'Totes les seves cançons preferides'; + + @override + String get playlist => 'Llista de reproducció'; + + @override + String get create_a_playlist => 'Crear una llista de reproducció'; + + @override + String get update_playlist => 'Actualitzar la llista de reproducció'; + + @override + String get create => 'Crear'; + + @override + String get cancel => 'Cancel·lar'; + + @override + String get update => 'Actualitzar'; + + @override + String get playlist_name => 'Nom de la llista'; + + @override + String get name_of_playlist => 'Nom de la lista'; + + @override + String get description => 'Descripció'; + + @override + String get public => 'Pública'; + + @override + String get collaborative => 'Col·laborativa'; + + @override + String get search_local_tracks => 'Cercar cançons locals...'; + + @override + String get play => 'Reproduir'; + + @override + String get delete => 'Eliminar'; + + @override + String get none => 'Cap'; + + @override + String get sort_a_z => 'Ordenar de la A a la Z'; + + @override + String get sort_z_a => 'Ordenar de la Z a la A'; + + @override + String get sort_artist => 'Ordenar per Artista'; + + @override + String get sort_album => 'Ordenar per Àlbum'; + + @override + String get sort_duration => 'Ordenar per Durada'; + + @override + String get sort_tracks => 'Ordenar Cançons'; + + @override + String currently_downloading(Object tracks_length) { + return 'Descàrrega en curs ($tracks_length)'; + } + + @override + String get cancel_all => 'Cancel·lar todo'; + + @override + String get filter_artist => 'Filtrar artistes...'; + + @override + String followers(Object followers) { + return '$followers Seguidors'; + } + + @override + String get add_artist_to_blacklist => 'Afegir artista a la llista negra'; + + @override + String get top_tracks => 'Millors Cançons'; + + @override + String get fans_also_like => 'Als fans també els hi agrada'; + + @override + String get loading => 'Carregant...'; + + @override + String get artist => 'Artista'; + + @override + String get blacklisted => 'A la llista negra'; + + @override + String get following => 'Seguint'; + + @override + String get follow => 'Seguir'; + + @override + String get artist_url_copied => 'URL de l\'artista copiada al porta-retalls '; + + @override + String added_to_queue(Object tracks) { + return '$tracks cançons afegides a la llista'; + } + + @override + String get filter_albums => 'Filtrar àlbums...'; + + @override + String get synced => 'Sincronitzat'; + + @override + String get plain => 'Normal'; + + @override + String get shuffle => 'Aleatori'; + + @override + String get search_tracks => 'Buscar cançons...'; + + @override + String get released => 'Publicat'; + + @override + String error(Object error) { + return 'Error $error'; + } + + @override + String get title => 'Títul'; + + @override + String get time => 'Duració'; + + @override + String get more_actions => 'Més accios'; + + @override + String download_count(Object count) { + return 'Descarregar ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Afegir ($count) a la llista de reproducció'; + } + + @override + String add_count_to_queue(Object count) { + return 'Agregar ($count) a la llista'; + } + + @override + String play_count_next(Object count) { + return 'Reproduir ($count) a continuació'; + } + + @override + String get album => 'Àlbum'; + + @override + String copied_to_clipboard(Object data) { + return '$data copiado al porta-retalls'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Afegir $track a les llistes de reproducció següents'; + } + + @override + String get add => 'Afegir'; + + @override + String added_track_to_queue(Object track) { + return '$track afegida a la llista'; + } + + @override + String get add_to_queue => 'Afegir a la llista'; + + @override + String track_will_play_next(Object track) { + return '$track es reproduirà a continuació'; + } + + @override + String get play_next => 'Reproduir a continuació'; + + @override + String removed_track_from_queue(Object track) { + return '$track eliminada de la llista'; + } + + @override + String get remove_from_queue => 'Eliminar de la llista'; + + @override + String get remove_from_favorites => 'Eliminar de preferits'; + + @override + String get save_as_favorite => 'Guardar a preferits'; + + @override + String get add_to_playlist => 'Afegir a la llista de reproducció'; + + @override + String get remove_from_playlist => 'Eliminar de la llista de reproducció'; + + @override + String get add_to_blacklist => 'Afegir a la llista negra'; + + @override + String get remove_from_blacklist => 'Eliminar de la llista negra'; + + @override + String get share => 'Compartir'; + + @override + String get mini_player => 'Reproductor Petit'; + + @override + String get slide_to_seek => 'Lliscar per cercar endavant o endarrere'; + + @override + String get shuffle_playlist => 'Mesclar la llista de reproducció'; + + @override + String get unshuffle_playlist => 'No mesclar la llista de reproducció'; + + @override + String get previous_track => 'Cançó anterior'; + + @override + String get next_track => 'Canço següent'; + + @override + String get pause_playback => 'Pausar reproducció'; + + @override + String get resume_playback => 'Continuar reproducció'; + + @override + String get loop_track => 'Repetir canço'; + + @override + String get no_loop => 'Sense repetició'; + + @override + String get repeat_playlist => 'Repetir la llista de reproducció'; + + @override + String get queue => 'Llista'; + + @override + String get alternative_track_sources => 'Fonts alternatives de cançons'; + + @override + String get download_track => 'Descarregar cançó'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks cançons a la llista'; + } + + @override + String get clear_all => 'Netejar tot'; + + @override + String get show_hide_ui_on_hover => + 'Mostrar/Ocultar interfície al passar el cursor'; + + @override + String get always_on_top => 'Sempre visible'; + + @override + String get exit_mini_player => 'Sortir del reproductor petit'; + + @override + String get download_location => 'Ubicació de descàrregues'; + + @override + String get local_library => 'Biblioteca local'; + + @override + String get add_library_location => 'Afegeix a la biblioteca'; + + @override + String get remove_library_location => 'Elimina de la biblioteca'; + + @override + String get account => 'Compte'; + + @override + String get logout => 'Tancar sessió'; + + @override + String get logout_of_this_account => 'Tancar sessió d\'aquest compte'; + + @override + String get language_region => 'Idioma i Regió'; + + @override + String get language => 'Idioma'; + + @override + String get system_default => 'Predeterminat del sistema'; + + @override + String get market_place_region => 'Regió de la botiga'; + + @override + String get recommendation_country => 'País de recomanació'; + + @override + String get appearance => 'Apariència'; + + @override + String get layout_mode => 'Mode de disseny'; + + @override + String get override_layout_settings => + 'Anul·leu la configuració del mode de disseny responsiu'; + + @override + String get adaptive => 'Adaptable'; + + @override + String get compact => 'Compacte'; + + @override + String get extended => 'Extès'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Fosc'; + + @override + String get light => 'Clar'; + + @override + String get system => 'Sistema'; + + @override + String get accent_color => 'Color d\'accent'; + + @override + String get sync_album_color => 'Sincronitzar color de l\'àlbum'; + + @override + String get sync_album_color_description => + 'Utilitza el color dominant de l\'álbum com a color d\'accent'; + + @override + String get playback => 'Reproducció'; + + @override + String get audio_quality => 'Qualitat d\'àudio'; + + @override + String get high => 'Alta'; + + @override + String get low => 'Baixa'; + + @override + String get pre_download_play => 'Descàrrega prèvia i reproduir'; + + @override + String get pre_download_play_description => + 'En lloc de transmetre l\'àudio, descarrega bytes i ho reprodueix (recomendat per usuaris amb un bon ample de banda)'; + + @override + String get skip_non_music => + 'Ometre segments que no son música (SponsorBlock)'; + + @override + String get blacklist_description => 'Cançons i artistes de la llista negra'; + + @override + String get wait_for_download_to_finish => + 'Si us plau, esperi que acabi la descàrrega actual'; + + @override + String get desktop => 'Escriptori'; + + @override + String get close_behavior => 'Comportament al tancar'; + + @override + String get close => 'Tancar'; + + @override + String get minimize_to_tray => 'Minimizar a la safata del sistema'; + + @override + String get show_tray_icon => 'Mostrar icona a la safata del sistema'; + + @override + String get about => 'Sobre'; + + @override + String get u_love_spotube => 'Sabem que li encanta Spotube'; + + @override + String get check_for_updates => 'Buscar actualitzacions'; + + @override + String get about_spotube => 'Sobre Spotube'; + + @override + String get blacklist => 'Llista negra'; + + @override + String get please_sponsor => 'Si us plau, patrocina/dona'; + + @override + String get spotube_description => + 'Spotube, un client lleuger, multiplataforma i gratuït de Spotify'; + + @override + String get version => 'Versió'; + + @override + String get build_number => 'Número de compilació'; + + @override + String get founder => 'Fundador'; + + @override + String get repository => 'Repositori'; + + @override + String get bug_issues => 'Errors i problemes'; + + @override + String get made_with => 'Fet amb ❤️ a Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Llicència'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'No es preocupi, les seves credencials no seran recollides ni compartides amb ningú'; + + @override + String get know_how_to_login => 'No sap com fer-ho?'; + + @override + String get follow_step_by_step_guide => 'Segueixi la guia pas a pas'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Si us plau, completi tots els camps'; + + @override + String get submit => 'Enviar'; + + @override + String get exit => 'Sortir'; + + @override + String get previous => 'Anterior'; + + @override + String get next => 'Següent'; + + @override + String get done => 'Fet'; + + @override + String get step_1 => 'Pas 1'; + + @override + String get first_go_to => 'Primer, vagi a'; + + @override + String get something_went_wrong => 'Quelcom ha sortit malament'; + + @override + String get piped_instance => 'Instància del servidor Piped'; + + @override + String get piped_description => + 'La instància del servidor Piped a utilitzar per la coincidència de cançons'; + + @override + String get piped_warning => + 'Algunes poden no funcionar bé, utilitzi-les sota el seu propi risc'; + + @override + String get invidious_instance => 'Instància del servidor Invidious'; + + @override + String get invidious_description => + 'La instància del servidor Invidious per fer coincidir pistes'; + + @override + String get invidious_warning => + 'Algunes instàncies podrien no funcionar bé. Feu-les servir sota la vostra responsabilitat'; + + @override + String get generate => 'Generar'; + + @override + String track_exists(Object track) { + return 'La cançó $track ja existeix'; + } + + @override + String get replace_downloaded_tracks => + 'Substituir totes les cançons descarregades'; + + @override + String get skip_download_tracks => + 'Ometre la descàrrega de totes les cançons descarregades'; + + @override + String get do_you_want_to_replace => 'Vol substituir la cançó existent?'; + + @override + String get replace => 'Substituir'; + + @override + String get skip => 'Ometre'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Seleccionar fins$count $type'; + } + + @override + String get select_genres => 'Seleccionar Gèneres'; + + @override + String get add_genres => 'Afegir Gèneres'; + + @override + String get country => 'País'; + + @override + String get number_of_tracks_generate => 'Número de cançons a generar'; + + @override + String get acousticness => 'Acústica'; + + @override + String get danceability => 'Ballabilitat'; + + @override + String get energy => 'Energia'; + + @override + String get instrumentalness => 'Instrumental'; + + @override + String get liveness => 'En viu'; + + @override + String get loudness => 'Sonoritat'; + + @override + String get speechiness => 'Parla'; + + @override + String get valence => 'Valencia'; + + @override + String get popularity => 'Popularidad'; + + @override + String get key => 'To'; + + @override + String get duration => 'Duració (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'Signatura de temps'; + + @override + String get short => 'Curt'; + + @override + String get medium => 'Mig'; + + @override + String get long => 'Llarg'; + + @override + String get min => 'Mín.'; + + @override + String get max => 'Màx.'; + + @override + String get target => 'Objetiu'; + + @override + String get moderate => 'Moderat'; + + @override + String get deselect_all => 'Desseleccionar tot'; + + @override + String get select_all => 'Seleccionar tot'; + + @override + String get are_you_sure => 'Està segur?'; + + @override + String get generating_playlist => + 'Generant la seva llista de reproducció personalitzada...'; + + @override + String selected_count_tracks(Object count) { + return 'Cançons $count seleccionades'; + } + + @override + String get download_warning => + 'Si descarrega totes les cançons de cop, està piratejant música clarament i causant dany a la societat creativa de la música. Espero que sigui conscient d\'això i sempre intenti respectar i recolzar la forta feina dels artístes'; + + @override + String get download_ip_ban_warning => + 'Per cert, la seva IP pot ser bloquejada a YouTube degut a solicituds de descàrrega excessives. El bloqueig d\'IP vol dir que no podrà utilitzar YouTube (fins i tot si ha iniciat sessió) durant un mínim de 2-3 meses desde esa dirección IP. I Spotube no es fa responsable si això succeeix en alguna ocasió'; + + @override + String get by_clicking_accept_terms => + 'Al fer clic a \'Acceptar\', acepta els següents termes:'; + + @override + String get download_agreement_1 => + 'Se que estic piratejant música. Sóc dolent'; + + @override + String get download_agreement_2 => + 'Recolzaré l\'artista quan pugui i només ho faig perquè no tinc diners per comprar el seu art'; + + @override + String get download_agreement_3 => + 'Sóc completament conscient que la meva IP pot ser bloqueada per YouTube i no responsabilizo a Spotube ni als seus propietaris/contribuents per qualsevol incident causat per la meva acció actual'; + + @override + String get decline => 'Rebutjar'; + + @override + String get accept => 'Acceptar'; + + @override + String get details => 'Detalls'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Canal'; + + @override + String get likes => 'M\'agrada'; + + @override + String get dislikes => 'No m\'agrada'; + + @override + String get views => 'Vistes'; + + @override + String get streamUrl => 'URL del streaming'; + + @override + String get stop => 'Parar'; + + @override + String get sort_newest => 'Ordenar per més noves'; + + @override + String get sort_oldest => 'Ordenar per més antigues'; + + @override + String get sleep_timer => 'Temporitzador d\'apagat'; + + @override + String mins(Object minutes) { + return '$minutes minuts'; + } + + @override + String hours(Object hours) { + return '$hours hores'; + } + + @override + String hour(Object hours) { + return '$hours hora'; + } + + @override + String get custom_hours => 'Hores personalitzades'; + + @override + String get logs => 'Registres'; + + @override + String get developers => 'Desenvolupadors'; + + @override + String get not_logged_in => 'No ha iniciat sesió'; + + @override + String get search_mode => 'Mode de cerca'; + + @override + String get audio_source => 'Font d\'àudio'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => 'Error al xifrar'; + + @override + String get encryption_failed_warning => + 'Spotube utilitza el xifrado per emmagatzemar les seves dades de forma segura. Però ha fallat. Per tant, tornarà a un emmagatzament no segur\nSi estè utilizant Linux, asseguri\'s de tenir instal·lats els serveis secrets com gnome-keyring, kde-wallet i keepassxc'; + + @override + String get querying_info => 'Consultant informació...'; + + @override + String get piped_api_down => 'La API de Piped no està operativa'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'La instància de Piped $pipedInstance no està operativa en aquest moment\n\nCanvieu la instància o canvieu el \'Tipus d\'API\' a l\'API oficial de YouTube\n\nAssegureu-vos de reiniciar l\'aplicació després del canvi'; + } + + @override + String get you_are_offline => 'Actualment no teniu connexió a internet'; + + @override + String get connection_restored => 'S\'ha restablert la connexió a internet'; + + @override + String get use_system_title_bar => 'Utilitza la barra de títol del sistema'; + + @override + String get crunching_results => 'Processant resultats...'; + + @override + String get search_to_get_results => 'Cerca per obtenir resultats'; + + @override + String get use_amoled_mode => 'Utilitza el mode AMOLED'; + + @override + String get pitch_dark_theme => 'Tema de dart negre intens'; + + @override + String get normalize_audio => 'Normalitza l\'àudio'; + + @override + String get change_cover => 'Canvia la coberta'; + + @override + String get add_cover => 'Afegeix una coberta'; + + @override + String get restore_defaults => 'Restaura els valors per defecte'; + + @override + String get download_music_format => 'Format de descàrrega de música'; + + @override + String get streaming_music_format => + 'Format de reproducció de música en temps real'; + + @override + String get download_music_quality => 'Qualitat de descàrrega de música'; + + @override + String get streaming_music_quality => + 'Qualitat de reproducció de música en temps real'; + + @override + String get login_with_lastfm => 'Inicia la sessió amb Last.fm'; + + @override + String get connect => 'Connecta'; + + @override + String get disconnect_lastfm => 'Desconnecta de Last.fm'; + + @override + String get disconnect => 'Desconnecta'; + + @override + String get username => 'Nom d\'usuari'; + + @override + String get password => 'Contrasenya'; + + @override + String get login => 'Inicia la sessió'; + + @override + String get login_with_your_lastfm => + 'Inicia la sessió amb el teu compte de Last.fm'; + + @override + String get scrobble_to_lastfm => 'Scrobble a Last.fm'; + + @override + String get go_to_album => 'Anar a l\'àlbum'; + + @override + String get discord_rich_presence => 'Presència rica de Discord'; + + @override + String get browse_all => 'Navega per tot'; + + @override + String get genres => 'Gèneres'; + + @override + String get explore_genres => 'Explora els gèneres'; + + @override + String get friends => 'Amics'; + + @override + String get no_lyrics_available => + 'Ho sentim, no es poden trobar les lletres d\'aquesta pista'; + + @override + String get start_a_radio => 'Inicia una ràdio'; + + @override + String get how_to_start_radio => 'Com vols començar la ràdio?'; + + @override + String get replace_queue_question => + 'Voleu substituir la cua actual o afegir-hi?'; + + @override + String get endless_playback => 'Reproducció infinita'; + + @override + String get delete_playlist => 'Suprimeix la llista de reproducció'; + + @override + String get delete_playlist_confirmation => + 'Esteu segur que voleu suprimir aquesta llista de reproducció?'; + + @override + String get local_tracks => 'Pistes locals'; + + @override + String get local_tab => 'Local'; + + @override + String get song_link => 'Enllaç de la cançó'; + + @override + String get skip_this_nonsense => 'Omet aquesta tonteria'; + + @override + String get freedom_of_music => '“Llibertat de la música”'; + + @override + String get freedom_of_music_palm => + '“Llibertat de la música a la palma de la mà”'; + + @override + String get get_started => 'Comencem'; + + @override + String get youtube_source_description => 'Recomanat i funciona millor.'; + + @override + String get piped_source_description => + 'Et sents lliure? El mateix que YouTube però més lliure.'; + + @override + String get jiosaavn_source_description => + 'El millor per a la regió del sud d\'Àsia.'; + + @override + String get invidious_source_description => + 'Similar a Piped però amb més disponibilitat'; + + @override + String highest_quality(Object quality) { + return 'Qualitat més alta: $quality'; + } + + @override + String get select_audio_source => 'Seleccioneu la font d\'àudio'; + + @override + String get endless_playback_description => + 'Afegiu automàticament noves cançons\nal final de la cua'; + + @override + String get choose_your_region => 'Trieu la vostra regió'; + + @override + String get choose_your_region_description => + 'Això ajudarà a Spotube a mostrar-vos el contingut adequat\nper a la vostra ubicació.'; + + @override + String get choose_your_language => 'Trieu el vostre idioma'; + + @override + String get help_project_grow => 'Ajuda a fer créixer aquest projecte'; + + @override + String get help_project_grow_description => + 'Spotube és un projecte de codi obert. Podeu ajudar a fer créixer aquest projecte contribuint al projecte, informant d\'errors o suggerint noves funcionalitats.'; + + @override + String get contribute_on_github => 'Contribueix a GitHub'; + + @override + String get donate_on_open_collective => 'Fes una donació a Open Collective'; + + @override + String get browse_anonymously => 'Navega de manera anònima'; + + @override + String get enable_connect => 'Habilita la connexió'; + + @override + String get enable_connect_description => + 'Controla Spotube des d\'altres dispositius'; + + @override + String get devices => 'Dispositius'; + + @override + String get select => 'Selecciona'; + + @override + String connect_client_alert(Object client) { + return 'Estàs sent controlat per $client'; + } + + @override + String get this_device => 'Aquest dispositiu'; + + @override + String get remote => 'Remot'; + + @override + String get stats => 'Estadístiques'; + + @override + String and_n_more(Object count) { + return 'i $count més'; + } + + @override + String get recently_played => 'Reproduït recentment'; + + @override + String get browse_more => 'Navega més'; + + @override + String get no_title => 'Sense títol'; + + @override + String get not_playing => 'No s\'està reproduint'; + + @override + String get epic_failure => 'Fracàs èpic!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Afegit $tracks_length pistes a la cua'; + } + + @override + String get spotube_has_an_update => 'Spotube té una actualització'; + + @override + String get download_now => 'Descarregar ara'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum ha estat publicat'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version ha estat publicat'; + } + + @override + String get read_the_latest => 'Llegeix el més recent'; + + @override + String get release_notes => 'notes de la versió'; + + @override + String get pick_color_scheme => 'Tria l\'esquema de colors'; + + @override + String get save => 'Desar'; + + @override + String get choose_the_device => 'Tria el dispositiu:'; + + @override + String get multiple_device_connected => + 'Hi ha diversos dispositius connectats.\nTria el dispositiu on vols realitzar aquesta acció'; + + @override + String get nothing_found => 'No s\'ha trobat res'; + + @override + String get the_box_is_empty => 'La caixa està buida'; + + @override + String get top_artists => 'Millors artistes'; + + @override + String get top_albums => 'Millors àlbums'; + + @override + String get this_week => 'Aquesta setmana'; + + @override + String get this_month => 'Aquest mes'; + + @override + String get last_6_months => 'Últims 6 mesos'; + + @override + String get this_year => 'Aquest any'; + + @override + String get last_2_years => 'Últims 2 anys'; + + @override + String get all_time => 'Tots els temps'; + + @override + String powered_by_provider(Object providerName) { + return 'Funciona amb $providerName'; + } + + @override + String get email => 'Correu electrònic'; + + @override + String get profile_followers => 'Seguidors'; + + @override + String get birthday => 'Aniversari'; + + @override + String get subscription => 'Subscripció'; + + @override + String get not_born => 'No ha nascut'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Perfil'; + + @override + String get no_name => 'Sense nom'; + + @override + String get edit => 'Editar'; + + @override + String get user_profile => 'Perfil d\'usuari'; + + @override + String count_plays(Object count) { + return '$count reproduccions'; + } + + @override + String get streaming_fees_hypothetical => + 'Comissions de streaming (hipotètic)'; + + @override + String get minutes_listened => 'minuts escoltats'; + + @override + String get streamed_songs => 'cançons reproduïdes'; + + @override + String count_streams(Object count) { + return '$count reproduccions'; + } + + @override + String get owned_by_you => 'De la teva propietat'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'S\'ha copiat $shareUrl al porta-retalls'; + } + + @override + String get hipotetical_calculation => + '*Això està calculat en funció d’un pagament mitjà per reproducció de 0,003–0,005 USD en plataformes de reproducció musical en línia. És un càlcul hipotètic per ajudar l’usuari a entendre quant hauria pagat als artistes si hagués escoltat la seva cançó en diferents plataformes.'; + + @override + String count_mins(Object minutes) { + return '$minutes minuts'; + } + + @override + String get summary_minutes => 'minuts'; + + @override + String get summary_listened_to_music => 'has escoltat música'; + + @override + String get summary_songs => 'cançons'; + + @override + String get summary_streamed_overall => 'reproduït en general'; + + @override + String get summary_owed_to_artists => 'degut als artistes\nAquest mes'; + + @override + String get summary_artists => 'artistes'; + + @override + String get summary_music_reached_you => 'La música t\'ha arribat'; + + @override + String get summary_full_albums => 'Àlbums complets'; + + @override + String get summary_got_your_love => 'ha aconseguit el teu amor'; + + @override + String get summary_playlists => 'llistes de reproducció'; + + @override + String get summary_were_on_repeat => 'estaven en repetició'; + + @override + String total_money(Object money) { + return 'total $money'; + } + + @override + String get webview_not_found => 'No s\'ha trobat el Webview'; + + @override + String get webview_not_found_description => + 'No hi ha cap temps d\'execució de Webview instal·lat al dispositiu.\nSi està instal·lat, assegureu-vos que estigui en el environment PATH\n\nDesprés d\'instal·lar-lo, reinicieu l\'aplicació'; + + @override + String get unsupported_platform => 'Plataforma no compatible'; + + @override + String get cache_music => 'Música en caché'; + + @override + String get open => 'Obrir'; + + @override + String get cache_folder => 'Carpeta de caché'; + + @override + String get export => 'Exportar'; + + @override + String get clear_cache => 'Netejar caché'; + + @override + String get clear_cache_confirmation => 'Voleu netejar la memòria cau?'; + + @override + String get export_cache_files => 'Exportar arxius en caché'; + + @override + String found_n_files(Object count) { + return 'S\'han trobat $count arxius'; + } + + @override + String get export_cache_confirmation => 'Voleu exportar aquests arxius a'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'S\'han exportat $filesExported de $files arxius'; + } + + @override + String get undo => 'Desfer'; + + @override + String get download_all => 'Descarregar tot'; + + @override + String get add_all_to_playlist => 'Afegir tot a la llista de reproducció'; + + @override + String get add_all_to_queue => 'Afegir tot a la cua'; + + @override + String get play_all_next => 'Reproduir tot a continuació'; + + @override + String get pause => 'Pausa'; + + @override + String get view_all => 'Veure tot'; + + @override + String get no_tracks_added_yet => 'Sembla que encara no has afegit cap pista'; + + @override + String get no_tracks => 'Sembla que no hi ha pistes aquí'; + + @override + String get no_tracks_listened_yet => 'Sembla que no has escoltat res encara'; + + @override + String get not_following_artists => 'No estàs seguint cap artista'; + + @override + String get no_favorite_albums_yet => + 'Sembla que encara no has afegit cap àlbum als teus favorits'; + + @override + String get no_logs_found => 'No s\'han trobat registres'; + + @override + String get youtube_engine => 'Motor de YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine no està instal·lat'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine no està instal·lat al teu sistema.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Assegura\'t que estigui disponible a la variable PATH o\nestableix el camí absolut a l\'executable de $engine a continuació'; + } + + @override + String get youtube_engine_unix_issue_message => + 'En macOS/Linux/Unix com a sistemes operatius, establir el camí a .zshrc/.bashrc/.bash_profile etc. no funcionarà.\nHas de configurar el camí al fitxer de configuració de la shell'; + + @override + String get download => 'Descarregar'; + + @override + String get file_not_found => 'Fitxer no trobat'; + + @override + String get custom => 'Personalitzat'; + + @override + String get add_custom_url => 'Afegir URL personalitzada'; + + @override + String get edit_port => 'Editar port'; + + @override + String get port_helper_msg => + 'El valor per defecte és -1, que indica un número aleatori. Si teniu un tallafoc configurat, es recomana establir-ho.'; + + @override + String connect_request(Object client) { + return 'Permetre que $client es connecti?'; + } + + @override + String get connection_request_denied => + 'Connexió denegada. L\'usuari ha denegat l\'accés.'; + + @override + String get an_error_occurred => 'S’ha produït un error'; + + @override + String get copy_to_clipboard => 'Copiar al porta-retalls'; + + @override + String get view_logs => 'Veure registres'; + + @override + String get retry => 'Tornar-ho a provar'; + + @override + String get no_default_metadata_provider_selected => + 'No has configurat cap proveïdor de metadades predeterminat'; + + @override + String get manage_metadata_providers => 'Gestionar proveïdors de metadades'; + + @override + String get open_link_in_browser => 'Obrir l’enllaç en el navegador?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Vols obrir l’enllaç següent?'; + + @override + String get unsafe_url_warning => + 'Pot ser perillós obrir enllaços de fonts no fiables. Sigues precavís!\nTambé pots copiar l’enllaç al porta-retalls.'; + + @override + String get copy_link => 'Copiar enllaç'; + + @override + String get building_your_timeline => + 'Construint la teva cronologia en funció de les teves escoltes...'; + + @override + String get official => 'Oficial'; + + @override + String author_name(Object author) { + return 'Autor: $author'; + } + + @override + String get third_party => 'Tercers'; + + @override + String get plugin_requires_authentication => + 'El complement requereix autenticació'; + + @override + String get update_available => 'Actualització disponible'; + + @override + String get supports_scrobbling => 'Admet scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Aquest complement fa scrobbling de la teva música per generar l’historial d’escoltes.'; + + @override + String get default_metadata_source => 'Font de metadades per defecte'; + + @override + String get set_default_metadata_source => + 'Estableix la font de metadades per defecte'; + + @override + String get default_audio_source => 'Font d\'àudio per defecte'; + + @override + String get set_default_audio_source => + 'Estableix la font d\'àudio per defecte'; + + @override + String get set_default => 'Establir com a predeterminat'; + + @override + String get support => 'Suport'; + + @override + String get support_plugin_development => + 'Suportar el desenvolupament del complement'; + + @override + String can_access_name_api(Object name) { + return '- Pot accedir a l’API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Vols instal·lar aquest complement?'; + + @override + String get third_party_plugin_warning => + 'Aquest complement prové d’un repositori de tercers. Assegura’t de confiar en la font abans d’instal·lar-lo.'; + + @override + String get author => 'Autor'; + + @override + String get this_plugin_can_do_following => + 'Aquest complement pot fer el següent'; + + @override + String get install => 'Instal·lar'; + + @override + String get install_a_metadata_provider => + 'Instal·lar un proveïdor de metadades'; + + @override + String get no_tracks_playing => 'No s’està reproduint cap pista actualment'; + + @override + String get synced_lyrics_not_available => + 'Les lletres sincronitzades no estan disponibles per a aquesta cançó. Si us plau, usa'; + + @override + String get plain_lyrics => 'Lletres sense format'; + + @override + String get tab_instead => 'en lloc d’això, utilitza la tecla Tab.'; + + @override + String get disclaimer => 'Avís legal'; + + @override + String get third_party_plugin_dmca_notice => + 'L’equip de Spotube no accepta cap responsabilitat (inclosa legal) pels complements de “tercers”.\nFes-los servir sota la teva responsabilitat. Si detectes errors/problemes, informa’ls al repositori del complement.\n\nSi algun complement de “tercers” incompleix els ToS/DMCA d’un servei o entitat legal, contacta amb l’autor del complement o amb la plataforma d’allotjament (per exemple GitHub/Codeberg) per prendre mesures. Els complements etiquetats com a “tercers” són públics i gestionats per la comunitat; no els curatem, per la qual cosa no podem intervenir-hi.\n\n'; + + @override + String get input_does_not_match_format => + 'L’entrada no coincideix amb el format requerit'; + + @override + String get plugins => 'Connectors'; + + @override + String get paste_plugin_download_url => + 'Enllaça l’URL de descàrrega o el repositori de GitHub/Codeberg o l’enllaç directe al fitxer .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Descarrega i instal·la el complement des d’un URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Error en afegir el complement: $error'; + } + + @override + String get upload_plugin_from_file => 'Penja el complement des d’un fitxer'; + + @override + String get installed => 'Instal·lat'; + + @override + String get available_plugins => 'Complements disponibles'; + + @override + String get configure_plugins => + 'Configura els teus propis connectors de proveïdor de metadades i de font d\'àudio'; + + @override + String get audio_scrobblers => 'Scrobblers d’àudio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Font: '; + + @override + String get uncompressed => 'Sense comprimir'; + + @override + String get dab_music_source_description => + 'Per als audiòfils. Ofereix fluxos d\'àudio d\'alta qualitat/sense pèrdua. Coincidència precisa de pistes basada en ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_cs.dart b/lib/l10n/generated/app_localizations_cs.dart new file mode 100644 index 00000000..24d5b34b --- /dev/null +++ b/lib/l10n/generated/app_localizations_cs.dart @@ -0,0 +1,1566 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Czech (`cs`). +class AppLocalizationsCs extends AppLocalizations { + AppLocalizationsCs([String locale = 'cs']) : super(locale); + + @override + String get guest => 'Host'; + + @override + String get browse => 'Procházet'; + + @override + String get search => 'Hledat'; + + @override + String get library => 'Knihovna'; + + @override + String get lyrics => 'Texty'; + + @override + String get settings => 'Nastavení'; + + @override + String get genre_categories_filter => 'Filtrovat kategorie nebo žánry...'; + + @override + String get genre => 'Žánr'; + + @override + String get personalized => 'Personalizované'; + + @override + String get featured => 'Doporučené'; + + @override + String get new_releases => 'Nově vydané'; + + @override + String get songs => 'Skladby'; + + @override + String playing_track(Object track) { + return 'Hraje $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Toto vymaže aktuální frontu. $track_length skladeb bude odstraněno\nChcete pokračovat?'; + } + + @override + String get load_more => 'Načíst více'; + + @override + String get playlists => 'Playlisty'; + + @override + String get artists => 'Umělci'; + + @override + String get albums => 'Alba'; + + @override + String get tracks => 'Skladby'; + + @override + String get downloads => 'Stahování'; + + @override + String get filter_playlists => 'Filtrovat playlisty...'; + + @override + String get liked_tracks => 'Oblíbené skladby'; + + @override + String get liked_tracks_description => 'Všechny vaše oblíbené skladby'; + + @override + String get playlist => 'Seznam skladeb'; + + @override + String get create_a_playlist => 'Vytvořit playlist'; + + @override + String get update_playlist => 'Aktualizovat playlist'; + + @override + String get create => 'Vytvořit'; + + @override + String get cancel => 'Zrušit'; + + @override + String get update => 'Aktualizovat'; + + @override + String get playlist_name => 'Název playlistu'; + + @override + String get name_of_playlist => 'Název playlistu'; + + @override + String get description => 'Popis'; + + @override + String get public => 'Veřejné'; + + @override + String get collaborative => 'Společný'; + + @override + String get search_local_tracks => 'Hledat místní skladby...'; + + @override + String get play => 'Přehrát'; + + @override + String get delete => 'Smazat'; + + @override + String get none => 'Žádné'; + + @override + String get sort_a_z => 'Seřadit od A-Z'; + + @override + String get sort_z_a => 'Seřadit od Z-A'; + + @override + String get sort_artist => 'Seřadit podle umělce'; + + @override + String get sort_album => 'Seřadit podle alba'; + + @override + String get sort_duration => 'Seřadit podle délky'; + + @override + String get sort_tracks => 'Seřadit skladby'; + + @override + String currently_downloading(Object tracks_length) { + return 'Právě se stahuje ($tracks_length)'; + } + + @override + String get cancel_all => 'Zrušit vše'; + + @override + String get filter_artist => 'Filtrovat umělce...'; + + @override + String followers(Object followers) { + return '$followers Sledující'; + } + + @override + String get add_artist_to_blacklist => 'Přidat umělce na černou listinu'; + + @override + String get top_tracks => 'Top skladby'; + + @override + String get fans_also_like => 'Fanoušci mají také rádi'; + + @override + String get loading => 'Načítání...'; + + @override + String get artist => 'Umělec'; + + @override + String get blacklisted => 'Na černé listině'; + + @override + String get following => 'Sleduje'; + + @override + String get follow => 'Sledovat'; + + @override + String get artist_url_copied => 'URL umělce zkopírována do schránky'; + + @override + String added_to_queue(Object tracks) { + return 'Přidáno $tracks skladeb do fronty'; + } + + @override + String get filter_albums => 'Filtrovat alba...'; + + @override + String get synced => 'Synchronizováno'; + + @override + String get plain => 'Jednoduché'; + + @override + String get shuffle => 'Zamíchat'; + + @override + String get search_tracks => 'Hledat skladby...'; + + @override + String get released => 'Vydáno'; + + @override + String error(Object error) { + return 'Chyba $error'; + } + + @override + String get title => 'Název'; + + @override + String get time => 'Čas'; + + @override + String get more_actions => 'Více akcí'; + + @override + String download_count(Object count) { + return 'Stáhnout ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Přidat ($count) do playlistu'; + } + + @override + String add_count_to_queue(Object count) { + return 'Přidat ($count) do fronty'; + } + + @override + String play_count_next(Object count) { + return 'Přehrát ($count) dalších'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return 'Zkopírováno $data do schránky'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Přidat $track do následujících playlistů'; + } + + @override + String get add => 'Přidat'; + + @override + String added_track_to_queue(Object track) { + return 'Přidána skladba $track do fronty'; + } + + @override + String get add_to_queue => 'Přidat do fronty'; + + @override + String track_will_play_next(Object track) { + return '$track se přehraje jako další'; + } + + @override + String get play_next => 'Přehrát další'; + + @override + String removed_track_from_queue(Object track) { + return 'Odstraněna skladba $track z fronty'; + } + + @override + String get remove_from_queue => 'Odstranit z fronty'; + + @override + String get remove_from_favorites => 'Odstranit z oblíbených'; + + @override + String get save_as_favorite => 'Uložit jako oblíbené'; + + @override + String get add_to_playlist => 'Přidat do playlistu'; + + @override + String get remove_from_playlist => 'Odstranit z playlistu'; + + @override + String get add_to_blacklist => 'Přidat na černou listinu'; + + @override + String get remove_from_blacklist => 'Odstranit z černé listiny'; + + @override + String get share => 'Sdílet'; + + @override + String get mini_player => 'Mini přehrávač'; + + @override + String get slide_to_seek => 'Táhněte pro posunutí vpřed nebo vzad'; + + @override + String get shuffle_playlist => 'Zamíchat playlist'; + + @override + String get unshuffle_playlist => 'Zrušit zamíchání playlistu'; + + @override + String get previous_track => 'Předchozí skladba'; + + @override + String get next_track => 'Další skladba'; + + @override + String get pause_playback => 'Pozastavit přehrávání'; + + @override + String get resume_playback => 'Pokračovat v přehrávání'; + + @override + String get loop_track => 'Opakovat skladbu'; + + @override + String get no_loop => 'Žádné opakování'; + + @override + String get repeat_playlist => 'Opakovat playlist'; + + @override + String get queue => 'Fronta'; + + @override + String get alternative_track_sources => 'Alternativní zdroje skladeb'; + + @override + String get download_track => 'Stáhnout skladbu'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks skladeb ve frontě'; + } + + @override + String get clear_all => 'Vymazat vše'; + + @override + String get show_hide_ui_on_hover => 'Zobrazit/Skrýt UI při najetí'; + + @override + String get always_on_top => 'Vždy nahoře'; + + @override + String get exit_mini_player => 'Zavřít mini přehrávač'; + + @override + String get download_location => 'Umístění stahování'; + + @override + String get local_library => 'Místní knihovna'; + + @override + String get add_library_location => 'Přidat do knihovny'; + + @override + String get remove_library_location => 'Odebrat z knihovny'; + + @override + String get account => 'Účet'; + + @override + String get logout => 'Odhlásit se'; + + @override + String get logout_of_this_account => 'Odhlásit se z tohoto účtu'; + + @override + String get language_region => 'Jazyk a region'; + + @override + String get language => 'Jazyk'; + + @override + String get system_default => 'Systém'; + + @override + String get market_place_region => 'Region'; + + @override + String get recommendation_country => 'Země pro doporučení'; + + @override + String get appearance => 'Vzhled'; + + @override + String get layout_mode => 'Režim rozložení'; + + @override + String get override_layout_settings => 'Přepsat režim rozložení'; + + @override + String get adaptive => 'Adaptivní'; + + @override + String get compact => 'Kompaktní'; + + @override + String get extended => 'Rozšířený'; + + @override + String get theme => 'Téma'; + + @override + String get dark => 'Tmavé'; + + @override + String get light => 'Světlé'; + + @override + String get system => 'Systém'; + + @override + String get accent_color => 'Barva akcentu'; + + @override + String get sync_album_color => 'Synchronizovat barvu alba'; + + @override + String get sync_album_color_description => + 'Používá dominantní barvu obalu alba jako barvu akcentu'; + + @override + String get playback => 'Přehrávání'; + + @override + String get audio_quality => 'Kvalita zvuku'; + + @override + String get high => 'Vysoká'; + + @override + String get low => 'Nízká'; + + @override + String get pre_download_play => 'Předstáhnout a přehrát'; + + @override + String get pre_download_play_description => + 'Místo streamování audia stáhnout skladbu a přehrát (doporučeno pro uživatele s rychlejším internetem)'; + + @override + String get skip_non_music => 'Přeskočit nehudební segmenty (SponsorBlock)'; + + @override + String get blacklist_description => 'Zakázané skladby a umělci'; + + @override + String get wait_for_download_to_finish => 'Počkejte, až se dokončí stahování'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Chování při zavření'; + + @override + String get close => 'Zavřít'; + + @override + String get minimize_to_tray => 'Minimalizovat do lišty'; + + @override + String get show_tray_icon => 'Zobrazit ikonu v systémové liště'; + + @override + String get about => 'O aplikaci'; + + @override + String get u_love_spotube => 'Víme, že milujete Spotube'; + + @override + String get check_for_updates => 'Zkontrolovat aktualizace'; + + @override + String get about_spotube => 'O Spotube'; + + @override + String get blacklist => 'Černá listina'; + + @override + String get please_sponsor => 'Sponzorovat/darovat'; + + @override + String get spotube_description => + 'Spotube, rychlý, multiplatformní, bezplatný Spotify klient'; + + @override + String get version => 'Verze'; + + @override + String get build_number => 'Číslo sestavení'; + + @override + String get founder => 'Zakladatel'; + + @override + String get repository => 'Repozitář'; + + @override + String get bug_issues => 'Chyby+Problémy'; + + @override + String get made_with => 'Vytvořeno s ❤️ v Bangladéši🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licence'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Nebojte, žádné z vašich údajů nebudou shromažďovány ani s nikým sdíleny'; + + @override + String get know_how_to_login => 'Nevíte, jak na to?'; + + @override + String get follow_step_by_step_guide => 'Postupujte podle návodu'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Vyplňte prosím všechna pole'; + + @override + String get submit => 'Odeslat'; + + @override + String get exit => 'Ukončit'; + + @override + String get previous => 'Předchozí'; + + @override + String get next => 'Další'; + + @override + String get done => 'Hotovo'; + + @override + String get step_1 => 'Krok 1'; + + @override + String get first_go_to => 'Nejprve jděte na'; + + @override + String get something_went_wrong => 'Něco se pokazilo'; + + @override + String get piped_instance => 'Instance serveru Piped'; + + @override + String get piped_description => + 'Instance serveru Piped, kterou použít pro hledání skladeb'; + + @override + String get piped_warning => + 'Některé z nich nemusí dobře fungovat. Používejte na vlastní riziko'; + + @override + String get invidious_instance => 'Instance serveru Invidious'; + + @override + String get invidious_description => + 'Instance serveru Invidious pro párování stop'; + + @override + String get invidious_warning => + 'Některé instance nemusí fungovat správně. Používejte na vlastní riziko'; + + @override + String get generate => 'Generovat'; + + @override + String track_exists(Object track) { + return 'Skladba $track již existuje'; + } + + @override + String get replace_downloaded_tracks => 'Nahradit všechny stažené skladby'; + + @override + String get skip_download_tracks => + 'Přeskočit stahování všech stažených skladeb'; + + @override + String get do_you_want_to_replace => 'Chcete nahradit existující skladbu??'; + + @override + String get replace => 'Nahradit'; + + @override + String get skip => 'Přeskočit'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Vyberte až $count $type'; + } + + @override + String get select_genres => 'Vyberte žánry'; + + @override + String get add_genres => 'Přidat žánry'; + + @override + String get country => 'Země'; + + @override + String get number_of_tracks_generate => 'Počet skladeb k vygenerování'; + + @override + String get acousticness => 'Akustičnost'; + + @override + String get danceability => 'Tanečnost'; + + @override + String get energy => 'Energie'; + + @override + String get instrumentalness => 'Instrumentálnost'; + + @override + String get liveness => 'Živost'; + + @override + String get loudness => 'Hlasitost'; + + @override + String get speechiness => 'Mluvnost'; + + @override + String get valence => 'Valence'; + + @override + String get popularity => 'Popularita'; + + @override + String get key => 'Klíč'; + + @override + String get duration => 'Délka (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Režim'; + + @override + String get time_signature => 'Udání taktu'; + + @override + String get short => 'Krátký'; + + @override + String get medium => 'Střední'; + + @override + String get long => 'Dlouhý'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Cíl'; + + @override + String get moderate => 'Mírný'; + + @override + String get deselect_all => 'Zrušit výběr'; + + @override + String get select_all => 'Vybrat vše'; + + @override + String get are_you_sure => 'Jste si jisti?'; + + @override + String get generating_playlist => 'Generování vašeho vlastního playlistu...'; + + @override + String selected_count_tracks(Object count) { + return 'Vybráno $count skladeb'; + } + + @override + String get download_warning => + 'Pokud stáhnete všechny skladby najednou, pirátíte tím hudbu a škodíte kreativní společnosti hudby. Doufám, že jste si toho vědomi. Vždy se snažte respektovat a podporovat tvrdou práci umělců'; + + @override + String get download_ip_ban_warning => + 'Mimochodem, vaše IP může být na YouTube zablokována kvůli nadměrným požadavkům na stahování. Blokování IP znamená, že nemůžete používat YouTube (i když jste přihlášeni) alespoň 2-3 měsíce ze zařízení s touto IP. A Spotube nenese žádnou odpovědnost, pokud se to někdy stane'; + + @override + String get by_clicking_accept_terms => + 'Kliknutím na \'přijmout\' souhlasíte s následujícími podmínkami:'; + + @override + String get download_agreement_1 => 'Vím, že pirátím hudbu. Jsem špatný'; + + @override + String get download_agreement_2 => + 'Budu podporovat umělce, kdekoliv to bude možné, a dělám to jen proto, že nemám peníze na koupi jejich umění'; + + @override + String get download_agreement_3 => + 'Jsem si naprosto vědom toho, že moje IP může být na YouTube zablokována a nenesu žádnou odpovědnost za nehody způsobené mým současným jednáním'; + + @override + String get decline => 'Odmítnout'; + + @override + String get accept => 'Přijmout'; + + @override + String get details => 'Podrobnosti'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanál'; + + @override + String get likes => 'Líbí se'; + + @override + String get dislikes => 'Nelíbí se'; + + @override + String get views => 'Zobrazení'; + + @override + String get streamUrl => 'URL streamu'; + + @override + String get stop => 'Zastavit'; + + @override + String get sort_newest => 'Seřadit od nejnovějších'; + + @override + String get sort_oldest => 'Seřadit od nejstarších'; + + @override + String get sleep_timer => 'Časovač spánku'; + + @override + String mins(Object minutes) { + return '$minutes Minut'; + } + + @override + String hours(Object hours) { + return '$hours Hodin'; + } + + @override + String hour(Object hours) { + return '$hours Hodina'; + } + + @override + String get custom_hours => 'Vlastní hodiny'; + + @override + String get logs => 'Protokoly'; + + @override + String get developers => 'Vývojáři'; + + @override + String get not_logged_in => 'Nejste přihlášeni'; + + @override + String get search_mode => 'Režim hledání'; + + @override + String get audio_source => 'Zdroj zvuku'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Šifrování selhalo'; + + @override + String get encryption_failed_warning => + 'Spotube používá šifrování k bezpečnému ukládání vašich dat. Ale selhalo. Takže se vrátí k nezabezpečenému úložišti\nPokud používáte linux, ujistěte se, že máte nainstalovanou jakoukoli službu k ukládání bezpečnostních pověření (gnome-keyring, kde-wallet, keepassxc atd.)'; + + @override + String get querying_info => 'Získávání informací...'; + + @override + String get piped_api_down => 'Piped API je mimo provoz'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Instance Piped $pipedInstance je momentálně mimo provoz\n\nBuď změňte instanci nebo změňte \'Typ API\' na oficiální YouTube API\n\nPo změně se ujistěte, že aplikaci restartujete'; + } + + @override + String get you_are_offline => 'Momentálně jste offline'; + + @override + String get connection_restored => 'Vaše internetové připojení bylo obnoveno'; + + @override + String get use_system_title_bar => 'Použít systémové záhlaví okna'; + + @override + String get crunching_results => 'Zpracovávání výsledků...'; + + @override + String get search_to_get_results => 'Hledejte pro získání výsledků'; + + @override + String get use_amoled_mode => 'Úplně černé téma'; + + @override + String get pitch_dark_theme => 'AMOLED režim'; + + @override + String get normalize_audio => 'Normalizovat audio'; + + @override + String get change_cover => 'Změnit obal'; + + @override + String get add_cover => 'Přidat obal'; + + @override + String get restore_defaults => 'Obnovit výchozí'; + + @override + String get download_music_format => 'Formát stahování hudby'; + + @override + String get streaming_music_format => 'Formát streamování hudby'; + + @override + String get download_music_quality => 'Kvalita stahování hudby'; + + @override + String get streaming_music_quality => 'Kvalita streamování hudby'; + + @override + String get login_with_lastfm => 'Přihlásit se pomocí Last.fm'; + + @override + String get connect => 'Připojit'; + + @override + String get disconnect_lastfm => 'Odpojit Last.fm'; + + @override + String get disconnect => 'Odpojit'; + + @override + String get username => 'Uživatelské jméno'; + + @override + String get password => 'Heslo'; + + @override + String get login => 'Přihlásit se'; + + @override + String get login_with_your_lastfm => + 'Přihlásit se pomocí vašeho Last.fm účtu'; + + @override + String get scrobble_to_lastfm => 'Scrobble na Last.fm'; + + @override + String get go_to_album => 'Přejít na album'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'Procházet vše'; + + @override + String get genres => 'Žánry'; + + @override + String get explore_genres => 'Prozkoumat žánry'; + + @override + String get friends => 'Přátelé'; + + @override + String get no_lyrics_available => + 'Omlouváme se, není možné najít texty pro tuto skladbu'; + + @override + String get start_a_radio => 'Vytvořit rádio'; + + @override + String get how_to_start_radio => 'Jak chcete vytvořit rádio?'; + + @override + String get replace_queue_question => + 'Chcete nahradit aktuální frontu nebo k ní přidat?'; + + @override + String get endless_playback => 'Nekonečné přehrávání'; + + @override + String get delete_playlist => 'Smazat playlist'; + + @override + String get delete_playlist_confirmation => + 'Jste si jisti, že chcete smazat tento playlist?'; + + @override + String get local_tracks => 'Místní skladby'; + + @override + String get local_tab => 'Místní'; + + @override + String get song_link => 'Odkaz na skladbu'; + + @override + String get skip_this_nonsense => 'Přeskočit tenhle nesmysl'; + + @override + String get freedom_of_music => '“Svobodná hudba”'; + + @override + String get freedom_of_music_palm => '“Svobodná hudba ve vaší dlani”'; + + @override + String get get_started => 'Začít'; + + @override + String get youtube_source_description => 'Doporučeno a funguje nejlépe.'; + + @override + String get piped_source_description => + 'Nechcete být sledováni? Stejné jako YouTube, ale respektuje soukromí.'; + + @override + String get jiosaavn_source_description => 'Nejlepší pro jihoasijský region.'; + + @override + String get invidious_source_description => + 'Podobné Piped, ale s vyšší dostupností'; + + @override + String highest_quality(Object quality) { + return 'Nejvyšší kvalita: $quality'; + } + + @override + String get select_audio_source => 'Vyberte zdroj zvuku'; + + @override + String get endless_playback_description => + 'Automaticky přidávat nové skladby\nna konec fronty'; + + @override + String get choose_your_region => 'Vyberte svůj region'; + + @override + String get choose_your_region_description => + 'To pomůže Spotube ukázat vám správný obsah\npro vaši lokalitu.'; + + @override + String get choose_your_language => 'Vyberte svůj jazyk'; + + @override + String get help_project_grow => 'Pomozte tomuto projektu růst'; + + @override + String get help_project_grow_description => + 'Spotube je open-source projekt. Můžete pomoci tomuto projektu růst tím, že přispějete do projektu, nahlásíte chyby nebo navrhnete nové funkce.'; + + @override + String get contribute_on_github => 'Přispějte na GitHub'; + + @override + String get donate_on_open_collective => 'Darujte na Open Collective'; + + @override + String get browse_anonymously => 'Procházet anonymně'; + + @override + String get enable_connect => 'Povolit ovládání'; + + @override + String get enable_connect_description => + 'Ovládejte Spotube z jiného zařízení'; + + @override + String get devices => 'Zařízení'; + + @override + String get select => 'Vybrat'; + + @override + String connect_client_alert(Object client) { + return 'Zařízení je ovládáno z $client'; + } + + @override + String get this_device => 'Toto zařízení'; + + @override + String get remote => 'Ovladač'; + + @override + String get stats => 'Statistiky'; + + @override + String and_n_more(Object count) { + return 'a dalších $count'; + } + + @override + String get recently_played => 'Nedávno přehráno'; + + @override + String get browse_more => 'Procházet více'; + + @override + String get no_title => 'Bez názvu'; + + @override + String get not_playing => 'Nepřehrává se'; + + @override + String get epic_failure => 'Epické selhání!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Přidáno $tracks_length skladeb do fronty'; + } + + @override + String get spotube_has_an_update => 'Spotube má aktualizaci'; + + @override + String get download_now => 'Stáhnout nyní'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Byla vydána noční verze Spotube $nightlyBuildNum'; + } + + @override + String release_version(Object version) { + return 'Byla vydána verze Spotube v$version'; + } + + @override + String get read_the_latest => 'Přečtěte si nejnovější '; + + @override + String get release_notes => 'poznámky k vydání'; + + @override + String get pick_color_scheme => 'Vyberte barevné schéma'; + + @override + String get save => 'Uložit'; + + @override + String get choose_the_device => 'Vyberte zařízení:'; + + @override + String get multiple_device_connected => + 'Je připojeno více zařízení.\nVyberte zařízení, na kterém chcete provést tuto akci'; + + @override + String get nothing_found => 'Nic nenalezeno'; + + @override + String get the_box_is_empty => 'Krabice je prázdná'; + + @override + String get top_artists => 'Nejlepší umělci'; + + @override + String get top_albums => 'Nejlepší alba'; + + @override + String get this_week => 'Tento týden'; + + @override + String get this_month => 'Tento měsíc'; + + @override + String get last_6_months => 'Posledních 6 měsíců'; + + @override + String get this_year => 'Tento rok'; + + @override + String get last_2_years => 'Poslední 2 roky'; + + @override + String get all_time => 'Všechny časy'; + + @override + String powered_by_provider(Object providerName) { + return 'Pohání $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Sledující'; + + @override + String get birthday => 'Narozeniny'; + + @override + String get subscription => 'Předplatné'; + + @override + String get not_born => 'Nenarozen'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profil'; + + @override + String get no_name => 'Bez jména'; + + @override + String get edit => 'Upravit'; + + @override + String get user_profile => 'Uživatelský profil'; + + @override + String count_plays(Object count) { + return '$count přehrání'; + } + + @override + String get streaming_fees_hypothetical => + 'Poplatky za streamování (hypotetické)'; + + @override + String get minutes_listened => 'Poslouchané minuty'; + + @override + String get streamed_songs => 'Streamované skladby'; + + @override + String count_streams(Object count) { + return '$count streamů'; + } + + @override + String get owned_by_you => 'Vlastněno vámi'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'Zkopírováno $shareUrl do schránky'; + } + + @override + String get hipotetical_calculation => + '*Toto je vypočítáno na základě průměrného výplatu za přehrání 0,003–0,005 USD na online hudebních streamovacích platformách. Jedná se o hypotetický výpočet, který má uživateli ukázat, kolik by umělci dostali, pokud by jeho píseň poslouchal na jiné platformě.'; + + @override + String count_mins(Object minutes) { + return '$minutes minut'; + } + + @override + String get summary_minutes => 'minuty'; + + @override + String get summary_listened_to_music => 'Poslouchal(a) hudbu'; + + @override + String get summary_songs => 'písně'; + + @override + String get summary_streamed_overall => 'Streamováno celkově'; + + @override + String get summary_owed_to_artists => 'Dluženo umělcům\nTento měsíc'; + + @override + String get summary_artists => 'umělců'; + + @override + String get summary_music_reached_you => 'Hudba vás oslovila'; + + @override + String get summary_full_albums => 'plná alba'; + + @override + String get summary_got_your_love => 'Získal vaši lásku'; + + @override + String get summary_playlists => 'playlisty'; + + @override + String get summary_were_on_repeat => 'Byly na opakování'; + + @override + String total_money(Object money) { + return 'Celkem $money'; + } + + @override + String get webview_not_found => 'Webview nebyl nalezen'; + + @override + String get webview_not_found_description => + 'Na vašem zařízení není nainstalováno žádné runtime prostředí Webview.\nPokud je nainstalováno, ujistěte se, že je v environment PATH\n\nPo instalaci restartujte aplikaci'; + + @override + String get unsupported_platform => 'Nepodporovaná platforma'; + + @override + String get cache_music => 'Hudba v mezipaměti'; + + @override + String get open => 'Otevřít'; + + @override + String get cache_folder => 'Složka mezipaměti'; + + @override + String get export => 'Exportovat'; + + @override + String get clear_cache => 'Vymazat mezipaměť'; + + @override + String get clear_cache_confirmation => 'Opravdu chcete vymazat mezipaměť?'; + + @override + String get export_cache_files => 'Exportovat soubory z mezipaměti'; + + @override + String found_n_files(Object count) { + return 'Nalezeno $count souborů'; + } + + @override + String get export_cache_confirmation => 'Chcete exportovat tyto soubory do'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Exportováno $filesExported z $files souborů'; + } + + @override + String get undo => 'Zpět'; + + @override + String get download_all => 'Stáhnout vše'; + + @override + String get add_all_to_playlist => 'Přidat vše do seznamu skladeb'; + + @override + String get add_all_to_queue => 'Přidat vše do fronty'; + + @override + String get play_all_next => 'Přehrát vše následně'; + + @override + String get pause => 'Pauza'; + + @override + String get view_all => 'Zobrazit vše'; + + @override + String get no_tracks_added_yet => + 'Zdá se, že jste ještě nepřidali žádné skladby'; + + @override + String get no_tracks => 'Zdá se, že zde nejsou žádné skladby'; + + @override + String get no_tracks_listened_yet => + 'Zdá se, že jste ještě nic neposlouchali'; + + @override + String get not_following_artists => 'Nezajímáte se o žádné umělce'; + + @override + String get no_favorite_albums_yet => + 'Zdá se, že jste ještě nepřidali žádné alba mezi oblíbené'; + + @override + String get no_logs_found => 'Žádné záznamy nenalezeny'; + + @override + String get youtube_engine => 'YouTube Engine'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine není nainstalován'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine není nainstalován ve vašem systému.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Ujistěte se, že je k dispozici v proměnné PATH nebo\nnastavte absolutní cestu k $engine spustitelnému souboru níže'; + } + + @override + String get youtube_engine_unix_issue_message => + 'V macOS/Linux/Unixových systémech nebude fungovat nastavení cesty v .zshrc/.bashrc/.bash_profile atd.\nMusíte nastavit cestu v konfiguračním souboru shellu'; + + @override + String get download => 'Stáhnout'; + + @override + String get file_not_found => 'Soubor nenalezen'; + + @override + String get custom => 'Vlastní'; + + @override + String get add_custom_url => 'Přidat vlastní URL'; + + @override + String get edit_port => 'Upravit port'; + + @override + String get port_helper_msg => + 'Výchozí hodnota je -1, což znamená náhodné číslo. Pokud máte nakonfigurován firewall, doporučuje se to nastavit.'; + + @override + String connect_request(Object client) { + return 'Povolit $client připojení?'; + } + + @override + String get connection_request_denied => + 'Připojení bylo zamítnuto. Uživatel odmítl přístup.'; + + @override + String get an_error_occurred => 'Došlo k chybě'; + + @override + String get copy_to_clipboard => 'Kopírovat do schránky'; + + @override + String get view_logs => 'Zobrazit protokoly'; + + @override + String get retry => 'Zkusit znovu'; + + @override + String get no_default_metadata_provider_selected => + 'Nemáte nastaven výchozí poskytovatel metadat'; + + @override + String get manage_metadata_providers => 'Spravovat poskytovatele metadat'; + + @override + String get open_link_in_browser => 'Otevřít odkaz v prohlížeči?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Chcete otevřít následující odkaz?'; + + @override + String get unsafe_url_warning => + 'Odkazy z nedůvěryhodných zdrojů mohou být nebezpečné. Buďte opatrní!\nOdkaz si také můžete zkopírovat do schránky.'; + + @override + String get copy_link => 'Zkopírovat odkaz'; + + @override + String get building_your_timeline => + 'Vytváří se váš časový přehled podle poslechů...'; + + @override + String get official => 'Oficiální'; + + @override + String author_name(Object author) { + return 'Autor: $author'; + } + + @override + String get third_party => 'Třetí strana'; + + @override + String get plugin_requires_authentication => 'Plugin vyžaduje ověření'; + + @override + String get update_available => 'Aktualizace dostupná'; + + @override + String get supports_scrobbling => 'Podpora scrobblování'; + + @override + String get plugin_scrobbling_info => + 'Tento plugin scrobbles vaši hudbu pro vytvoření historie poslechů.'; + + @override + String get default_metadata_source => 'Výchozí zdroj metadat'; + + @override + String get set_default_metadata_source => 'Nastavit výchozí zdroj metadat'; + + @override + String get default_audio_source => 'Výchozí zdroj zvuku'; + + @override + String get set_default_audio_source => 'Nastavit výchozí zdroj zvuku'; + + @override + String get set_default => 'Nastavit jako výchozí'; + + @override + String get support => 'Podpora'; + + @override + String get support_plugin_development => 'Podpořit vývoj pluginu'; + + @override + String can_access_name_api(Object name) { + return '- Může přistupovat k API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Chcete tento plugin nainstalovat?'; + + @override + String get third_party_plugin_warning => + 'Tento plugin pochází z repozitáře třetí strany. Ujistěte se, že důvěřujete zdroji, než ho nainstalujete.'; + + @override + String get author => 'Autor'; + + @override + String get this_plugin_can_do_following => + 'Tento plugin může provádět následující úkony'; + + @override + String get install => 'Instalovat'; + + @override + String get install_a_metadata_provider => + 'Nainstalovat poskytovatele metadat'; + + @override + String get no_tracks_playing => 'Momentálně není přehrávána žádná skladba'; + + @override + String get synced_lyrics_not_available => + 'Synchronizované texty nejsou k dispozici k této písni. Prosím použijte'; + + @override + String get plain_lyrics => 'Prostý text'; + + @override + String get tab_instead => 'místo toho použijte tabulátor.'; + + @override + String get disclaimer => 'Prohlášení'; + + @override + String get third_party_plugin_dmca_notice => + 'Tým Spotube nenese žádnou odpovědnost (včetně právní) za pluginy „třetích stran“.\nPoužívejte je na vlastní riziko. Pro chyby/problémy je nahlaste do repozitáře pluginu.\n\nPokud jakýkoli plugin „třetí strany“ porušuje podmínky služby nebo DMCA kteréhokoli poskytovatele či právního subjektu, požádejte autora pluginu nebo hostingovou platformu (např. GitHub/Codeberg), aby podnikla kroky. Pluginy označené jako „třetí strana“ jsou otevřené a spravovány komunitou; nespravujeme je, tudíž nemůžeme jednat.\n\n'; + + @override + String get input_does_not_match_format => + 'Vstup neodpovídá požadovanému formátu'; + + @override + String get plugins => 'Pluginy'; + + @override + String get paste_plugin_download_url => + 'Vložte URL ke stažení nebo GitHub/Codeberg repozitář či přímý odkaz na soubor .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Stáhnout a nainstalovat plugin z URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Nepodařilo se přidat plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'Nahrát plugin ze souboru'; + + @override + String get installed => 'Nainstalováno'; + + @override + String get available_plugins => 'Dostupné pluginy'; + + @override + String get configure_plugins => + 'Konfigurujte své vlastní pluginy poskytovatele metadat a zdroje zvuku'; + + @override + String get audio_scrobblers => 'Audio scrobblers'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Zdroj: '; + + @override + String get uncompressed => 'Nekomprimováno'; + + @override + String get dab_music_source_description => + 'Pro audiofily. Poskytuje vysoce kvalitní/bezztrátové zvukové toky. Přesná shoda skladeb na základě ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_de.dart b/lib/l10n/generated/app_localizations_de.dart new file mode 100644 index 00000000..4ab10266 --- /dev/null +++ b/lib/l10n/generated/app_localizations_de.dart @@ -0,0 +1,1579 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for German (`de`). +class AppLocalizationsDe extends AppLocalizations { + AppLocalizationsDe([String locale = 'de']) : super(locale); + + @override + String get guest => 'Gast'; + + @override + String get browse => 'Durchsuchen'; + + @override + String get search => 'Suchen'; + + @override + String get library => 'Bibliothek'; + + @override + String get lyrics => 'Songtexte'; + + @override + String get settings => 'Einstellungen'; + + @override + String get genre_categories_filter => 'Filtere Kategorien oder Genres...'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Personalisiert'; + + @override + String get featured => 'Empfohlen'; + + @override + String get new_releases => 'Neue Veröffentlichungen'; + + @override + String get songs => 'Songs'; + + @override + String playing_track(Object track) { + return 'Wiedergabe: $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Dadurch wird die aktuelle Warteschlange gelöscht. $track_length Titel werden entfernt.\nMöchten Sie fortfahren?'; + } + + @override + String get load_more => 'Mehr laden'; + + @override + String get playlists => 'Playlists'; + + @override + String get artists => 'Künstler'; + + @override + String get albums => 'Alben'; + + @override + String get tracks => 'Titel'; + + @override + String get downloads => 'Downloads'; + + @override + String get filter_playlists => 'Filtere deine Playlists...'; + + @override + String get liked_tracks => 'Gefällt mir-Titel'; + + @override + String get liked_tracks_description => 'Alle deine geliketen Titel'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Erstelle eine Playlist'; + + @override + String get update_playlist => 'Wiedergabeliste aktualisieren'; + + @override + String get create => 'Erstellen'; + + @override + String get cancel => 'Abbrechen'; + + @override + String get update => 'Aktualisieren'; + + @override + String get playlist_name => 'Playlist-Name'; + + @override + String get name_of_playlist => 'Name der Playlist'; + + @override + String get description => 'Beschreibung'; + + @override + String get public => 'Öffentlich'; + + @override + String get collaborative => 'Kollaborativ'; + + @override + String get search_local_tracks => 'Lokale Titel durchsuchen...'; + + @override + String get play => 'Wiedergabe'; + + @override + String get delete => 'Löschen'; + + @override + String get none => 'Keine'; + + @override + String get sort_a_z => 'Sortieren nach A-Z'; + + @override + String get sort_z_a => 'Sortieren nach Z-A'; + + @override + String get sort_artist => 'Sortieren nach Künstler'; + + @override + String get sort_album => 'Sortieren nach Album'; + + @override + String get sort_duration => 'Nach Dauer sortieren'; + + @override + String get sort_tracks => 'Titel sortieren'; + + @override + String currently_downloading(Object tracks_length) { + return 'Derzeitige Downloads ($tracks_length)'; + } + + @override + String get cancel_all => 'Alle abbrechen'; + + @override + String get filter_artist => 'Künstler filtern...'; + + @override + String followers(Object followers) { + return '$followers Follower'; + } + + @override + String get add_artist_to_blacklist => + 'Künstler zur Schwarzen Liste hinzufügen'; + + @override + String get top_tracks => 'Top-Titel'; + + @override + String get fans_also_like => 'Fans mögen auch'; + + @override + String get loading => 'Laden...'; + + @override + String get artist => 'Künstler'; + + @override + String get blacklisted => 'Auf der Schwarzen Liste'; + + @override + String get following => 'Folgen'; + + @override + String get follow => 'Folgen'; + + @override + String get artist_url_copied => 'Künstler-URL in Zwischenablage kopiert'; + + @override + String added_to_queue(Object tracks) { + return '$tracks Titel zur Warteschlange hinzugefügt'; + } + + @override + String get filter_albums => 'Alben filtern...'; + + @override + String get synced => 'Synchronisiert'; + + @override + String get plain => 'Einfach'; + + @override + String get shuffle => 'Zufällige Wiedergabe'; + + @override + String get search_tracks => 'Titel durchsuchen...'; + + @override + String get released => 'Veröffentlicht'; + + @override + String error(Object error) { + return 'Fehler $error'; + } + + @override + String get title => 'Titel'; + + @override + String get time => 'Dauer'; + + @override + String get more_actions => 'Weitere Aktionen'; + + @override + String download_count(Object count) { + return 'Download ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Zu Playlist hinzufügen ($count)'; + } + + @override + String add_count_to_queue(Object count) { + return 'Zur Warteschlange hinzufügen ($count)'; + } + + @override + String play_count_next(Object count) { + return 'Als nächstes abspielen ($count)'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return '$data in Zwischenablage kopiert'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track zu folgenden Playlists hinzufügen'; + } + + @override + String get add => 'Hinzufügen'; + + @override + String added_track_to_queue(Object track) { + return '$track zur Warteschlange hinzugefügt'; + } + + @override + String get add_to_queue => 'Zur Warteschlange hinzufügen'; + + @override + String track_will_play_next(Object track) { + return '$track wird als nächstes abgespielt'; + } + + @override + String get play_next => 'Als nächstes abspielen'; + + @override + String removed_track_from_queue(Object track) { + return '$track aus der Warteschlange entfernt'; + } + + @override + String get remove_from_queue => 'Aus der Warteschlange entfernen'; + + @override + String get remove_from_favorites => 'Aus Favoriten entfernen'; + + @override + String get save_as_favorite => 'Als Favorit speichern'; + + @override + String get add_to_playlist => 'Zur Playlist hinzufügen'; + + @override + String get remove_from_playlist => 'Aus der Playlist entfernen'; + + @override + String get add_to_blacklist => 'Zur Schwarzen Liste hinzufügen'; + + @override + String get remove_from_blacklist => 'Aus der Schwarzen Liste entfernen'; + + @override + String get share => 'Teilen'; + + @override + String get mini_player => 'Mini-Player'; + + @override + String get slide_to_seek => 'Zum Vor- oder Zurückspulen ziehen'; + + @override + String get shuffle_playlist => 'Playlist mischen'; + + @override + String get unshuffle_playlist => 'Playlist nicht mehr mischen'; + + @override + String get previous_track => 'Vorheriger Track'; + + @override + String get next_track => 'Nächster Track'; + + @override + String get pause_playback => 'Wiedergabe pausieren'; + + @override + String get resume_playback => 'Wiedergabe fortsetzen'; + + @override + String get loop_track => 'Track wiederholen'; + + @override + String get no_loop => 'Kein Loop'; + + @override + String get repeat_playlist => 'Playlist wiederholen'; + + @override + String get queue => 'Warteschlange'; + + @override + String get alternative_track_sources => 'Alternative Track-Quellen'; + + @override + String get download_track => 'Track herunterladen'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks Tracks in der Warteschlange'; + } + + @override + String get clear_all => 'Alle löschen'; + + @override + String get show_hide_ui_on_hover => 'UI beim Überfahren anzeigen/ausblenden'; + + @override + String get always_on_top => 'Immer im Vordergrund'; + + @override + String get exit_mini_player => 'Mini-Player verlassen'; + + @override + String get download_location => 'Download-Speicherort'; + + @override + String get local_library => 'Lokale Bibliothek'; + + @override + String get add_library_location => 'Zur Bibliothek hinzufügen'; + + @override + String get remove_library_location => 'Aus der Bibliothek entfernen'; + + @override + String get account => 'Konto'; + + @override + String get logout => 'Abmelden'; + + @override + String get logout_of_this_account => 'Von diesem Konto abmelden'; + + @override + String get language_region => 'Sprache & Region'; + + @override + String get language => 'Sprache'; + + @override + String get system_default => 'Systemstandard'; + + @override + String get market_place_region => 'Marktplatzregion'; + + @override + String get recommendation_country => 'Empfehlungsland'; + + @override + String get appearance => 'Erscheinungsbild'; + + @override + String get layout_mode => 'Layout-Modus'; + + @override + String get override_layout_settings => + 'Responsiven Layout-Modus-Einstellungen überschreiben'; + + @override + String get adaptive => 'Adaptiv'; + + @override + String get compact => 'Kompakt'; + + @override + String get extended => 'Erweitert'; + + @override + String get theme => 'Design'; + + @override + String get dark => 'Dunkel'; + + @override + String get light => 'Hell'; + + @override + String get system => 'System'; + + @override + String get accent_color => 'Akzentfarbe'; + + @override + String get sync_album_color => 'Albumfarbe synchronisieren'; + + @override + String get sync_album_color_description => + 'Verwendet die dominante Farbe des Album Covers als Akzentfarbe'; + + @override + String get playback => 'Wiedergabe'; + + @override + String get audio_quality => 'Audioqualität'; + + @override + String get high => 'Hoch'; + + @override + String get low => 'Niedrig'; + + @override + String get pre_download_play => 'Vorab herunterladen und abspielen'; + + @override + String get pre_download_play_description => + 'Anstatt Audio zu streamen, Bytes herunterladen und abspielen (Empfohlen für Benutzer mit hoher Bandbreite)'; + + @override + String get skip_non_music => + 'Überspringe Nicht-Musik-Segmente (SponsorBlock)'; + + @override + String get blacklist_description => 'Gesperrte Titel und Künstler'; + + @override + String get wait_for_download_to_finish => + 'Bitte warten Sie, bis der aktuelle Download abgeschlossen ist'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Verhalten beim Schließen'; + + @override + String get close => 'Schließen'; + + @override + String get minimize_to_tray => 'In Taskleiste minimieren'; + + @override + String get show_tray_icon => 'Systemsymbol anzeigen'; + + @override + String get about => 'Über'; + + @override + String get u_love_spotube => 'Wir wissen, dass Sie Spotube lieben'; + + @override + String get check_for_updates => 'Nach Updates suchen'; + + @override + String get about_spotube => 'Über Spotube'; + + @override + String get blacklist => 'Gesperrte Titel'; + + @override + String get please_sponsor => 'Bitte unterstützen/Spenden Sie'; + + @override + String get spotube_description => + 'Spotube, ein leichtgewichtiger, plattformübergreifender und kostenloser Spotify-Client'; + + @override + String get version => 'Version'; + + @override + String get build_number => 'Build-Nummer'; + + @override + String get founder => 'Gründer'; + + @override + String get repository => 'Repository'; + + @override + String get bug_issues => 'Fehler und Probleme'; + + @override + String get made_with => 'Entwickelt mit ❤️ in Bangladesch 🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Lizenz'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Keine Sorge, Ihre Anmeldeinformationen werden nicht erfasst oder mit anderen geteilt'; + + @override + String get know_how_to_login => 'Wissen Sie nicht, wie es geht?'; + + @override + String get follow_step_by_step_guide => + 'Befolgen Sie die schrittweise Anleitung'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => 'Bitte füllen Sie alle Felder aus'; + + @override + String get submit => 'Senden'; + + @override + String get exit => 'Beenden'; + + @override + String get previous => 'Zurück'; + + @override + String get next => 'Weiter'; + + @override + String get done => 'Fertig'; + + @override + String get step_1 => 'Schritt 1'; + + @override + String get first_go_to => 'Gehe zuerst zu'; + + @override + String get something_went_wrong => 'Etwas ist schiefgelaufen'; + + @override + String get piped_instance => 'Piped-Serverinstanz'; + + @override + String get piped_description => + 'Die Piped-Serverinstanz, die zur Titelzuordnung verwendet werden soll'; + + @override + String get piped_warning => + 'Einige von ihnen funktionieren möglicherweise nicht gut. Verwende sie also auf eigenes Risiko'; + + @override + String get invidious_instance => 'Invidious-Serverinstanz'; + + @override + String get invidious_description => + 'Die Invidious-Serverinstanz zur Titelerkennung'; + + @override + String get invidious_warning => + 'Einige Instanzen funktionieren möglicherweise nicht gut. Benutzung auf eigene Gefahr'; + + @override + String get generate => 'Generieren'; + + @override + String track_exists(Object track) { + return 'Track $track existiert bereits'; + } + + @override + String get replace_downloaded_tracks => + 'Alle heruntergeladenen Titel ersetzen'; + + @override + String get skip_download_tracks => + 'Das Herunterladen aller heruntergeladenen Titel überspringen'; + + @override + String get do_you_want_to_replace => + 'Möchtest du den vorhandenen Track ersetzen?'; + + @override + String get replace => 'Ersetzen'; + + @override + String get skip => 'Überspringen'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Wähle bis zu $count $type aus'; + } + + @override + String get select_genres => 'Genres auswählen'; + + @override + String get add_genres => 'Genres hinzufügen'; + + @override + String get country => 'Land'; + + @override + String get number_of_tracks_generate => 'Anzahl der zu generierenden Titel'; + + @override + String get acousticness => 'Akustik'; + + @override + String get danceability => 'Tanzbarkeit'; + + @override + String get energy => 'Energie'; + + @override + String get instrumentalness => 'Instrumentalität'; + + @override + String get liveness => 'Lebendigkeit'; + + @override + String get loudness => 'Lautstärke'; + + @override + String get speechiness => 'Sprechanteil'; + + @override + String get valence => 'Stimmung'; + + @override + String get popularity => 'Beliebtheit'; + + @override + String get key => 'Tonart'; + + @override + String get duration => 'Dauer (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Modus'; + + @override + String get time_signature => 'Taktart'; + + @override + String get short => 'Kurz'; + + @override + String get medium => 'Mittel'; + + @override + String get long => 'Lang'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Ziel'; + + @override + String get moderate => 'Mäßig'; + + @override + String get deselect_all => 'Alle abwählen'; + + @override + String get select_all => 'Alle auswählen'; + + @override + String get are_you_sure => 'Bist du sicher?'; + + @override + String get generating_playlist => + 'Erstelle deine individuelle Wiedergabeliste...'; + + @override + String selected_count_tracks(Object count) { + return '$count Titel ausgewählt'; + } + + @override + String get download_warning => + 'Wenn du alle Titel in großen Mengen herunterlädst, betreibst du eindeutig Raubkopien von Musik und schadest der kreativen Gesellschaft der Musik. Ich hoffe, dir ist dies bewusst. Versuche immer, die harte Arbeit der Künstler zu respektieren und zu unterstützen.'; + + @override + String get download_ip_ban_warning => + 'Übrigens, deine IP-Adresse kann aufgrund übermäßiger Downloadanfragen von YouTube gesperrt werden. Eine IP-Sperre bedeutet, dass du YouTube (auch wenn du angemeldet bist) für mindestens 2-3 Monate von diesem IP-Gerät aus nicht nutzen kannst. Spotube übernimmt keine Verantwortung, falls dies jemals geschieht.'; + + @override + String get by_clicking_accept_terms => + 'Durch Klicken auf \'Akzeptieren\' stimmst du den folgenden Bedingungen zu:'; + + @override + String get download_agreement_1 => + 'Ich weiß, dass ich Raubkopien von Musik betreibe. Ich bin böse.'; + + @override + String get download_agreement_2 => + 'Ich werde die Künstler, wo immer ich kann, unterstützen, und ich tue dies nur, weil ich kein Geld habe, um ihre Kunst zu kaufen.'; + + @override + String get download_agreement_3 => + 'Mir ist vollkommen bewusst, dass meine IP-Adresse auf YouTube gesperrt werden kann, und ich halte Spotube oder seine Eigentümer/Mitarbeiter nicht für etwaige Unfälle verantwortlich, die durch meine derzeitige Handlung verursacht werden.'; + + @override + String get decline => 'Ablehnen'; + + @override + String get accept => 'Akzeptieren'; + + @override + String get details => 'Details'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanal'; + + @override + String get likes => 'Likes'; + + @override + String get dislikes => 'Dislikes'; + + @override + String get views => 'Aufrufe'; + + @override + String get streamUrl => 'Stream-URL'; + + @override + String get stop => 'Stopp'; + + @override + String get sort_newest => 'Nach neuesten Hinzufügungen sortieren'; + + @override + String get sort_oldest => 'Nach ältesten Hinzufügungen sortieren'; + + @override + String get sleep_timer => 'Schlaftimer'; + + @override + String mins(Object minutes) { + return '$minutes Minuten'; + } + + @override + String hours(Object hours) { + return '$hours Stunden'; + } + + @override + String hour(Object hours) { + return '$hours Stunde'; + } + + @override + String get custom_hours => 'Benutzerdefinierte Stunden'; + + @override + String get logs => 'Protokolle'; + + @override + String get developers => 'Entwickler'; + + @override + String get not_logged_in => 'Sie sind nicht angemeldet'; + + @override + String get search_mode => 'Suchmodus'; + + @override + String get audio_source => 'Audioquelle'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => 'Verschlüsselung fehlgeschlagen'; + + @override + String get encryption_failed_warning => + 'Spotube verwendet Verschlüsselung, um Ihre Daten sicher zu speichern. Dies ist jedoch fehlgeschlagen. Daher wird es auf unsichere Speicherung zurückgreifen\nWenn Sie Linux verwenden, stellen Sie bitte sicher, dass Sie Secret-Services wie gnome-keyring, kde-wallet und keepassxc installiert haben'; + + @override + String get querying_info => 'Abfrageinformationen...'; + + @override + String get piped_api_down => 'Die Piped API ist ausgefallen'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Die Piped-Instanz $pipedInstance ist derzeit nicht verfügbar\n\nEntweder ändern Sie die Instanz oder wechseln Sie den \'API-Typ\' zur offiziellen YouTube API\n\nStellen Sie sicher, dass Sie die App nach der Änderung neu starten'; + } + + @override + String get you_are_offline => 'Sie sind derzeit offline'; + + @override + String get connection_restored => + 'Ihre Internetverbindung wurde wiederhergestellt'; + + @override + String get use_system_title_bar => 'System-Titelleiste verwenden'; + + @override + String get crunching_results => 'Ergebnisse werden verarbeitet...'; + + @override + String get search_to_get_results => 'Suche, um Ergebnisse zu erhalten'; + + @override + String get use_amoled_mode => 'AMOLED-Modus verwenden'; + + @override + String get pitch_dark_theme => 'Pitch Black Dart Theme'; + + @override + String get normalize_audio => 'Audio normalisieren'; + + @override + String get change_cover => 'Cover ändern'; + + @override + String get add_cover => 'Cover hinzufügen'; + + @override + String get restore_defaults => 'Standardeinstellungen wiederherstellen'; + + @override + String get download_music_format => 'Musik-Downloadformat'; + + @override + String get streaming_music_format => 'Musik-Streamingformat'; + + @override + String get download_music_quality => 'Musik-Downloadqualität'; + + @override + String get streaming_music_quality => 'Musik-Streamingqualität'; + + @override + String get login_with_lastfm => 'Mit Last.fm anmelden'; + + @override + String get connect => 'Verbinden'; + + @override + String get disconnect_lastfm => 'Last.fm trennen'; + + @override + String get disconnect => 'Trennen'; + + @override + String get username => 'Benutzername'; + + @override + String get password => 'Passwort'; + + @override + String get login => 'Anmelden'; + + @override + String get login_with_your_lastfm => 'Mit Ihrem Last.fm-Konto anmelden'; + + @override + String get scrobble_to_lastfm => 'Auf Last.fm scrobbeln'; + + @override + String get go_to_album => 'Zum Album gehen'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'Alles durchsuchen'; + + @override + String get genres => 'Genres'; + + @override + String get explore_genres => 'Genres erkunden'; + + @override + String get friends => 'Freunde'; + + @override + String get no_lyrics_available => + 'Entschuldigung, Texte für diesen Track konnten nicht gefunden werden'; + + @override + String get start_a_radio => 'Radio starten'; + + @override + String get how_to_start_radio => 'Wie möchten Sie das Radio starten?'; + + @override + String get replace_queue_question => + 'Möchten Sie die aktuelle Wiedergabeliste ersetzen oder hinzufügen?'; + + @override + String get endless_playback => 'Endlose Wiedergabe'; + + @override + String get delete_playlist => 'Wiedergabeliste löschen'; + + @override + String get delete_playlist_confirmation => + 'Sind Sie sicher, dass Sie diese Wiedergabeliste löschen möchten?'; + + @override + String get local_tracks => 'Lokale Titel'; + + @override + String get local_tab => 'Lokal'; + + @override + String get song_link => 'Lied-Link'; + + @override + String get skip_this_nonsense => 'Diesen Unsinn überspringen'; + + @override + String get freedom_of_music => '“Freiheit der Musik”'; + + @override + String get freedom_of_music_palm => + '“Freiheit der Musik in Ihrer Handfläche”'; + + @override + String get get_started => 'Lass uns anfangen'; + + @override + String get youtube_source_description => + 'Empfohlen und funktioniert am besten.'; + + @override + String get piped_source_description => + 'Fühlen Sie sich frei? Wie YouTube, aber viel freier.'; + + @override + String get jiosaavn_source_description => + 'Am besten für die südasiatische Region.'; + + @override + String get invidious_source_description => + 'Ähnlich wie Piped, aber mit höherer Verfügbarkeit'; + + @override + String highest_quality(Object quality) { + return 'Höchste Qualität: $quality'; + } + + @override + String get select_audio_source => 'Audioquelle auswählen'; + + @override + String get endless_playback_description => + 'Neue Lieder automatisch\nam Ende der Wiedergabeliste hinzufügen'; + + @override + String get choose_your_region => 'Wählen Sie Ihre Region'; + + @override + String get choose_your_region_description => + 'Dies wird Spotube helfen, Ihnen den richtigen Inhalt\nfür Ihren Standort anzuzeigen.'; + + @override + String get choose_your_language => 'Wählen Sie Ihre Sprache'; + + @override + String get help_project_grow => 'Helfen Sie diesem Projekt zu wachsen'; + + @override + String get help_project_grow_description => + 'Spotube ist ein Open-Source-Projekt. Sie können diesem Projekt helfen, indem Sie zum Projekt beitragen, Fehler melden oder neue Funktionen vorschlagen.'; + + @override + String get contribute_on_github => 'Auf GitHub beitragen'; + + @override + String get donate_on_open_collective => 'Auf Open Collective spenden'; + + @override + String get browse_anonymously => 'Anonym durchsuchen'; + + @override + String get enable_connect => 'Verbindung aktivieren'; + + @override + String get enable_connect_description => + 'Spotube von anderen Geräten steuern'; + + @override + String get devices => 'Geräte'; + + @override + String get select => 'Auswählen'; + + @override + String connect_client_alert(Object client) { + return 'Du wirst von $client gesteuert'; + } + + @override + String get this_device => 'Dieses Gerät'; + + @override + String get remote => 'Fernbedienung'; + + @override + String get stats => 'Statistiken'; + + @override + String and_n_more(Object count) { + return 'und $count mehr'; + } + + @override + String get recently_played => 'Zuletzt gespielt'; + + @override + String get browse_more => 'Mehr durchsuchen'; + + @override + String get no_title => 'Kein Titel'; + + @override + String get not_playing => 'Wird nicht abgespielt'; + + @override + String get epic_failure => 'Episches Versagen!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length Titel zur Warteschlange hinzugefügt'; + } + + @override + String get spotube_has_an_update => 'Spotube hat ein Update'; + + @override + String get download_now => 'Jetzt herunterladen'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum wurde veröffentlicht'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version wurde veröffentlicht'; + } + + @override + String get read_the_latest => 'Lese die neuesten '; + + @override + String get release_notes => 'Versionshinweise'; + + @override + String get pick_color_scheme => 'Farbschema wählen'; + + @override + String get save => 'Speichern'; + + @override + String get choose_the_device => 'Wähle das Gerät:'; + + @override + String get multiple_device_connected => + 'Es sind mehrere Geräte verbunden.\nWähle das Gerät, auf dem diese Aktion ausgeführt werden soll'; + + @override + String get nothing_found => 'Nichts gefunden'; + + @override + String get the_box_is_empty => 'Die Box ist leer'; + + @override + String get top_artists => 'Top-Künstler'; + + @override + String get top_albums => 'Top-Alben'; + + @override + String get this_week => 'Diese Woche'; + + @override + String get this_month => 'Diesen Monat'; + + @override + String get last_6_months => 'Letzte 6 Monate'; + + @override + String get this_year => 'Dieses Jahr'; + + @override + String get last_2_years => 'Letzte 2 Jahre'; + + @override + String get all_time => 'Alle Zeiten'; + + @override + String powered_by_provider(Object providerName) { + return 'Bereitgestellt von $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Follower'; + + @override + String get birthday => 'Geburtstag'; + + @override + String get subscription => 'Abonnement'; + + @override + String get not_born => 'Nicht geboren'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profil'; + + @override + String get no_name => 'Kein Name'; + + @override + String get edit => 'Bearbeiten'; + + @override + String get user_profile => 'Benutzerprofil'; + + @override + String count_plays(Object count) { + return '$count Wiedergaben'; + } + + @override + String get streaming_fees_hypothetical => 'Streaming-Gebühren (hypothetisch)'; + + @override + String get minutes_listened => 'Gehörte Minuten'; + + @override + String get streamed_songs => 'Gestreamte Lieder'; + + @override + String count_streams(Object count) { + return '$count Streams'; + } + + @override + String get owned_by_you => 'In Ihrem Besitz'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl in die Zwischenablage kopiert'; + } + + @override + String get hipotetical_calculation => + '*Diese Berechnung basiert auf der durchschnittlichen Auszahlung pro Stream (0,003 USD bis 0,005 USD) auf Online-Musik-Streaming-Plattformen. Sie ist hypothetisch und soll dem Nutzer veranschaulichen, wie viel er den Künstlern bezahlt hätte, wenn er ihren Song auf verschiedenen Streaming-Plattformen gehört hätte.'; + + @override + String count_mins(Object minutes) { + return '$minutes Minuten'; + } + + @override + String get summary_minutes => 'Minuten'; + + @override + String get summary_listened_to_music => 'Hat Musik gehört'; + + @override + String get summary_songs => 'Lieder'; + + @override + String get summary_streamed_overall => 'Insgesamt gestreamt'; + + @override + String get summary_owed_to_artists => + 'Den Künstlern geschuldet\nDiesen Monat'; + + @override + String get summary_artists => 'Künstler'; + + @override + String get summary_music_reached_you => 'Musik hat Sie erreicht'; + + @override + String get summary_full_albums => 'volle Alben'; + + @override + String get summary_got_your_love => 'Hat Ihre Liebe gewonnen'; + + @override + String get summary_playlists => 'Wiedergabelisten'; + + @override + String get summary_were_on_repeat => 'Wurden wiederholt'; + + @override + String total_money(Object money) { + return 'Gesamt $money'; + } + + @override + String get webview_not_found => 'Webview nicht gefunden'; + + @override + String get webview_not_found_description => + 'Es ist keine Webview-Laufzeitumgebung auf Ihrem Gerät installiert.\nFalls installiert, stellen Sie sicher, dass es im environment PATH ist\n\nNach der Installation starten Sie die App neu'; + + @override + String get unsupported_platform => 'Nicht unterstützte Plattform'; + + @override + String get cache_music => 'Musik zwischenspeichern'; + + @override + String get open => 'Öffnen'; + + @override + String get cache_folder => 'Cache-Ordner'; + + @override + String get export => 'Exportieren'; + + @override + String get clear_cache => 'Cache leeren'; + + @override + String get clear_cache_confirmation => 'Möchten Sie den Cache leeren?'; + + @override + String get export_cache_files => 'Cachedateien exportieren'; + + @override + String found_n_files(Object count) { + return '$count Dateien gefunden'; + } + + @override + String get export_cache_confirmation => + 'Möchten Sie diese Dateien exportieren nach'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported von $files Dateien exportiert'; + } + + @override + String get undo => 'Rückgängig'; + + @override + String get download_all => 'Alle herunterladen'; + + @override + String get add_all_to_playlist => 'Alle zur Playlist hinzufügen'; + + @override + String get add_all_to_queue => 'Alle zur Warteschlange hinzufügen'; + + @override + String get play_all_next => 'Alle als Nächstes abspielen'; + + @override + String get pause => 'Pause'; + + @override + String get view_all => 'Alle ansehen'; + + @override + String get no_tracks_added_yet => 'Sie haben noch keine Titel hinzugefügt.'; + + @override + String get no_tracks => 'Es sieht so aus, als ob hier keine Titel sind.'; + + @override + String get no_tracks_listened_yet => + 'Es scheint, dass Sie noch nichts gehört haben.'; + + @override + String get not_following_artists => 'Sie folgen noch keinem Künstler.'; + + @override + String get no_favorite_albums_yet => + 'Es sieht so aus, als ob Sie noch keine Alben zu Ihren Favoriten hinzugefügt haben.'; + + @override + String get no_logs_found => 'Keine Protokolle gefunden'; + + @override + String get youtube_engine => 'YouTube-Engine'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine ist nicht installiert'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine ist nicht auf Ihrem System installiert.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Stellen Sie sicher, dass es im PATH verfügbar ist oder\nsetzen Sie den absoluten Pfad zur $engine ausführbaren Datei unten.'; + } + + @override + String get youtube_engine_unix_issue_message => + 'In macOS/Linux/unixähnlichen Betriebssystemen funktioniert das Setzen des Pfads in .zshrc/.bashrc/.bash_profile usw. nicht.\nSie müssen den Pfad in der Shell-Konfigurationsdatei festlegen.'; + + @override + String get download => 'Herunterladen'; + + @override + String get file_not_found => 'Datei nicht gefunden'; + + @override + String get custom => 'Benutzerdefiniert'; + + @override + String get add_custom_url => 'Benutzerdefinierte URL hinzufügen'; + + @override + String get edit_port => 'Port bearbeiten'; + + @override + String get port_helper_msg => + 'Der Standardwert ist -1, was eine zufällige Zahl bedeutet. Wenn Sie eine Firewall konfiguriert haben, wird empfohlen, dies einzustellen.'; + + @override + String connect_request(Object client) { + return '$client die Verbindung erlauben?'; + } + + @override + String get connection_request_denied => + 'Verbindung abgelehnt. Benutzer hat den Zugriff verweigert.'; + + @override + String get an_error_occurred => 'Ein Fehler ist aufgetreten'; + + @override + String get copy_to_clipboard => 'In die Zwischenablage kopieren'; + + @override + String get view_logs => 'Protokolle anzeigen'; + + @override + String get retry => 'Erneut versuchen'; + + @override + String get no_default_metadata_provider_selected => + 'Sie haben keinen Standard-Metadatenanbieter festgelegt'; + + @override + String get manage_metadata_providers => 'Metadatenanbieter verwalten'; + + @override + String get open_link_in_browser => 'Link im Browser öffnen?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Möchten Sie folgenden Link öffnen?'; + + @override + String get unsafe_url_warning => + 'Das Öffnen von Links aus nicht vertrauenswürdigen Quellen kann unsicher sein. Seien Sie vorsichtig!\nSie können den Link auch in Ihre Zwischenablage kopieren.'; + + @override + String get copy_link => 'Link kopieren'; + + @override + String get building_your_timeline => + 'Ihr Zeitverlauf wird basierend auf Ihren Hördaten erstellt…'; + + @override + String get official => 'Offiziell'; + + @override + String author_name(Object author) { + return 'Autor: $author'; + } + + @override + String get third_party => 'Drittanbieter'; + + @override + String get plugin_requires_authentication => + 'Plugin erfordert Authentifizierung'; + + @override + String get update_available => 'Update verfügbar'; + + @override + String get supports_scrobbling => 'Unterstützt Scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Dieses Plugin scrobbelt Ihre Musik, um Ihre Hörhistorie zu erstellen.'; + + @override + String get default_metadata_source => 'Standard-Metadatenquelle'; + + @override + String get set_default_metadata_source => + 'Standard-Metadatenquelle festlegen'; + + @override + String get default_audio_source => 'Standard-Audioquelle'; + + @override + String get set_default_audio_source => 'Standard-Audioquelle festlegen'; + + @override + String get set_default => 'Als Standard festlegen'; + + @override + String get support => 'Unterstützung'; + + @override + String get support_plugin_development => 'Plugin-Entwicklung unterstützen'; + + @override + String can_access_name_api(Object name) { + return '- Kann auf **$name**-API zugreifen'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Möchten Sie dieses Plugin installieren?'; + + @override + String get third_party_plugin_warning => + 'Dieses Plugin stammt aus einem Drittanbieter-Repository. Bitte stellen Sie sicher, dass Sie der Quelle vertrauen, bevor Sie es installieren.'; + + @override + String get author => 'Autor'; + + @override + String get this_plugin_can_do_following => 'Dieses Plugin kann Folgendes:'; + + @override + String get install => 'Installieren'; + + @override + String get install_a_metadata_provider => + 'Einen Metadatenanbieter installieren'; + + @override + String get no_tracks_playing => 'Derzeit wird kein Titel abgespielt'; + + @override + String get synced_lyrics_not_available => + 'Synchronisierte Liedtexte sind für dieses Lied nicht verfügbar. Bitte verwenden Sie stattdessen'; + + @override + String get plain_lyrics => 'Einfache Liedtexte'; + + @override + String get tab_instead => 'stattdessen die Tab-Taste verwenden.'; + + @override + String get disclaimer => 'Haftungsausschluss'; + + @override + String get third_party_plugin_dmca_notice => + 'Das Spotube-Team übernimmt keine Verantwortung (auch nicht rechtlicher Art) für Plugins \"Drittanbieter\". Nutzen Sie diese auf eigenes Risiko. Für Fehler/Probleme melden Sie sich bitte beim Plugin-Repository.\n\nWenn ein Plugin \"Drittanbieter\" gegen die ToS/DMCA eines Dienstes bzw. gesetzlicher Vorschriften verstößt, wenden Sie sich bitte an den Plugin-Autor oder die Hosting-Plattform (z. B. GitHub/Codeberg), um Maßnahmen zu ergreifen. Die genannten Plugins (mit \"Drittanbieter\"-Kennzeichnung) werden öffentlich und gemeinschaftlich gepflegt. Wir kuratieren sie nicht und können keine Maßnahmen ergreifen.\n\n'; + + @override + String get input_does_not_match_format => + 'Eingabe entspricht nicht dem geforderten Format'; + + @override + String get plugins => 'Plugins'; + + @override + String get paste_plugin_download_url => + 'Download-URL, GitHub/Codeberg-Repo-URL oder direkten Link zur .smplug-Datei einfügen'; + + @override + String get download_and_install_plugin_from_url => + 'Plugin per URL herunterladen und installieren'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Plugin konnte nicht hinzugefügt werden: $error'; + } + + @override + String get upload_plugin_from_file => 'Plugin per Datei hochladen'; + + @override + String get installed => 'Installiert'; + + @override + String get available_plugins => 'Verfügbare Plugins'; + + @override + String get configure_plugins => + 'Richte deine eigenen Metadatenanbieter- und Audioquellen-Plugins ein'; + + @override + String get audio_scrobblers => 'Audio-Scrobbler'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Quelle: '; + + @override + String get uncompressed => 'Unkomprimiert'; + + @override + String get dab_music_source_description => + 'Für Audiophile. Bietet hochwertige/verlustfreie Audiostreams. Präzises ISRC-basiertes Track-Matching.'; +} diff --git a/lib/l10n/generated/app_localizations_en.dart b/lib/l10n/generated/app_localizations_en.dart new file mode 100644 index 00000000..83a2c24c --- /dev/null +++ b/lib/l10n/generated/app_localizations_en.dart @@ -0,0 +1,1564 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for English (`en`). +class AppLocalizationsEn extends AppLocalizations { + AppLocalizationsEn([String locale = 'en']) : super(locale); + + @override + String get guest => 'Guest'; + + @override + String get browse => 'Browse'; + + @override + String get search => 'Search'; + + @override + String get library => 'Library'; + + @override + String get lyrics => 'Lyrics'; + + @override + String get settings => 'Settings'; + + @override + String get genre_categories_filter => 'Filter categories or genres...'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Personalized'; + + @override + String get featured => 'Featured'; + + @override + String get new_releases => 'New Releases'; + + @override + String get songs => 'Songs'; + + @override + String playing_track(Object track) { + return 'Playing $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'This will clear the current queue. $track_length tracks will be removed\nDo you want to continue?'; + } + + @override + String get load_more => 'Load more'; + + @override + String get playlists => 'Playlists'; + + @override + String get artists => 'Artists'; + + @override + String get albums => 'Albums'; + + @override + String get tracks => 'Tracks'; + + @override + String get downloads => 'Downloads'; + + @override + String get filter_playlists => 'Filter your playlists...'; + + @override + String get liked_tracks => 'Liked Tracks'; + + @override + String get liked_tracks_description => 'All your liked tracks'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Create a playlist'; + + @override + String get update_playlist => 'Update playlist'; + + @override + String get create => 'Create'; + + @override + String get cancel => 'Cancel'; + + @override + String get update => 'Update'; + + @override + String get playlist_name => 'Playlist Name'; + + @override + String get name_of_playlist => 'Name of the playlist'; + + @override + String get description => 'Description'; + + @override + String get public => 'Public'; + + @override + String get collaborative => 'Collaborative'; + + @override + String get search_local_tracks => 'Search local tracks...'; + + @override + String get play => 'Play'; + + @override + String get delete => 'Delete'; + + @override + String get none => 'None'; + + @override + String get sort_a_z => 'Sort by A-Z'; + + @override + String get sort_z_a => 'Sort by Z-A'; + + @override + String get sort_artist => 'Sort by Artist'; + + @override + String get sort_album => 'Sort by Album'; + + @override + String get sort_duration => 'Sort by Duration'; + + @override + String get sort_tracks => 'Sort Tracks'; + + @override + String currently_downloading(Object tracks_length) { + return 'Currently Downloading ($tracks_length)'; + } + + @override + String get cancel_all => 'Cancel All'; + + @override + String get filter_artist => 'Filter artists...'; + + @override + String followers(Object followers) { + return '$followers Followers'; + } + + @override + String get add_artist_to_blacklist => 'Add artist to blacklist'; + + @override + String get top_tracks => 'Top Tracks'; + + @override + String get fans_also_like => 'Fans also like'; + + @override + String get loading => 'Loading...'; + + @override + String get artist => 'Artist'; + + @override + String get blacklisted => 'Blacklisted'; + + @override + String get following => 'Following'; + + @override + String get follow => 'Follow'; + + @override + String get artist_url_copied => 'Artist URL copied to clipboard'; + + @override + String added_to_queue(Object tracks) { + return 'Added $tracks tracks to queue'; + } + + @override + String get filter_albums => 'Filter albums...'; + + @override + String get synced => 'Synced'; + + @override + String get plain => 'Plain'; + + @override + String get shuffle => 'Shuffle'; + + @override + String get search_tracks => 'Search tracks...'; + + @override + String get released => 'Released'; + + @override + String error(Object error) { + return 'Error $error'; + } + + @override + String get title => 'Title'; + + @override + String get time => 'Time'; + + @override + String get more_actions => 'More actions'; + + @override + String download_count(Object count) { + return 'Download ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Add ($count) to Playlist'; + } + + @override + String add_count_to_queue(Object count) { + return 'Add ($count) to Queue'; + } + + @override + String play_count_next(Object count) { + return 'Play ($count) next'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return 'Copied $data to clipboard'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Add $track to following Playlists'; + } + + @override + String get add => 'Add'; + + @override + String added_track_to_queue(Object track) { + return 'Added $track to queue'; + } + + @override + String get add_to_queue => 'Add to queue'; + + @override + String track_will_play_next(Object track) { + return '$track will play next'; + } + + @override + String get play_next => 'Play next'; + + @override + String removed_track_from_queue(Object track) { + return 'Removed $track from queue'; + } + + @override + String get remove_from_queue => 'Remove from queue'; + + @override + String get remove_from_favorites => 'Remove from favorites'; + + @override + String get save_as_favorite => 'Save as favorite'; + + @override + String get add_to_playlist => 'Add to playlist'; + + @override + String get remove_from_playlist => 'Remove from playlist'; + + @override + String get add_to_blacklist => 'Add to blacklist'; + + @override + String get remove_from_blacklist => 'Remove from blacklist'; + + @override + String get share => 'Share'; + + @override + String get mini_player => 'Mini Player'; + + @override + String get slide_to_seek => 'Slide to seek forward or backward'; + + @override + String get shuffle_playlist => 'Shuffle playlist'; + + @override + String get unshuffle_playlist => 'Unshuffle playlist'; + + @override + String get previous_track => 'Previous track'; + + @override + String get next_track => 'Next track'; + + @override + String get pause_playback => 'Pause Playback'; + + @override + String get resume_playback => 'Resume Playback'; + + @override + String get loop_track => 'Loop track'; + + @override + String get no_loop => 'No loop'; + + @override + String get repeat_playlist => 'Repeat playlist'; + + @override + String get queue => 'Queue'; + + @override + String get alternative_track_sources => 'Alternative track sources'; + + @override + String get download_track => 'Download track'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks tracks in queue'; + } + + @override + String get clear_all => 'Clear all'; + + @override + String get show_hide_ui_on_hover => 'Show/Hide UI on hover'; + + @override + String get always_on_top => 'Always on top'; + + @override + String get exit_mini_player => 'Exit Mini player'; + + @override + String get download_location => 'Download location'; + + @override + String get local_library => 'Local library'; + + @override + String get add_library_location => 'Add to library'; + + @override + String get remove_library_location => 'Remove from library'; + + @override + String get account => 'Account'; + + @override + String get logout => 'Logout'; + + @override + String get logout_of_this_account => 'Logout of this account'; + + @override + String get language_region => 'Language & Region'; + + @override + String get language => 'Language'; + + @override + String get system_default => 'System Default'; + + @override + String get market_place_region => 'Marketplace Region'; + + @override + String get recommendation_country => 'Recommendation Country'; + + @override + String get appearance => 'Appearance'; + + @override + String get layout_mode => 'Layout Mode'; + + @override + String get override_layout_settings => + 'Override responsive layout mode settings'; + + @override + String get adaptive => 'Adaptive'; + + @override + String get compact => 'Compact'; + + @override + String get extended => 'Extended'; + + @override + String get theme => 'Theme'; + + @override + String get dark => 'Dark'; + + @override + String get light => 'Light'; + + @override + String get system => 'System'; + + @override + String get accent_color => 'Accent Color'; + + @override + String get sync_album_color => 'Sync album color'; + + @override + String get sync_album_color_description => + 'Uses the dominant color of the album art as the accent color'; + + @override + String get playback => 'Playback'; + + @override + String get audio_quality => 'Audio Quality'; + + @override + String get high => 'High'; + + @override + String get low => 'Low'; + + @override + String get pre_download_play => 'Pre-download and play'; + + @override + String get pre_download_play_description => + 'Instead of streaming audio, download bytes and play instead (Recommended for higher bandwidth users)'; + + @override + String get skip_non_music => 'Skip non-music segments (SponsorBlock)'; + + @override + String get blacklist_description => 'Blacklisted tracks and artists'; + + @override + String get wait_for_download_to_finish => + 'Please wait for the current download to finish'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Close Behavior'; + + @override + String get close => 'Close'; + + @override + String get minimize_to_tray => 'Minimize to tray'; + + @override + String get show_tray_icon => 'Show System tray icon'; + + @override + String get about => 'About'; + + @override + String get u_love_spotube => 'We know you love Spotube'; + + @override + String get check_for_updates => 'Check for updates'; + + @override + String get about_spotube => 'About Spotube'; + + @override + String get blacklist => 'Blacklist'; + + @override + String get please_sponsor => 'Please Sponsor/Donate'; + + @override + String get spotube_description => + 'Open source extensible music streaming platform and app, based on BYOMM (Bring your own music metadata) concept'; + + @override + String get version => 'Version'; + + @override + String get build_number => 'Build Number'; + + @override + String get founder => 'Founder'; + + @override + String get repository => 'Repository'; + + @override + String get bug_issues => 'Bug+Issues'; + + @override + String get made_with => 'Made with ❤️ in Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'License'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Don\'t worry, any of your credentials won\'t be collected or shared with anyone'; + + @override + String get know_how_to_login => 'Don\'t know how to do this?'; + + @override + String get follow_step_by_step_guide => 'Follow along the Step by Step guide'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => 'Please fill in all the fields'; + + @override + String get submit => 'Submit'; + + @override + String get exit => 'Exit'; + + @override + String get previous => 'Previous'; + + @override + String get next => 'Next'; + + @override + String get done => 'Done'; + + @override + String get step_1 => 'Step 1'; + + @override + String get first_go_to => 'First, Go to'; + + @override + String get something_went_wrong => 'Something went wrong'; + + @override + String get piped_instance => 'Piped Server Instance'; + + @override + String get piped_description => + 'The Piped server instance to use for track matching'; + + @override + String get piped_warning => + 'Some of them might not work well. So use at your own risk'; + + @override + String get invidious_instance => 'Invidious Server Instance'; + + @override + String get invidious_description => + 'The Invidious server instance to use for track matching'; + + @override + String get invidious_warning => + 'Some of them might not work well. So use at your own risk'; + + @override + String get generate => 'Generate'; + + @override + String track_exists(Object track) { + return 'Track $track already exists'; + } + + @override + String get replace_downloaded_tracks => 'Replace all downloaded tracks'; + + @override + String get skip_download_tracks => 'Skip downloading all downloaded tracks'; + + @override + String get do_you_want_to_replace => + 'Do you want to replace the existing track??'; + + @override + String get replace => 'Replace'; + + @override + String get skip => 'Skip'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Select up to $count $type'; + } + + @override + String get select_genres => 'Select Genres'; + + @override + String get add_genres => 'Add Genres'; + + @override + String get country => 'Country'; + + @override + String get number_of_tracks_generate => 'Number of tracks to generate'; + + @override + String get acousticness => 'Acousticness'; + + @override + String get danceability => 'Danceability'; + + @override + String get energy => 'Energy'; + + @override + String get instrumentalness => 'Instrumentalness'; + + @override + String get liveness => 'Liveness'; + + @override + String get loudness => 'Loudness'; + + @override + String get speechiness => 'Speechiness'; + + @override + String get valence => 'Valence'; + + @override + String get popularity => 'Popularity'; + + @override + String get key => 'Key'; + + @override + String get duration => 'Duration (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'Time Signature'; + + @override + String get short => 'Short'; + + @override + String get medium => 'Medium'; + + @override + String get long => 'Long'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Target'; + + @override + String get moderate => 'Moderate'; + + @override + String get deselect_all => 'Deselect All'; + + @override + String get select_all => 'Select All'; + + @override + String get are_you_sure => 'Are you sure?'; + + @override + String get generating_playlist => 'Generating your custom playlist...'; + + @override + String selected_count_tracks(Object count) { + return 'Selected $count tracks'; + } + + @override + String get download_warning => + 'If you download all Tracks at bulk you\'re clearly pirating Music & causing damage to the creative society of Music. I hope you are aware of this. Always, try respecting & supporting Artist\'s hard work'; + + @override + String get download_ip_ban_warning => + 'BTW, your IP can get blocked on YouTube due excessive download requests than usual. IP block means you can\'t use YouTube (even if you\'re logged in) for at least 2-3 months from that IP device. And Spotube doesn\'t hold any responsibility if this ever happens'; + + @override + String get by_clicking_accept_terms => + 'By clicking \'accept\' you agree to following terms:'; + + @override + String get download_agreement_1 => 'I know I\'m pirating Music. I\'m bad'; + + @override + String get download_agreement_2 => + 'I\'ll support the Artist wherever I can and I\'m only doing this because I don\'t have money to buy their art'; + + @override + String get download_agreement_3 => + 'I\'m completely aware that my IP can get blocked on YouTube & I don\'t hold Spotube or his owners/contributors responsible for any accidents caused by my current action'; + + @override + String get decline => 'Decline'; + + @override + String get accept => 'Accept'; + + @override + String get details => 'Details'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Channel'; + + @override + String get likes => 'Likes'; + + @override + String get dislikes => 'Dislikes'; + + @override + String get views => 'Views'; + + @override + String get streamUrl => 'Stream URL'; + + @override + String get stop => 'Stop'; + + @override + String get sort_newest => 'Sort by newest added'; + + @override + String get sort_oldest => 'Sort by oldest added'; + + @override + String get sleep_timer => 'Sleep Timer'; + + @override + String mins(Object minutes) { + return '$minutes Minutes'; + } + + @override + String hours(Object hours) { + return '$hours Hours'; + } + + @override + String hour(Object hours) { + return '$hours Hour'; + } + + @override + String get custom_hours => 'Custom Hours'; + + @override + String get logs => 'Logs'; + + @override + String get developers => 'Developers'; + + @override + String get not_logged_in => 'You\'re not logged in'; + + @override + String get search_mode => 'Search Mode'; + + @override + String get audio_source => 'Audio Source'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Failed to encrypt'; + + @override + String get encryption_failed_warning => + 'Spotube uses encryption to securely store your data. But failed to do so. So it\'ll fallback to insecure storage\nIf you\'re using linux, please make sure you\'ve any secret-service (gnome-keyring, kde-wallet, keepassxc etc) installed'; + + @override + String get querying_info => 'Querying info...'; + + @override + String get piped_api_down => 'Piped API is down'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'The Piped instance $pipedInstance is currently down\n\nEither change the instance or change the \'API type\' to official YouTube API\n\nMake sure to restart the app after change'; + } + + @override + String get you_are_offline => 'You are currently offline'; + + @override + String get connection_restored => 'Your internet connection was restored'; + + @override + String get use_system_title_bar => 'Use system title bar'; + + @override + String get crunching_results => 'Crunching results...'; + + @override + String get search_to_get_results => 'Search to get results'; + + @override + String get use_amoled_mode => 'Pitch black dark theme'; + + @override + String get pitch_dark_theme => 'AMOLED Mode'; + + @override + String get normalize_audio => 'Normalize audio'; + + @override + String get change_cover => 'Change cover'; + + @override + String get add_cover => 'Add cover'; + + @override + String get restore_defaults => 'Restore defaults'; + + @override + String get download_music_format => 'Download music format'; + + @override + String get streaming_music_format => 'Streaming music format'; + + @override + String get download_music_quality => 'Download music quality'; + + @override + String get streaming_music_quality => 'Streaming music quality'; + + @override + String get login_with_lastfm => 'Login with Last.fm'; + + @override + String get connect => 'Connect'; + + @override + String get disconnect_lastfm => 'Disconnect Last.fm'; + + @override + String get disconnect => 'Disconnect'; + + @override + String get username => 'Username'; + + @override + String get password => 'Password'; + + @override + String get login => 'Login'; + + @override + String get login_with_your_lastfm => 'Login with your Last.fm account'; + + @override + String get scrobble_to_lastfm => 'Scrobble to Last.fm'; + + @override + String get go_to_album => 'Go to Album'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'Browse All'; + + @override + String get genres => 'Genres'; + + @override + String get explore_genres => 'Explore Genres'; + + @override + String get friends => 'Friends'; + + @override + String get no_lyrics_available => 'Sorry, unable find lyrics for this track'; + + @override + String get start_a_radio => 'Start a Radio'; + + @override + String get how_to_start_radio => 'How do you want to start the radio?'; + + @override + String get replace_queue_question => + 'Do you want to replace the current queue or append to it?'; + + @override + String get endless_playback => 'Endless Playback'; + + @override + String get delete_playlist => 'Delete Playlist'; + + @override + String get delete_playlist_confirmation => + 'Are you sure you want to delete this playlist?'; + + @override + String get local_tracks => 'Local Tracks'; + + @override + String get local_tab => 'Local'; + + @override + String get song_link => 'Song Link'; + + @override + String get skip_this_nonsense => 'Skip this nonsense'; + + @override + String get freedom_of_music => '“Freedom of Music”'; + + @override + String get freedom_of_music_palm => + '“Freedom of Music in the palm of your hand”'; + + @override + String get get_started => 'Let\'s get started'; + + @override + String get youtube_source_description => 'Recommended and works best.'; + + @override + String get piped_source_description => + 'Feeling free? Same as YouTube but a lot free.'; + + @override + String get jiosaavn_source_description => 'Best for South Asian region.'; + + @override + String get invidious_source_description => + 'Similar to Piped but with higher availability.'; + + @override + String highest_quality(Object quality) { + return 'Highest Quality: $quality'; + } + + @override + String get select_audio_source => 'Select Audio Source'; + + @override + String get endless_playback_description => + 'Automatically append new songs\nto the end of the queue'; + + @override + String get choose_your_region => 'Choose your region'; + + @override + String get choose_your_region_description => + 'This will help Spotube show you the right content\nfor your location.'; + + @override + String get choose_your_language => 'Choose your language'; + + @override + String get help_project_grow => 'Help this project grow'; + + @override + String get help_project_grow_description => + 'Spotube is an open-source project. You can help this project grow by contributing to the project, reporting bugs, or suggesting new features.'; + + @override + String get contribute_on_github => 'Contribute on GitHub'; + + @override + String get donate_on_open_collective => 'Donate on Open Collective'; + + @override + String get browse_anonymously => 'Browse Anonymously'; + + @override + String get enable_connect => 'Enable Connect'; + + @override + String get enable_connect_description => 'Control Spotube from other devices'; + + @override + String get devices => 'Devices'; + + @override + String get select => 'Select'; + + @override + String connect_client_alert(Object client) { + return 'You\'re being controlled by $client'; + } + + @override + String get this_device => 'This Device'; + + @override + String get remote => 'Remote'; + + @override + String get stats => 'Stats'; + + @override + String and_n_more(Object count) { + return 'and $count more'; + } + + @override + String get recently_played => 'Recently Played'; + + @override + String get browse_more => 'Browse More'; + + @override + String get no_title => 'No Title'; + + @override + String get not_playing => 'Not playing'; + + @override + String get epic_failure => 'Epic failure!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Added $tracks_length tracks to queue'; + } + + @override + String get spotube_has_an_update => 'Spotube has an update'; + + @override + String get download_now => 'Download Now'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum has been released'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version has been released'; + } + + @override + String get read_the_latest => 'Read the latest '; + + @override + String get release_notes => 'release notes'; + + @override + String get pick_color_scheme => 'Pick color scheme'; + + @override + String get save => 'Save'; + + @override + String get choose_the_device => 'Choose the device:'; + + @override + String get multiple_device_connected => + 'There are multiple device connected.\nChoose the device you want this action to take place'; + + @override + String get nothing_found => 'Nothing found'; + + @override + String get the_box_is_empty => 'The box is empty'; + + @override + String get top_artists => 'Top Artists'; + + @override + String get top_albums => 'Top Albums'; + + @override + String get this_week => 'This week'; + + @override + String get this_month => 'This month'; + + @override + String get last_6_months => 'Last 6 months'; + + @override + String get this_year => 'This year'; + + @override + String get last_2_years => 'Last 2 years'; + + @override + String get all_time => 'All time'; + + @override + String powered_by_provider(Object providerName) { + return 'Powered by $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Followers'; + + @override + String get birthday => 'Birthday'; + + @override + String get subscription => 'Subscription'; + + @override + String get not_born => 'Not born'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profile'; + + @override + String get no_name => 'No Name'; + + @override + String get edit => 'Edit'; + + @override + String get user_profile => 'User Profile'; + + @override + String count_plays(Object count) { + return '$count plays'; + } + + @override + String get streaming_fees_hypothetical => 'Streaming fees (hypothetical)'; + + @override + String get minutes_listened => 'Minutes listened'; + + @override + String get streamed_songs => 'Streamed songs'; + + @override + String count_streams(Object count) { + return '$count streams'; + } + + @override + String get owned_by_you => 'Owned by you'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'Copied $shareUrl to clipboard'; + } + + @override + String get hipotetical_calculation => + '*This is calculated based on average online music streaming platform\'s per stream\npayout of \$0.003 to \$0.005. This is a hypothetical\ncalculation to give user insight about how much they\nwould have paid to the artists if they were to listen\ntheir song in different music streaming platform.'; + + @override + String count_mins(Object minutes) { + return '$minutes mins'; + } + + @override + String get summary_minutes => 'minutes'; + + @override + String get summary_listened_to_music => 'Listened to music'; + + @override + String get summary_songs => 'songs'; + + @override + String get summary_streamed_overall => 'Streamed overall'; + + @override + String get summary_owed_to_artists => 'Owed to artists\nthis month'; + + @override + String get summary_artists => 'artist\'s'; + + @override + String get summary_music_reached_you => 'Music reached you'; + + @override + String get summary_full_albums => 'full albums'; + + @override + String get summary_got_your_love => 'Got your love'; + + @override + String get summary_playlists => 'playlists'; + + @override + String get summary_were_on_repeat => 'Were on repeat'; + + @override + String total_money(Object money) { + return 'Total $money'; + } + + @override + String get webview_not_found => 'Webview not found'; + + @override + String get webview_not_found_description => + 'No webview runtime is installed in your device.\nIf it\'s installed make sure it\'s in the Environment PATH\n\nAfter installing, restart the app'; + + @override + String get unsupported_platform => 'Unsupported platform'; + + @override + String get cache_music => 'Cache music'; + + @override + String get open => 'Open'; + + @override + String get cache_folder => 'Cache folder'; + + @override + String get export => 'Export'; + + @override + String get clear_cache => 'Clear cache'; + + @override + String get clear_cache_confirmation => 'Do you want to clear the cache?'; + + @override + String get export_cache_files => 'Export Cached Files'; + + @override + String found_n_files(Object count) { + return 'Found $count files'; + } + + @override + String get export_cache_confirmation => + 'Do you want to export these files to'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Exported $filesExported out of $files files'; + } + + @override + String get undo => 'Undo'; + + @override + String get download_all => 'Download all'; + + @override + String get add_all_to_playlist => 'Add all to playlist'; + + @override + String get add_all_to_queue => 'Add all to queue'; + + @override + String get play_all_next => 'Play all next'; + + @override + String get pause => 'Pause'; + + @override + String get view_all => 'View all'; + + @override + String get no_tracks_added_yet => + 'Looks like you haven\'t added any tracks yet'; + + @override + String get no_tracks => 'Looks like there are no tracks here'; + + @override + String get no_tracks_listened_yet => + 'Looks like you haven\'t listened to anything yet'; + + @override + String get not_following_artists => 'You\'re not following any artists'; + + @override + String get no_favorite_albums_yet => + 'Looks like you haven\'t added any albums to your favorites yet'; + + @override + String get no_logs_found => 'No logs found'; + + @override + String get youtube_engine => 'YouTube Engine'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine is not installed'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine is not installed in your system.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Make sure it\'s available in the PATH variable or\nset the absolute path to the $engine executable below'; + } + + @override + String get youtube_engine_unix_issue_message => + 'In macOS/Linux/unix like OS\'s, setting path on .zshrc/.bashrc/.bash_profile etc. won\'t work.\nYou need to set the path in the shell configuration file'; + + @override + String get download => 'Download'; + + @override + String get file_not_found => 'File not found'; + + @override + String get custom => 'Custom'; + + @override + String get add_custom_url => 'Add custom URL'; + + @override + String get edit_port => 'Edit port'; + + @override + String get port_helper_msg => + 'Default is -1 which indicates random number. If you\'ve firewall configured, setting this is recommended.'; + + @override + String connect_request(Object client) { + return 'Allow $client to connect?'; + } + + @override + String get connection_request_denied => + 'Connection denied. User denied access.'; + + @override + String get an_error_occurred => 'An error occurred'; + + @override + String get copy_to_clipboard => 'Copy to clipboard'; + + @override + String get view_logs => 'View logs'; + + @override + String get retry => 'Retry'; + + @override + String get no_default_metadata_provider_selected => + 'You\'ve no default metadata provider set'; + + @override + String get manage_metadata_providers => 'Manage metadata providers'; + + @override + String get open_link_in_browser => 'Open Link in Browser?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Do you want to open the following link'; + + @override + String get unsafe_url_warning => + 'It can be unsafe to open links from untrusted sources. Be cautious!\nYou can also copy the link to your clipboard.'; + + @override + String get copy_link => 'Copy Link'; + + @override + String get building_your_timeline => + 'Building your timeline based on your listenings...'; + + @override + String get official => 'Official'; + + @override + String author_name(Object author) { + return 'Author: $author'; + } + + @override + String get third_party => 'Third-party'; + + @override + String get plugin_requires_authentication => 'Plugin requires authentication'; + + @override + String get update_available => 'Update available'; + + @override + String get supports_scrobbling => 'Supports scrobbling'; + + @override + String get plugin_scrobbling_info => + 'This plugin scrobbles your music to generate your listening history.'; + + @override + String get default_metadata_source => 'Default metadata source'; + + @override + String get set_default_metadata_source => 'Set default metadata source'; + + @override + String get default_audio_source => 'Default audio source'; + + @override + String get set_default_audio_source => 'Set default audio source'; + + @override + String get set_default => 'Set default'; + + @override + String get support => 'Support'; + + @override + String get support_plugin_development => 'Support plugin development'; + + @override + String can_access_name_api(Object name) { + return '- Can access **$name** API'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Do you want to install this plugin?'; + + @override + String get third_party_plugin_warning => + 'This plugin is from a third-party repository. Please ensure you trust the source before installing.'; + + @override + String get author => 'Author'; + + @override + String get this_plugin_can_do_following => 'This plugin can do following'; + + @override + String get install => 'Install'; + + @override + String get install_a_metadata_provider => 'Install a Metadata Provider'; + + @override + String get no_tracks_playing => 'No Track being played currently'; + + @override + String get synced_lyrics_not_available => + 'Synced lyrics are not available for this song. Please use the'; + + @override + String get plain_lyrics => 'Plain Lyrics'; + + @override + String get tab_instead => 'tab instead.'; + + @override + String get disclaimer => 'Disclaimer'; + + @override + String get third_party_plugin_dmca_notice => + 'The Spotube team does not hold any responsibility (including legal) for any \"Third-party\" plugins.\nPlease use them at your own risk. For any bugs/issues, please report them to the plugin repository.\n\nIf any \"Third-party\" plugin is breaking ToS/DMCA of any service/legal entity, please ask the \"Third-party\" plugin author or the hosting platform .e.g GitHub/Codeberg to take action. Above listed (\"Third-party\" labelled) are all public/community maintained plugins. We\'re not curating them, so we cannot take any action on them.\n\n'; + + @override + String get input_does_not_match_format => + 'Input doesn\'t match the required format'; + + @override + String get plugins => 'Plugins'; + + @override + String get paste_plugin_download_url => + 'Paste download url or GitHub/Codeberg repo url or direct link to .smplug file'; + + @override + String get download_and_install_plugin_from_url => + 'Download and install plugin from url'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Failed to add plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'Upload plugin from file'; + + @override + String get installed => 'Installed'; + + @override + String get available_plugins => 'Available plugins'; + + @override + String get configure_plugins => + 'Configure your own metadata provider and audio source plugins'; + + @override + String get audio_scrobblers => 'Audio Scrobblers'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Source: '; + + @override + String get uncompressed => 'Uncompressed'; + + @override + String get dab_music_source_description => + 'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.'; +} diff --git a/lib/l10n/generated/app_localizations_es.dart b/lib/l10n/generated/app_localizations_es.dart new file mode 100644 index 00000000..0fcd6739 --- /dev/null +++ b/lib/l10n/generated/app_localizations_es.dart @@ -0,0 +1,1580 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Spanish Castilian (`es`). +class AppLocalizationsEs extends AppLocalizations { + AppLocalizationsEs([String locale = 'es']) : super(locale); + + @override + String get guest => 'Invitado'; + + @override + String get browse => 'Explorar'; + + @override + String get search => 'Buscar'; + + @override + String get library => 'Biblioteca'; + + @override + String get lyrics => 'Letras'; + + @override + String get settings => 'Configuración'; + + @override + String get genre_categories_filter => 'Filtrar categorías o géneros...'; + + @override + String get genre => 'Género'; + + @override + String get personalized => 'Personalizado'; + + @override + String get featured => 'Destacado'; + + @override + String get new_releases => 'Nuevos Lanzamientos'; + + @override + String get songs => 'Canciones'; + + @override + String playing_track(Object track) { + return 'Reproduciendo $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Esto eliminará la lista actual. Se eliminarán $track_length canciones.\n¿Deseas continuar?'; + } + + @override + String get load_more => 'Cargar más'; + + @override + String get playlists => 'Listas de reproducción'; + + @override + String get artists => 'Artistas'; + + @override + String get albums => 'Álbumes'; + + @override + String get tracks => 'Canciones'; + + @override + String get downloads => 'Descargas'; + + @override + String get filter_playlists => 'Filtrar tus listas de reproducción...'; + + @override + String get liked_tracks => 'Canciones Favoritas'; + + @override + String get liked_tracks_description => 'Todas tus canciones favoritas'; + + @override + String get playlist => 'Lista de reproducción'; + + @override + String get create_a_playlist => 'Crear una lista de reproducción'; + + @override + String get update_playlist => 'Actualizar lista de reproducción'; + + @override + String get create => 'Crear'; + + @override + String get cancel => 'Cancelar'; + + @override + String get update => 'Actualizar'; + + @override + String get playlist_name => 'Nombre de la lista'; + + @override + String get name_of_playlist => 'Nombre de la lista'; + + @override + String get description => 'Descripción'; + + @override + String get public => 'Pública'; + + @override + String get collaborative => 'Colaborativa'; + + @override + String get search_local_tracks => 'Buscar canciones locales...'; + + @override + String get play => 'Reproducir'; + + @override + String get delete => 'Eliminar'; + + @override + String get none => 'Ninguno'; + + @override + String get sort_a_z => 'Ordenar de la A a la Z'; + + @override + String get sort_z_a => 'Ordenar de la Z a la A'; + + @override + String get sort_artist => 'Ordenar por Artista'; + + @override + String get sort_album => 'Ordenar por Álbum'; + + @override + String get sort_duration => 'Ordenar por Duración'; + + @override + String get sort_tracks => 'Ordenar Canciones'; + + @override + String currently_downloading(Object tracks_length) { + return 'Descargando en curso ($tracks_length)'; + } + + @override + String get cancel_all => 'Cancelar todo'; + + @override + String get filter_artist => 'Filtrar artistas...'; + + @override + String followers(Object followers) { + return '$followers Seguidores'; + } + + @override + String get add_artist_to_blacklist => 'Agregar artista a la lista negra'; + + @override + String get top_tracks => 'Mejores Canciones'; + + @override + String get fans_also_like => 'A los fans también les gusta'; + + @override + String get loading => 'Cargando...'; + + @override + String get artist => 'Artista'; + + @override + String get blacklisted => 'En la lista negra'; + + @override + String get following => 'Siguiendo'; + + @override + String get follow => 'Seguir'; + + @override + String get artist_url_copied => 'URL del artista copiada al portapapeles'; + + @override + String added_to_queue(Object tracks) { + return 'Agregadas $tracks canciones a la lista'; + } + + @override + String get filter_albums => 'Filtrar álbumes...'; + + @override + String get synced => 'Sincronizado'; + + @override + String get plain => 'Normal'; + + @override + String get shuffle => 'Aleatorio'; + + @override + String get search_tracks => 'Buscar canciones...'; + + @override + String get released => 'Lanzado'; + + @override + String error(Object error) { + return 'Error $error'; + } + + @override + String get title => 'Título'; + + @override + String get time => 'Duración'; + + @override + String get more_actions => 'Más acciones'; + + @override + String download_count(Object count) { + return 'Descargas ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Agregar ($count) a la lista'; + } + + @override + String add_count_to_queue(Object count) { + return 'Agregar ($count) a la lista'; + } + + @override + String play_count_next(Object count) { + return 'Reproducir ($count) a continuación'; + } + + @override + String get album => 'Álbum'; + + @override + String copied_to_clipboard(Object data) { + return '$data copiado al portapapeles'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Agregar $track a las listas de reproducción siguientes'; + } + + @override + String get add => 'Agregar'; + + @override + String added_track_to_queue(Object track) { + return '$track agregada a la lista'; + } + + @override + String get add_to_queue => 'Agregar a la lista'; + + @override + String track_will_play_next(Object track) { + return '$track se reproducirá a continuación'; + } + + @override + String get play_next => 'Reproducir a continuación'; + + @override + String removed_track_from_queue(Object track) { + return '$track eliminada de la lista'; + } + + @override + String get remove_from_queue => 'Eliminar de la lista'; + + @override + String get remove_from_favorites => 'Eliminar de favoritos'; + + @override + String get save_as_favorite => 'Guardar como favorito'; + + @override + String get add_to_playlist => 'Agregar a la lista'; + + @override + String get remove_from_playlist => 'Eliminar de la lista'; + + @override + String get add_to_blacklist => 'Agregar a la lista negra'; + + @override + String get remove_from_blacklist => 'Eliminar de la lista negra'; + + @override + String get share => 'Compartir'; + + @override + String get mini_player => 'Reproductor Mini'; + + @override + String get slide_to_seek => 'Desliza para buscar adelante o atrás'; + + @override + String get shuffle_playlist => 'Reproducir lista en orden aleatorio'; + + @override + String get unshuffle_playlist => 'Desactivar reproducción aleatoria'; + + @override + String get previous_track => 'Pista anterior'; + + @override + String get next_track => 'Pista siguiente'; + + @override + String get pause_playback => 'Pausar reproducción'; + + @override + String get resume_playback => 'Reanudar reproducción'; + + @override + String get loop_track => 'Repetir pista'; + + @override + String get no_loop => 'Sin bucle'; + + @override + String get repeat_playlist => 'Repetir lista'; + + @override + String get queue => 'Lista'; + + @override + String get alternative_track_sources => 'Fuentes alternativas de canciones'; + + @override + String get download_track => 'Descargar canción'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks canciones en la lista'; + } + + @override + String get clear_all => 'Limpiar todo'; + + @override + String get show_hide_ui_on_hover => + 'Mostrar/Ocultar interfaz al pasar el cursor'; + + @override + String get always_on_top => 'Siempre visible'; + + @override + String get exit_mini_player => 'Salir del reproductor mini'; + + @override + String get download_location => 'Ubicación de descargas'; + + @override + String get local_library => 'Biblioteca local'; + + @override + String get add_library_location => 'Añadir a la biblioteca'; + + @override + String get remove_library_location => 'Eliminar de la biblioteca'; + + @override + String get account => 'Cuenta'; + + @override + String get logout => 'Cerrar sesión'; + + @override + String get logout_of_this_account => 'Cerrar sesión de esta cuenta'; + + @override + String get language_region => 'Idioma y Región'; + + @override + String get language => 'Idioma'; + + @override + String get system_default => 'Predeterminado del sistema'; + + @override + String get market_place_region => 'Región de la tienda'; + + @override + String get recommendation_country => 'País de recomendación'; + + @override + String get appearance => 'Apariencia'; + + @override + String get layout_mode => 'Modo de diseño'; + + @override + String get override_layout_settings => + 'Anular la configuración del modo de diseño responsive'; + + @override + String get adaptive => 'Adaptable'; + + @override + String get compact => 'Compacto'; + + @override + String get extended => 'Extendido'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Oscuro'; + + @override + String get light => 'Claro'; + + @override + String get system => 'Sistema'; + + @override + String get accent_color => 'Color de acento'; + + @override + String get sync_album_color => 'Sincronizar color del álbum'; + + @override + String get sync_album_color_description => + 'Usa el color dominante del arte del álbum como color de acento'; + + @override + String get playback => 'Reproducción'; + + @override + String get audio_quality => 'Calidad de audio'; + + @override + String get high => 'Alta'; + + @override + String get low => 'Baja'; + + @override + String get pre_download_play => 'Pre-descargar y reproducir'; + + @override + String get pre_download_play_description => + 'En lugar de transmitir audio, descarga bytes y reproduce en su lugar (recomendado para usuarios con mayor ancho de banda)'; + + @override + String get skip_non_music => + 'Omitir segmentos que no son música (SponsorBlock)'; + + @override + String get blacklist_description => 'Canciones y artistas en la lista negra'; + + @override + String get wait_for_download_to_finish => + 'Por favor, espera a que termine la descarga actual'; + + @override + String get desktop => 'Escritorio'; + + @override + String get close_behavior => 'Comportamiento al cerrar'; + + @override + String get close => 'Cerrar'; + + @override + String get minimize_to_tray => 'Minimizar en la bandeja del sistema'; + + @override + String get show_tray_icon => 'Mostrar icono en la bandeja del sistema'; + + @override + String get about => 'Acerca de'; + + @override + String get u_love_spotube => 'Sabemos que te encanta Spotube'; + + @override + String get check_for_updates => 'Buscar actualizaciones'; + + @override + String get about_spotube => 'Acerca de Spotube'; + + @override + String get blacklist => 'Lista negra'; + + @override + String get please_sponsor => 'Por favor, apoya/dona'; + + @override + String get spotube_description => + 'Spotube, un cliente ligero, multiplataforma y gratuito de Spotify'; + + @override + String get version => 'Versión'; + + @override + String get build_number => 'Número de compilación'; + + @override + String get founder => 'Fundador'; + + @override + String get repository => 'Repositorio'; + + @override + String get bug_issues => 'Errores y problemas'; + + @override + String get made_with => 'Hecho con ❤️ en Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licencia'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'No te preocupes, tus credenciales no serán recopiladas ni compartidas con nadie'; + + @override + String get know_how_to_login => '¿No sabes cómo hacerlo?'; + + @override + String get follow_step_by_step_guide => 'Sigue la guía paso a paso'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Por favor, completa todos los campos'; + + @override + String get submit => 'Enviar'; + + @override + String get exit => 'Salir'; + + @override + String get previous => 'Anterior'; + + @override + String get next => 'Siguiente'; + + @override + String get done => 'Listo'; + + @override + String get step_1 => 'Paso 1'; + + @override + String get first_go_to => 'Primero, ve a'; + + @override + String get something_went_wrong => 'Algo salió mal'; + + @override + String get piped_instance => 'Instancia del servidor Piped'; + + @override + String get piped_description => + 'La instancia del servidor Piped a utilizar para la coincidencia de pistas'; + + @override + String get piped_warning => + 'Algunas pueden no funcionar bien, úsalas bajo tu propio riesgo'; + + @override + String get invidious_instance => 'Instancia del Servidor Invidious'; + + @override + String get invidious_description => + 'La instancia del servidor Invidious para identificar pistas'; + + @override + String get invidious_warning => + 'Algunas instancias podrían no funcionar bien. Úselas bajo su propio riesgo'; + + @override + String get generate => 'Generar'; + + @override + String track_exists(Object track) { + return 'La canción $track ya existe'; + } + + @override + String get replace_downloaded_tracks => + 'Reemplazar todas las canciones descargadas'; + + @override + String get skip_download_tracks => + 'Omitir la descarga de todas las canciones descargadas'; + + @override + String get do_you_want_to_replace => + '¿Deseas reemplazar la canción existente?'; + + @override + String get replace => 'Reemplazar'; + + @override + String get skip => 'Omitir'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Seleccionar hasta $count $type'; + } + + @override + String get select_genres => 'Seleccionar Géneros'; + + @override + String get add_genres => 'Agregar Géneros'; + + @override + String get country => 'País'; + + @override + String get number_of_tracks_generate => 'Número de canciones a generar'; + + @override + String get acousticness => 'Acousticness'; + + @override + String get danceability => 'Danceability'; + + @override + String get energy => 'Energía'; + + @override + String get instrumentalness => 'Instrumentalidad'; + + @override + String get liveness => 'En vivo'; + + @override + String get loudness => 'Volumen'; + + @override + String get speechiness => 'Habla'; + + @override + String get valence => 'Valencia'; + + @override + String get popularity => 'Popularidad'; + + @override + String get key => 'Tono'; + + @override + String get duration => 'Duración (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Modo'; + + @override + String get time_signature => 'Compás'; + + @override + String get short => 'Corto'; + + @override + String get medium => 'Medio'; + + @override + String get long => 'Largo'; + + @override + String get min => 'Mín.'; + + @override + String get max => 'Máx.'; + + @override + String get target => 'Objetivo'; + + @override + String get moderate => 'Moderado'; + + @override + String get deselect_all => 'Deseleccionar todo'; + + @override + String get select_all => 'Seleccionar todo'; + + @override + String get are_you_sure => '¿Estás seguro?'; + + @override + String get generating_playlist => + 'Generando tu lista de reproducción personalizada...'; + + @override + String selected_count_tracks(Object count) { + return 'Seleccionadas $count canciones'; + } + + @override + String get download_warning => + 'Si descargas todas las canciones de golpe, estás claramente pirateando música y causando daño a la sociedad creativa de la música. Espero que seas consciente de esto y siempre intentes respetar y apoyar el arduo trabajo de los artistas'; + + @override + String get download_ip_ban_warning => + 'Por cierto, tu IP puede ser bloqueada en YouTube debido a solicitudes de descarga excesivas. El bloqueo de IP significa que no podrás usar YouTube (incluso si has iniciado sesión) durante al menos 2-3 meses desde esa dirección IP. Y Spotube no se hace responsable si esto ocurre alguna vez'; + + @override + String get by_clicking_accept_terms => + 'Al hacer clic en \'Aceptar\', aceptas los siguientes términos:'; + + @override + String get download_agreement_1 => 'Sé que estoy pirateando música. Soy malo'; + + @override + String get download_agreement_2 => + 'Apoyaré al artista donde pueda y solo lo hago porque no tengo dinero para comprar su arte'; + + @override + String get download_agreement_3 => + 'Soy completamente consciente de que mi IP puede ser bloqueada en YouTube y no responsabilizo a Spotube ni a sus dueños/contribuyentes por cualquier incidente causado por mi acción actual'; + + @override + String get decline => 'Rechazar'; + + @override + String get accept => 'Aceptar'; + + @override + String get details => 'Detalles'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Canal'; + + @override + String get likes => 'Me gusta'; + + @override + String get dislikes => 'No me gusta'; + + @override + String get views => 'Vistas'; + + @override + String get streamUrl => 'URL del streaming'; + + @override + String get stop => 'Detener'; + + @override + String get sort_newest => 'Ordenar por más recientes'; + + @override + String get sort_oldest => 'Ordenar por más antiguos'; + + @override + String get sleep_timer => 'Temporizador de apagado'; + + @override + String mins(Object minutes) { + return '$minutes minutos'; + } + + @override + String hours(Object hours) { + return '$hours horas'; + } + + @override + String hour(Object hours) { + return '$hours hora'; + } + + @override + String get custom_hours => 'Horas personalizadas'; + + @override + String get logs => 'Registros'; + + @override + String get developers => 'Desarrolladores'; + + @override + String get not_logged_in => 'No has iniciado sesión'; + + @override + String get search_mode => 'Modo de búsqueda'; + + @override + String get audio_source => 'Fuente de audio'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => 'Error al cifrar'; + + @override + String get encryption_failed_warning => + 'Spotube utiliza el cifrado para almacenar sus datos de forma segura. Pero ha fallado. Por lo tanto, volverá a un almacenamiento no seguro\nSi está utilizando Linux, asegúrese de tener instalados servicios secretos como gnome-keyring, kde-wallet y keepassxc'; + + @override + String get querying_info => 'Consultando información...'; + + @override + String get piped_api_down => 'La API de Piped no está disponible'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'La instancia de Piped $pipedInstance no está funcionando en este momento\n\nCambie la instancia o cambie el \'Tipo de API\' a la API oficial de YouTube\n\nAsegúrese de reiniciar la aplicación después del cambio'; + } + + @override + String get you_are_offline => 'Actualmente estás sin conexión'; + + @override + String get connection_restored => 'Se ha restablecido tu conexión a internet'; + + @override + String get use_system_title_bar => 'Usar la barra de título del sistema'; + + @override + String get crunching_results => 'Procesando resultados...'; + + @override + String get search_to_get_results => 'Buscar para obtener resultados'; + + @override + String get use_amoled_mode => 'Usar modo AMOLED'; + + @override + String get pitch_dark_theme => 'Tema oscuro de dart'; + + @override + String get normalize_audio => 'Normalizar audio'; + + @override + String get change_cover => 'Cambiar portada'; + + @override + String get add_cover => 'Agregar portada'; + + @override + String get restore_defaults => 'Restaurar valores predeterminados'; + + @override + String get download_music_format => 'Formato de descarga de música'; + + @override + String get streaming_music_format => 'Formato de transmisión de música'; + + @override + String get download_music_quality => 'Calidad de descarga de música'; + + @override + String get streaming_music_quality => 'Calidad de transmisión de música'; + + @override + String get login_with_lastfm => 'Iniciar sesión con Last.fm'; + + @override + String get connect => 'Conectar'; + + @override + String get disconnect_lastfm => 'Desconectar de Last.fm'; + + @override + String get disconnect => 'Desconectar'; + + @override + String get username => 'Nombre de usuario'; + + @override + String get password => 'Contraseña'; + + @override + String get login => 'Iniciar sesión'; + + @override + String get login_with_your_lastfm => + 'Iniciar sesión con tu cuenta de Last.fm'; + + @override + String get scrobble_to_lastfm => 'Scrobble a Last.fm'; + + @override + String get go_to_album => 'Ir al álbum'; + + @override + String get discord_rich_presence => 'Presencia rica en Discord'; + + @override + String get browse_all => 'Explorar todo'; + + @override + String get genres => 'Géneros'; + + @override + String get explore_genres => 'Explorar géneros'; + + @override + String get friends => 'Amigos'; + + @override + String get no_lyrics_available => + 'Lo siento, no se pueden encontrar las letras de esta pista'; + + @override + String get start_a_radio => 'Iniciar una Radio'; + + @override + String get how_to_start_radio => '¿Cómo quieres iniciar la radio?'; + + @override + String get replace_queue_question => + '¿Quieres reemplazar la lista de reproducción actual o añadir a ella?'; + + @override + String get endless_playback => 'Reproducción Infinita'; + + @override + String get delete_playlist => 'Eliminar Lista de Reproducción'; + + @override + String get delete_playlist_confirmation => + '¿Estás seguro de que quieres eliminar esta lista de reproducción?'; + + @override + String get local_tracks => 'Pistas Locales'; + + @override + String get local_tab => 'Local'; + + @override + String get song_link => 'Enlace de la Canción'; + + @override + String get skip_this_nonsense => 'Saltar esta tontería'; + + @override + String get freedom_of_music => '“Libertad de la Música”'; + + @override + String get freedom_of_music_palm => + '“Libertad de la Música en la palma de tu mano”'; + + @override + String get get_started => 'Empecemos'; + + @override + String get youtube_source_description => 'Recomendado y funciona mejor.'; + + @override + String get piped_source_description => + '¿Te sientes libre? Igual que YouTube pero más libre.'; + + @override + String get jiosaavn_source_description => + 'Lo mejor para la región del sur de Asia.'; + + @override + String get invidious_source_description => + 'Similar a Piped, pero con mayor disponibilidad'; + + @override + String highest_quality(Object quality) { + return 'Mayor Calidad: $quality'; + } + + @override + String get select_audio_source => 'Seleccionar Fuente de Audio'; + + @override + String get endless_playback_description => + 'Añadir automáticamente nuevas canciones\nal final de la cola de reproducción'; + + @override + String get choose_your_region => 'Elige tu región'; + + @override + String get choose_your_region_description => + 'Esto ayudará a Spotube a mostrarte el contenido adecuado\npara tu ubicación.'; + + @override + String get choose_your_language => 'Elige tu idioma'; + + @override + String get help_project_grow => 'Ayuda a que este proyecto crezca'; + + @override + String get help_project_grow_description => + 'Spotube es un proyecto de código abierto. Puedes ayudar a que este proyecto crezca contribuyendo al proyecto, informando errores o sugiriendo nuevas funciones.'; + + @override + String get contribute_on_github => 'Contribuir en GitHub'; + + @override + String get donate_on_open_collective => 'Donar en Open Collective'; + + @override + String get browse_anonymously => 'Navegar Anónimamente'; + + @override + String get enable_connect => 'Habilitar conexión'; + + @override + String get enable_connect_description => + 'Controla Spotube desde otros dispositivos'; + + @override + String get devices => 'Dispositivos'; + + @override + String get select => 'Seleccionar'; + + @override + String connect_client_alert(Object client) { + return 'Estás siendo controlado por $client'; + } + + @override + String get this_device => 'Este dispositivo'; + + @override + String get remote => 'Remoto'; + + @override + String get stats => 'Estadísticas'; + + @override + String and_n_more(Object count) { + return 'y $count más'; + } + + @override + String get recently_played => 'Recién reproducido'; + + @override + String get browse_more => 'Explorar más'; + + @override + String get no_title => 'Sin título'; + + @override + String get not_playing => 'No reproduciendo'; + + @override + String get epic_failure => '¡Fallo épico!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Se añadieron $tracks_length canciones a la cola'; + } + + @override + String get spotube_has_an_update => 'Spotube tiene una actualización'; + + @override + String get download_now => 'Descargar ahora'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum ha sido lanzado'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version ha sido lanzado'; + } + + @override + String get read_the_latest => 'Lee las últimas '; + + @override + String get release_notes => 'notas de la versión'; + + @override + String get pick_color_scheme => 'Elige esquema de color'; + + @override + String get save => 'Guardar'; + + @override + String get choose_the_device => 'Elige el dispositivo:'; + + @override + String get multiple_device_connected => + 'Hay múltiples dispositivos conectados.\nElige el dispositivo en el que deseas realizar esta acción'; + + @override + String get nothing_found => 'Nada encontrado'; + + @override + String get the_box_is_empty => 'La caja está vacía'; + + @override + String get top_artists => 'Artistas principales'; + + @override + String get top_albums => 'Álbumes principales'; + + @override + String get this_week => 'Esta semana'; + + @override + String get this_month => 'Este mes'; + + @override + String get last_6_months => 'Últimos 6 meses'; + + @override + String get this_year => 'Este año'; + + @override + String get last_2_years => 'Últimos 2 años'; + + @override + String get all_time => 'Todos los tiempos'; + + @override + String powered_by_provider(Object providerName) { + return 'Impulsado por $providerName'; + } + + @override + String get email => 'Correo electrónico'; + + @override + String get profile_followers => 'Seguidores'; + + @override + String get birthday => 'Cumpleaños'; + + @override + String get subscription => 'Suscripción'; + + @override + String get not_born => 'No nacido'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Perfil'; + + @override + String get no_name => 'Sin nombre'; + + @override + String get edit => 'Editar'; + + @override + String get user_profile => 'Perfil de usuario'; + + @override + String count_plays(Object count) { + return '$count reproducciones'; + } + + @override + String get streaming_fees_hypothetical => + 'Tarifas de streaming (hipotéticas)'; + + @override + String get minutes_listened => 'Minutos escuchados'; + + @override + String get streamed_songs => 'Canciones reproducidas'; + + @override + String count_streams(Object count) { + return '$count streams'; + } + + @override + String get owned_by_you => 'En tu posesión'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'Copiado $shareUrl al portapapeles'; + } + + @override + String get hipotetical_calculation => + '*Este cálculo se basa en el pago promedio por reproducción en plataformas de música en línea (de 0,003 a 0,005 USD). Es hipotético y sirve para dar al usuario una idea de cuánto habría pagado a los artistas si hubiera escuchado su canción en distintas plataformas.'; + + @override + String count_mins(Object minutes) { + return '$minutes minutos'; + } + + @override + String get summary_minutes => 'minutos'; + + @override + String get summary_listened_to_music => 'Escuchó música'; + + @override + String get summary_songs => 'canciones'; + + @override + String get summary_streamed_overall => 'Transmitido en general'; + + @override + String get summary_owed_to_artists => 'Debido a los artistas\nEste mes'; + + @override + String get summary_artists => 'artistas'; + + @override + String get summary_music_reached_you => 'La música te alcanzó'; + + @override + String get summary_full_albums => 'álbumes completos'; + + @override + String get summary_got_your_love => 'Obtuvo tu amor'; + + @override + String get summary_playlists => 'listas de reproducción'; + + @override + String get summary_were_on_repeat => 'Estaban en repetición'; + + @override + String total_money(Object money) { + return 'Total $money'; + } + + @override + String get webview_not_found => 'No se encontró el Webview'; + + @override + String get webview_not_found_description => + 'No hay tiempo de ejecución de Webview instalado en su dispositivo.\nSi está instalado, asegúrese de que esté en el environment PATH\n\nDespués de instalar, reinicie la aplicación'; + + @override + String get unsupported_platform => 'Plataforma no soportada'; + + @override + String get cache_music => 'Caché de música'; + + @override + String get open => 'Abrir'; + + @override + String get cache_folder => 'Carpeta de caché'; + + @override + String get export => 'Exportar'; + + @override + String get clear_cache => 'Limpiar caché'; + + @override + String get clear_cache_confirmation => '¿Desea limpiar la caché?'; + + @override + String get export_cache_files => 'Exportar archivos en caché'; + + @override + String found_n_files(Object count) { + return 'Se encontraron $count archivos'; + } + + @override + String get export_cache_confirmation => '¿Desea exportar estos archivos a'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Se exportaron $filesExported de $files archivos'; + } + + @override + String get undo => 'Deshacer'; + + @override + String get download_all => 'Descargar todo'; + + @override + String get add_all_to_playlist => 'Agregar todo a la lista de reproducción'; + + @override + String get add_all_to_queue => 'Agregar todo a la cola'; + + @override + String get play_all_next => 'Reproducir todo a continuación'; + + @override + String get pause => 'Pausa'; + + @override + String get view_all => 'Ver todo'; + + @override + String get no_tracks_added_yet => + 'Parece que aún no has agregado ninguna canción.'; + + @override + String get no_tracks => 'Parece que no hay canciones aquí.'; + + @override + String get no_tracks_listened_yet => + 'Parece que no has escuchado nada todavía.'; + + @override + String get not_following_artists => 'No sigues a ningún artista.'; + + @override + String get no_favorite_albums_yet => + 'Parece que aún no has agregado ningún álbum a tus favoritos.'; + + @override + String get no_logs_found => 'No se encontraron registros'; + + @override + String get youtube_engine => 'Motor de YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine no está instalado'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine no está instalado en tu sistema.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Asegúrate de que esté disponible en la variable PATH o\nestablece la ruta absoluta del ejecutable de $engine a continuación.'; + } + + @override + String get youtube_engine_unix_issue_message => + 'En macOS/Linux/sistemas operativos similares a Unix, establecer la ruta en .zshrc/.bashrc/.bash_profile etc. no funcionará.\nNecesitas establecer la ruta en el archivo de configuración del shell.'; + + @override + String get download => 'Descargar'; + + @override + String get file_not_found => 'Archivo no encontrado'; + + @override + String get custom => 'Personalizado'; + + @override + String get add_custom_url => 'Agregar URL personalizada'; + + @override + String get edit_port => 'Editar puerto'; + + @override + String get port_helper_msg => + 'El valor predeterminado es -1, lo que indica un número aleatorio. Si tienes un firewall configurado, se recomienda establecer esto.'; + + @override + String connect_request(Object client) { + return '¿Permitir que $client se conecte?'; + } + + @override + String get connection_request_denied => + 'Conexión denegada. El usuario denegó el acceso.'; + + @override + String get an_error_occurred => 'Ocurrió un error'; + + @override + String get copy_to_clipboard => 'Copiar al portapapeles'; + + @override + String get view_logs => 'Ver registros'; + + @override + String get retry => 'Reintentar'; + + @override + String get no_default_metadata_provider_selected => + 'No has configurado un proveedor de metadatos predeterminado'; + + @override + String get manage_metadata_providers => 'Gestionar proveedores de metadatos'; + + @override + String get open_link_in_browser => '¿Abrir enlace en el navegador?'; + + @override + String get do_you_want_to_open_the_following_link => + '¿Quieres abrir el siguiente enlace?'; + + @override + String get unsafe_url_warning => + 'Abrir enlaces de fuentes no confiables puede ser inseguro. ¡Ten cuidado!\nTambién puedes copiar el enlace al portapapeles.'; + + @override + String get copy_link => 'Copiar enlace'; + + @override + String get building_your_timeline => + 'Construyendo tu línea de tiempo según tus escuchas…'; + + @override + String get official => 'Oficial'; + + @override + String author_name(Object author) { + return 'Autor: $author'; + } + + @override + String get third_party => 'Terceros'; + + @override + String get plugin_requires_authentication => + 'El complemento requiere autenticación'; + + @override + String get update_available => 'Actualización disponible'; + + @override + String get supports_scrobbling => 'Admite scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Este complemento scrobblea tu música para generar tu historial de reproducción.'; + + @override + String get default_metadata_source => 'Fuente de metadatos predeterminada'; + + @override + String get set_default_metadata_source => + 'Establecer fuente de metadatos predeterminada'; + + @override + String get default_audio_source => 'Fuente de audio predeterminada'; + + @override + String get set_default_audio_source => + 'Establecer fuente de audio predeterminada'; + + @override + String get set_default => 'Establecer como predeterminado'; + + @override + String get support => 'Soporte'; + + @override + String get support_plugin_development => + 'Apoyar el desarrollo del complemento'; + + @override + String can_access_name_api(Object name) { + return '- Puede acceder a la API de **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + '¿Deseas instalar este complemento?'; + + @override + String get third_party_plugin_warning => + 'Este complemento proviene de un repositorio de terceros. Asegúrate de confiar en la fuente antes de instalarlo.'; + + @override + String get author => 'Autor'; + + @override + String get this_plugin_can_do_following => + 'Este complemento puede hacer lo siguiente'; + + @override + String get install => 'Instalar'; + + @override + String get install_a_metadata_provider => + 'Instalar un proveedor de metadatos'; + + @override + String get no_tracks_playing => + 'No hay ninguna pista reproduciéndose actualmente'; + + @override + String get synced_lyrics_not_available => + 'Las letras sincronizadas no están disponibles para esta canción. Por favor, utiliza'; + + @override + String get plain_lyrics => 'Letras sin formato'; + + @override + String get tab_instead => 'en su lugar, usa la tecla Tab.'; + + @override + String get disclaimer => 'Descargo de responsabilidad'; + + @override + String get third_party_plugin_dmca_notice => + 'El equipo de Spotube no asume ninguna responsabilidad (incluida la legal) por complementos de \"terceros\". Úsalos bajo tu propio riesgo. Para errores o problemas, repórtalos en el repositorio del complemento.\n\nSi algún complemento de “terceros” infringe los ToS/DMCA de algún servicio o entidad legal, por favor, solicita al autor del complemento o a la plataforma de alojamiento (p. ej., GitHub/Codeberg) que tome medidas. Los complementos etiquetados como “de terceros” son mantenidos públicamente por la comunidad; no los gestionamos y no podemos intervenir.\n\n'; + + @override + String get input_does_not_match_format => + 'La entrada no coincide con el formato requerido'; + + @override + String get plugins => 'Plugins'; + + @override + String get paste_plugin_download_url => + 'Pega la URL de descarga, el repositorio de GitHub/Codeberg o el enlace directo al archivo .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Descargar e instalar el complemento desde una URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Error al añadir el complemento: $error'; + } + + @override + String get upload_plugin_from_file => 'Subir complemento desde archivo'; + + @override + String get installed => 'Instalado'; + + @override + String get available_plugins => 'Complementos disponibles'; + + @override + String get configure_plugins => + 'Configura tus propios plugins de proveedor de metadatos y fuente de audio'; + + @override + String get audio_scrobblers => 'Scrobblers de audio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Fuente: '; + + @override + String get uncompressed => 'Sin comprimir'; + + @override + String get dab_music_source_description => + 'Para audiófilos. Proporciona transmisiones de audio de alta calidad/sin pérdida. Coincidencia precisa de pistas basada en ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_eu.dart b/lib/l10n/generated/app_localizations_eu.dart new file mode 100644 index 00000000..5f80397e --- /dev/null +++ b/lib/l10n/generated/app_localizations_eu.dart @@ -0,0 +1,1577 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Basque (`eu`). +class AppLocalizationsEu extends AppLocalizations { + AppLocalizationsEu([String locale = 'eu']) : super(locale); + + @override + String get guest => 'Gonbidatua'; + + @override + String get browse => 'Arakatu'; + + @override + String get search => 'Bilatu'; + + @override + String get library => 'Liburutegia'; + + @override + String get lyrics => 'Hitzak'; + + @override + String get settings => 'Ezarpenak'; + + @override + String get genre_categories_filter => 'Kategoria edo generoak filtratu...'; + + @override + String get genre => 'Generoa'; + + @override + String get personalized => 'Pertsonalizatua'; + + @override + String get featured => 'Nabarmenduak'; + + @override + String get new_releases => 'Argitaratze berriak'; + + @override + String get songs => 'Abestiak'; + + @override + String playing_track(Object track) { + return '$track erreproduzitzen'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Uneko zerrenda ezabatuko da. $track_length abesti ezabatuko dira.\nJarraitu nahi duzu?'; + } + + @override + String get load_more => 'Gehiago kargatu'; + + @override + String get playlists => 'Zerrendak'; + + @override + String get artists => 'Artistak'; + + @override + String get albums => 'Albumak'; + + @override + String get tracks => 'Kantak'; + + @override + String get downloads => 'Deskargak'; + + @override + String get filter_playlists => 'Zure zerrendak filtratu...'; + + @override + String get liked_tracks => 'Gustuko Kantak'; + + @override + String get liked_tracks_description => 'Zure gustuko kanta guztiak'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Sortu zerrenda bat'; + + @override + String get update_playlist => 'Eguneratu zerrenda'; + + @override + String get create => 'Sortu'; + + @override + String get cancel => 'Ezeztatu'; + + @override + String get update => 'Eguneratu'; + + @override + String get playlist_name => 'Zerrenda Izena'; + + @override + String get name_of_playlist => 'Zerrendaren izena'; + + @override + String get description => 'Deskribapena'; + + @override + String get public => 'Publikoa'; + + @override + String get collaborative => 'Kolaboratiboa'; + + @override + String get search_local_tracks => 'Bilatu kanta lokalak...'; + + @override + String get play => 'Erreproduzitu'; + + @override + String get delete => 'Ezabatu'; + + @override + String get none => 'Batere ez'; + + @override + String get sort_a_z => 'Ordenatu A-Z'; + + @override + String get sort_z_a => 'Ordenatu Z-A'; + + @override + String get sort_artist => 'Ordenatu Artistaren arabera'; + + @override + String get sort_album => 'Ordenatu Albumaren arabera'; + + @override + String get sort_duration => 'Ordenar Iraupenaren arabera'; + + @override + String get sort_tracks => 'Ordenatu Kantak'; + + @override + String currently_downloading(Object tracks_length) { + return 'Oraintxe ($tracks_length) deskargatzen'; + } + + @override + String get cancel_all => 'Ezeztatu dena'; + + @override + String get filter_artist => 'Filtratu artistak...'; + + @override + String followers(Object followers) { + return '$followers Jarraitzaile'; + } + + @override + String get add_artist_to_blacklist => 'Gehitu artista zerrenda beltzera'; + + @override + String get top_tracks => 'Top Kantak'; + + @override + String get fans_also_like => 'Fan-ek hau ere gustuko dute'; + + @override + String get loading => 'Kargatzen...'; + + @override + String get artist => 'Artista'; + + @override + String get blacklisted => 'Zerrenda beltzean'; + + @override + String get following => 'Jarraitzen'; + + @override + String get follow => 'Jarraitu'; + + @override + String get artist_url_copied => 'Artistaren URL-a arbelera kopiatua'; + + @override + String added_to_queue(Object tracks) { + return '$tracks kanta zerrendara gehituak'; + } + + @override + String get filter_albums => 'Albumak filtratu...'; + + @override + String get synced => 'Sinkronizatuta'; + + @override + String get plain => 'Arrunta'; + + @override + String get shuffle => 'Ausaz'; + + @override + String get search_tracks => 'Bilatu kantak...'; + + @override + String get released => 'Argitaratua'; + + @override + String error(Object error) { + return 'Errorea: $error'; + } + + @override + String get title => 'Izenburua'; + + @override + String get time => 'Iraupena'; + + @override + String get more_actions => 'Ekintza gehiago'; + + @override + String download_count(Object count) { + return '($count) deskarga'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Gehitu ($count) zerrendara'; + } + + @override + String add_count_to_queue(Object count) { + return 'Gehitu ($count) ilarara'; + } + + @override + String play_count_next(Object count) { + return 'Erreproduzitu hurrengo ($count)-ak'; + } + + @override + String get album => 'Albuma'; + + @override + String copied_to_clipboard(Object data) { + return '$data arbelean kopiatua'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Gehitu $track hurrengo erreprodukzio-zerrendetara'; + } + + @override + String get add => 'Gehitu'; + + @override + String added_track_to_queue(Object track) { + return '$track zerrendan gehitua'; + } + + @override + String get add_to_queue => 'Gehitu zerrendan'; + + @override + String track_will_play_next(Object track) { + return '$track erreproduzituko da ondoren'; + } + + @override + String get play_next => 'Hurrengo erreprodukzioa'; + + @override + String removed_track_from_queue(Object track) { + return '$track zerrendatik ezabatua'; + } + + @override + String get remove_from_queue => 'Ezabatu ilaratik'; + + @override + String get remove_from_favorites => 'Ezabatu gogokoetatik'; + + @override + String get save_as_favorite => 'Gorde gogokoetan'; + + @override + String get add_to_playlist => 'Gehitu zerrendara'; + + @override + String get remove_from_playlist => 'Ezabatu zerrendatik'; + + @override + String get add_to_blacklist => 'Gehitu zerrenda beltzera'; + + @override + String get remove_from_blacklist => 'Ezabatu zerrenda beltzetik'; + + @override + String get share => 'Elkarbanatu'; + + @override + String get mini_player => 'Mini Erreproduzitzailea'; + + @override + String get slide_to_seek => 'Arrastatu aurrerantz edo atzearantz bilatzeko'; + + @override + String get shuffle_playlist => 'Erreproduzitu zerrenda ausazko ordenean'; + + @override + String get unshuffle_playlist => 'Desgaitu ausazko erreprodukzioa'; + + @override + String get previous_track => 'Aurreko pista'; + + @override + String get next_track => 'Hurrengo pista'; + + @override + String get pause_playback => 'Pausatu erreprodukzioa'; + + @override + String get resume_playback => 'Berrabiarazi erreprodukzioa'; + + @override + String get loop_track => 'Kanta begiztan'; + + @override + String get no_loop => 'Ez dago loop-ik'; + + @override + String get repeat_playlist => 'Errepikatu lista'; + + @override + String get queue => 'Ilara'; + + @override + String get alternative_track_sources => 'Kanten iturri alternatiboak'; + + @override + String get download_track => 'Deskargatu kanta'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks kanta zerrendan'; + } + + @override + String get clear_all => 'Garbitu dena'; + + @override + String get show_hide_ui_on_hover => + 'Erakutsi/Ezkutatu interfazea kurtsorea pasatzean'; + + @override + String get always_on_top => 'Beti ikusgai'; + + @override + String get exit_mini_player => 'Irten mini erreproduzitzailetik'; + + @override + String get download_location => 'Deskargen kokapena'; + + @override + String get local_library => 'Liburutegi lokala'; + + @override + String get add_library_location => 'Gehitu liburutegira'; + + @override + String get remove_library_location => 'Kendu liburutegitik'; + + @override + String get account => 'Kontua'; + + @override + String get logout => 'Itxi saioa'; + + @override + String get logout_of_this_account => 'Itxi kontu honen saioa'; + + @override + String get language_region => 'Hizkuntza eta Herrialdea'; + + @override + String get language => 'Hizkuntza'; + + @override + String get system_default => 'Sisteman lehenetsia'; + + @override + String get market_place_region => 'Dendaren herrialdea'; + + @override + String get recommendation_country => 'Gomendio herrialdea'; + + @override + String get appearance => 'Itxura'; + + @override + String get layout_mode => 'Diseinua'; + + @override + String get override_layout_settings => + 'Responsive diseinuaren ezarpenak ezeztatu'; + + @override + String get adaptive => 'Moldagarria'; + + @override + String get compact => 'Trinkoa'; + + @override + String get extended => 'Hedatua'; + + @override + String get theme => 'Gaia'; + + @override + String get dark => 'Iluna'; + + @override + String get light => 'Argia'; + + @override + String get system => 'Sistema'; + + @override + String get accent_color => 'Azentu kolorea'; + + @override + String get sync_album_color => 'Sinkronizatu albumaren kolorea'; + + @override + String get sync_album_color_description => + 'Albumaren artearen kolore nagusia erabili azentu kolore bezala'; + + @override + String get playback => 'Erreprodukzioa'; + + @override + String get audio_quality => 'Audioaren kalitatea'; + + @override + String get high => 'Altua'; + + @override + String get low => 'Baxua'; + + @override + String get pre_download_play => 'Aurre-deskargatu eta erreproduzitu'; + + @override + String get pre_download_play_description => + 'Streaming egin beharrean, byte-ak deskargatu eta erreproduzitu (banda-zabalera handia duten erabiltzaileentzat gomendagarria)'; + + @override + String get skip_non_music => + 'Musika ez diren segmentuak baztertu (SponsorBlock)'; + + @override + String get blacklist_description => 'Zerrenda beltzeko abesti eta artistak'; + + @override + String get wait_for_download_to_finish => + 'Mesedez, itxaron uneko deskarga bukatu arte'; + + @override + String get desktop => 'Mahaigaina'; + + @override + String get close_behavior => 'Ixterako Portaera'; + + @override + String get close => 'Itxi'; + + @override + String get minimize_to_tray => 'Sistemako erretilura minimizatu'; + + @override + String get show_tray_icon => 'Erakutsi ikonoa sistemaren erretiluan'; + + @override + String get about => 'Honi buruz'; + + @override + String get u_love_spotube => 'Badakigu Spotube maite duzula'; + + @override + String get check_for_updates => 'Bilatu eguneraketak'; + + @override + String get about_spotube => 'Spotube-ri buruz'; + + @override + String get blacklist => 'Zerrenda beltza'; + + @override + String get please_sponsor => 'Mesedez, babestu/diruz lagundu'; + + @override + String get spotube_description => + 'Spotube, arina, plataforma-anitza eta doakoa den Spotify-ren bezeroa'; + + @override + String get version => 'Bertsioa'; + + @override + String get build_number => 'Konpilazio zenbakia'; + + @override + String get founder => 'Sortzailea'; + + @override + String get repository => 'Errepositorioa'; + + @override + String get bug_issues => 'Erroreak eta arazoak'; + + @override + String get made_with => 'Bangladesh🇧🇩-en ❤️-z egina'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Lizentzia'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Ez arduratu, zure kredentzialak ez ditugu bilduko edo inorekin elkarbanatuko'; + + @override + String get know_how_to_login => 'Ez dakizu nola egin?'; + + @override + String get follow_step_by_step_guide => 'Jarraitu pausoz-pausoko gida'; + + @override + String cookie_name_cookie(Object name) { + return '$name cookiea'; + } + + @override + String get fill_in_all_fields => 'Mesedez, osatu eremu guztiak'; + + @override + String get submit => 'Bidali'; + + @override + String get exit => 'Irten'; + + @override + String get previous => 'Aurrekoa'; + + @override + String get next => 'Hurrengoa'; + + @override + String get done => 'Eginda'; + + @override + String get step_1 => '1. pausua'; + + @override + String get first_go_to => 'Hasteko, joan hona'; + + @override + String get something_went_wrong => 'Zerbaitek huts egin du'; + + @override + String get piped_instance => 'Piped zerbitzariaren instantzia'; + + @override + String get piped_description => + 'Kanten koizidentzietan erabiltzeko Piped zerbitzariaren instantzia'; + + @override + String get piped_warning => + 'Batzuk agian ez dute ongi funtzionatuko, zure ardurapean erabili'; + + @override + String get invidious_instance => 'Invidious zerbitzari instantzia'; + + @override + String get invidious_description => + 'Invidious zerbitzari instantzia, pistak bat egiteko'; + + @override + String get invidious_warning => + 'Instantzia batzuek ez dute ondo funtzionatuko. Zure erantzukizunpean erabili'; + + @override + String get generate => 'Sortu'; + + @override + String track_exists(Object track) { + return '$track kanta dagoeneko badago'; + } + + @override + String get replace_downloaded_tracks => + 'Ordezkatu deskargatutako kanta guztiak'; + + @override + String get skip_download_tracks => + 'Deskargatutako kanta guztien deskarga baztertu'; + + @override + String get do_you_want_to_replace => 'Dagoen kanta ordezkatu nahi duzu??'; + + @override + String get replace => 'Ordezkatu'; + + @override + String get skip => 'Baztertu'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Aukertu $count $type'; + } + + @override + String get select_genres => 'Aukeratu Generoak'; + + @override + String get add_genres => 'Gehitu Generoak'; + + @override + String get country => 'Herrialdea'; + + @override + String get number_of_tracks_generate => 'Sortzeko kanta kopurua'; + + @override + String get acousticness => 'Akustikotasuna'; + + @override + String get danceability => 'Dantzagarritasuna'; + + @override + String get energy => 'Energia'; + + @override + String get instrumentalness => 'Instrumentaltasuna'; + + @override + String get liveness => 'Zuzenean'; + + @override + String get loudness => 'Ozentasuna'; + + @override + String get speechiness => 'Hitzaldia'; + + @override + String get valence => 'Balentzia'; + + @override + String get popularity => 'Populartasuna'; + + @override + String get key => 'Tonua'; + + @override + String get duration => 'Iraupena (s)'; + + @override + String get tempo => 'Tenpoa (BPM)'; + + @override + String get mode => 'Modua'; + + @override + String get time_signature => 'Konpasa'; + + @override + String get short => 'Motza'; + + @override + String get medium => 'Ertaina'; + + @override + String get long => 'Luzea'; + + @override + String get min => 'Min.'; + + @override + String get max => 'Max.'; + + @override + String get target => 'Helburua'; + + @override + String get moderate => 'Moderatua'; + + @override + String get deselect_all => 'Desaukeratu dena'; + + @override + String get select_all => 'Aukeratu dena'; + + @override + String get are_you_sure => 'Ziur zaude?'; + + @override + String get generating_playlist => + 'Zure pertsonalizatutako zerrenda sortzen...'; + + @override + String selected_count_tracks(Object count) { + return '$count kanta aukeratuta'; + } + + @override + String get download_warning => + 'Abesti guztiak aldi berean deskargatuz gero, argi dago musika pirateatzen ari zarela eta musikaren gizarte sortzaileari kalte egiten diozula. Honen jakitun izan eta artisten lan gogorra errespetatu eta babestea espero dut'; + + @override + String get download_ip_ban_warning => + 'Bidenabar, baliteke zure IPa YouTuben blokeatzea deskarga eskera gehiegi egiten badituzu. IPa blokeatzeak esan nahi du ezin izango duzula YouTube erabili (nahiz eta saioa hasia izan) gutxienez 2-3 hilabetez IP helbide horretatik. Eta Spotube ez da erantzule izango hori gertatzen bazaizu'; + + @override + String get by_clicking_accept_terms => + '\'Onartu\' klikatzean, ondorengo baldintzak onartzen dituzu:'; + + @override + String get download_agreement_1 => + 'Badakit musika pirateatzen ari naizela. Gaiztoa naiz'; + + @override + String get download_agreement_2 => + 'Ahal dudanean lagunduko diot artistari baina oraingoz ez dut bere artea erosteko dirurik'; + + @override + String get download_agreement_3 => + 'Erabat jakitun naiz YouTubek nire IPa blokea dezakeela eta ez diot Spotube-ri edo bere jabe/laguntzaileei erantzukizunik eskatuko nire oraingo jokaerak ekar ditzakeen arazoengatik'; + + @override + String get decline => 'Baztertu'; + + @override + String get accept => 'Onartu'; + + @override + String get details => 'Xehetasunak'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanala'; + + @override + String get likes => 'Gustukoak'; + + @override + String get dislikes => 'Ez gustukoak'; + + @override + String get views => 'Ikuspenak'; + + @override + String get streamUrl => 'Streaming-aren URLa'; + + @override + String get stop => 'Gelditu'; + + @override + String get sort_newest => 'Ordenatu gehitu berrienetik'; + + @override + String get sort_oldest => 'Ordenatu gehitu zaharrenetik'; + + @override + String get sleep_timer => 'Itzaltzeko tenporizadorea'; + + @override + String mins(Object minutes) { + return '$minutes minutu'; + } + + @override + String hours(Object hours) { + return '$hours ordu'; + } + + @override + String hour(Object hours) { + return '$hours ordu'; + } + + @override + String get custom_hours => 'Ordu pertsonalizatuak'; + + @override + String get logs => 'Log-ak'; + + @override + String get developers => 'Garatzaileak'; + + @override + String get not_logged_in => 'Ez duzu saioa hasi'; + + @override + String get search_mode => 'Bilaketa modua'; + + @override + String get audio_source => 'Audio Iturria'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => 'Errorea zifratzean'; + + @override + String get encryption_failed_warning => + 'Spotube-ek zifratzea darabil datuak modu seguruan biltegiratzeko. Baina huts egin du. Hori dela eta, biltegiratzea ez da segurua izango\nLinux erabiltzen ari bazara, ziurtatu edozein sekretu-zerbitzu (gnome-keyring, kde-wallet, keepassxc etab.) instalatuta duzula'; + + @override + String get querying_info => 'Informazioa egiaztatzen...'; + + @override + String get piped_api_down => 'Piped-en APIa ez dago eskuragarri'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Piped-en $pipedInstance instantzia ez dago martxan une honetan\n\nAldatu instantzia edo aldatu \'API mota\' YouTuberen API ofizialera\n\nZiurtatu aplikazioa berrabiarazten duzula aldaketa eta gero'; + } + + @override + String get you_are_offline => 'Une honetan konexiorik gabe zaude'; + + @override + String get connection_restored => 'Internet konexioa berrezarri egin da'; + + @override + String get use_system_title_bar => 'Erabili sistemako izenburu barra'; + + @override + String get crunching_results => 'Emaitzak prozesatzen...'; + + @override + String get search_to_get_results => 'Bilatu emaitzak lortzeko'; + + @override + String get use_amoled_mode => 'Erabili AMOLED modua'; + + @override + String get pitch_dark_theme => 'Dart-en gai iluna'; + + @override + String get normalize_audio => 'Normalizatu audioa'; + + @override + String get change_cover => 'Aldatu azala'; + + @override + String get add_cover => 'Gehitu azala'; + + @override + String get restore_defaults => 'Berrezarri berezko balioak'; + + @override + String get download_music_format => 'Musika deskargatzeko formatua'; + + @override + String get streaming_music_format => 'Musika streaming bidezko formatua'; + + @override + String get download_music_quality => 'Musika deskargaren kalitatea'; + + @override + String get streaming_music_quality => 'Streaming bidezko musika kalitatea'; + + @override + String get login_with_lastfm => 'Hasi saioa Last.fm-n'; + + @override + String get connect => 'Konektatu'; + + @override + String get disconnect_lastfm => 'Deskonektatu Last.fm-tik'; + + @override + String get disconnect => 'Deskonektatu'; + + @override + String get username => 'Erabiltzaile izena'; + + @override + String get password => 'Pasahitza'; + + @override + String get login => 'Hasi saioa'; + + @override + String get login_with_your_lastfm => 'Hasi saioa Last.fm-ko zure kontuarekin'; + + @override + String get scrobble_to_lastfm => 'Scrobble Last.fm-ra'; + + @override + String get go_to_album => 'Albumera joan'; + + @override + String get discord_rich_presence => 'Discord-en presentzia aberatsa'; + + @override + String get browse_all => 'Esploratu dena'; + + @override + String get genres => 'Generoak'; + + @override + String get explore_genres => 'Esploratu generoak'; + + @override + String get friends => 'Lagunak'; + + @override + String get no_lyrics_available => + 'Sentitzen dugu, ezin dira kanta honen hitzak aurkitu'; + + @override + String get start_a_radio => 'Hasi Irrati bat'; + + @override + String get how_to_start_radio => 'Nola hasi nahi duzu irratia?'; + + @override + String get replace_queue_question => + 'Uneko zerrenda ordezkatu nahi duzu edo bertan gehitu?'; + + @override + String get endless_playback => 'Amaigabeko erreprodukzioa'; + + @override + String get delete_playlist => 'Ezabatu zerrenda'; + + @override + String get delete_playlist_confirmation => + 'Ziur zaude zerrenda ezabatu nahi duzula?'; + + @override + String get local_tracks => 'Kanta lokalak'; + + @override + String get local_tab => 'Lokalean'; + + @override + String get song_link => 'Kantaren lotura'; + + @override + String get skip_this_nonsense => 'Utzi txorakeria hau'; + + @override + String get freedom_of_music => '“Musika Askatasuna”'; + + @override + String get freedom_of_music_palm => '“Musika Askatasuna zure eskuetan”'; + + @override + String get get_started => 'Has gaitezen'; + + @override + String get youtube_source_description => 'Gomendatua eta hobekien dabilena.'; + + @override + String get piped_source_description => + 'Aske zara? YouTube bezala, baino askeago.'; + + @override + String get jiosaavn_source_description => + 'Asia hegoaldeko herrialdeetarako hoberena.'; + + @override + String get invidious_source_description => + 'Piped-en antzekoa, baina eskuragarritasun handiagoarekin'; + + @override + String highest_quality(Object quality) { + return 'Kalitate Onena: $quality'; + } + + @override + String get select_audio_source => 'Aukeratu Audio Iturria'; + + @override + String get endless_playback_description => + 'Gehitu automatikoki kanta berriak\n ilararen bukaeran'; + + @override + String get choose_your_region => 'Aukeratu zure herrialdea'; + + @override + String get choose_your_region_description => + 'Honekin Spotube-k zure kokalerakuari dagokion edukia\neskeiniko dizu.'; + + @override + String get choose_your_language => 'Aukeratu zure hizkuntza'; + + @override + String get help_project_grow => 'Lagundu proiektu honi hazten'; + + @override + String get help_project_grow_description => + 'Spotube kode irekiko proiektu bat da. Proiektu hau hazten lagundu dezakezu, erroreak jakinaraziz edo ezaugarri berriak proposatuz.'; + + @override + String get contribute_on_github => 'GitHub-en lagundu'; + + @override + String get donate_on_open_collective => 'Open Collective-en diruz lagundu'; + + @override + String get browse_anonymously => 'Nabigatu Anonimoki'; + + @override + String get enable_connect => 'Gaitu konexioa'; + + @override + String get enable_connect_description => + 'Kontrolatu Spotube beste gailu batzuetatik'; + + @override + String get devices => 'Gailuak'; + + @override + String get select => 'Aukeratu'; + + @override + String connect_client_alert(Object client) { + return '$client gailuak kontrolatzen zaitu'; + } + + @override + String get this_device => 'Gailu hau'; + + @override + String get remote => 'Urrunekoa'; + + @override + String get stats => 'Estatistikak'; + + @override + String and_n_more(Object count) { + return 'eta $count gehiago'; + } + + @override + String get recently_played => 'Berriki entzunak'; + + @override + String get browse_more => 'Gehiago Bilatu'; + + @override + String get no_title => 'Titulurik ez'; + + @override + String get not_playing => 'Erreprodukziorik ez'; + + @override + String get epic_failure => 'Sekulako errorea!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length kanta gehitu dira zerrendara'; + } + + @override + String get spotube_has_an_update => 'Spotube-ren eguneraketa bat dago'; + + @override + String get download_now => 'Orain deskargatu'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube $nightlyBuildNum Nightly-a argitaratu da'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version argitaratu da'; + } + + @override + String get read_the_latest => 'Irakurri azken '; + + @override + String get release_notes => 'argitatratze oharrak'; + + @override + String get pick_color_scheme => 'Aukeratu kolore eskema'; + + @override + String get save => 'Gorde'; + + @override + String get choose_the_device => 'Aukeratu gailua:'; + + @override + String get multiple_device_connected => + 'Hainbat gailu daude konektatuta.\nAukeratu zein gailutan aplikatu nahi duzun ekintza hau'; + + @override + String get nothing_found => 'Ezer ez da aurkitu'; + + @override + String get the_box_is_empty => 'Kaxa hutsik dago'; + + @override + String get top_artists => 'Top Artistak'; + + @override + String get top_albums => 'Top Albumak'; + + @override + String get this_week => 'Aste honetan'; + + @override + String get this_month => 'Hilabete honetan'; + + @override + String get last_6_months => 'Azken 6 hilabeteetan'; + + @override + String get this_year => 'Aurten'; + + @override + String get last_2_years => 'Azken 2 urtetan'; + + @override + String get all_time => 'Betidanik'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName-ren eskutik'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Jarraitzaileak'; + + @override + String get birthday => 'Jaiotze-data'; + + @override + String get subscription => 'Harpidetzak'; + + @override + String get not_born => 'Jaio gabe'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profila'; + + @override + String get no_name => 'Izenik Ez'; + + @override + String get edit => 'Editatu'; + + @override + String get user_profile => 'Erabiltzaile Profila'; + + @override + String count_plays(Object count) { + return '$count erreprodukzio'; + } + + @override + String get streaming_fees_hypothetical => + 'Streaming ordainketa (hipotetikoa)'; + + @override + String get minutes_listened => 'Entzundako minutuak'; + + @override + String get streamed_songs => 'Streaming-ez entzundako kantak'; + + @override + String count_streams(Object count) { + return '$count stream'; + } + + @override + String get owned_by_you => 'Zure jabetzakoa'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl arbelera kopiatua'; + } + + @override + String get hipotetical_calculation => + '*Kalkulu hau online musika-streaming plataformetako batez besteko irteerako ordainari (0,003–0,005 USD) oinarrituta dago. Hipotetikoa da eta erabiltzaileari ideia bat ematen laguntzen dio artista nork zenbat kobratu zuen jakiteko, bere abestia plataform desberdinetan entzungo balu.'; + + @override + String count_mins(Object minutes) { + return '$minutes minutu'; + } + + @override + String get summary_minutes => 'minutu'; + + @override + String get summary_listened_to_music => 'Musika entzuten'; + + @override + String get summary_songs => 'kanta'; + + @override + String get summary_streamed_overall => 'Streaming abesti oro har'; + + @override + String get summary_owed_to_artists => 'Hilabete honetan\nartistei zor zaiena'; + + @override + String get summary_artists => 'artisten'; + + @override + String get summary_music_reached_you => 'Musika ailegatu zaizu'; + + @override + String get summary_full_albums => 'album osok'; + + @override + String get summary_got_your_love => 'Jaso dute zure maitasuna'; + + @override + String get summary_playlists => 'zerrenda'; + + @override + String get summary_were_on_repeat => 'Dituzu errepikatze moduan'; + + @override + String total_money(Object money) { + return 'Guztira $money'; + } + + @override + String get webview_not_found => 'Ez da Webview aurkitu'; + + @override + String get webview_not_found_description => + 'Ez dago Webview abiarazte denbora-instalaziorik zure gailuan.\nInstalatuta badago, ziurtatu environment PATH-an dagoela\n\nInstalatu ondoren, berrabiarazi aplikazioa'; + + @override + String get unsupported_platform => 'Plataforma ez onartua'; + + @override + String get cache_music => 'Musika cachean'; + + @override + String get open => 'Ireki'; + + @override + String get cache_folder => 'Cache karpeta'; + + @override + String get export => 'Esportatu'; + + @override + String get clear_cache => 'Garbitu cachea'; + + @override + String get clear_cache_confirmation => 'Cachea garbitu nahi al duzu?'; + + @override + String get export_cache_files => 'Esportatu cache fitxategiak'; + + @override + String found_n_files(Object count) { + return '$count fitxategi aurkitu dira'; + } + + @override + String get export_cache_confirmation => + 'Fitxategi hauek esportatu nahi al dituzu'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported fitxategi esportatu dira $files -tik'; + } + + @override + String get undo => 'Desegondu'; + + @override + String get download_all => 'Guztia deskargatu'; + + @override + String get add_all_to_playlist => 'Guztia playlist-era gehitu'; + + @override + String get add_all_to_queue => 'Guztia zerrendara gehitu'; + + @override + String get play_all_next => 'Guztia hurrengoan jolastu'; + + @override + String get pause => 'Pausatu'; + + @override + String get view_all => 'Ikusi guztia'; + + @override + String get no_tracks_added_yet => + 'Dirudienez, oraindik ez duzu abestirik gehitu.'; + + @override + String get no_tracks => 'Ez dirudi hemen abestirik dagoenik.'; + + @override + String get no_tracks_listened_yet => + 'Dirudienez, oraindik ez duzu ezer entzun.'; + + @override + String get not_following_artists => 'Ez zaude artisten atzetik.'; + + @override + String get no_favorite_albums_yet => + 'Dirudienez, oraindik ez duzu albumik gehitu zure gogokoen artean.'; + + @override + String get no_logs_found => 'Ez dira log-ak aurkitu'; + + @override + String get youtube_engine => 'YouTube Motorra'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine ez dago instalatuta'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine ez dago zure sisteman instalatuta.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Ziurtatu PATH aldagaiaren barruan dagoela edo\nezarri $engine exekutagarriaren helbide absolutua behean.'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/Unix bezalako sistemetan, .zshrc/.bashrc/.bash_profile bezalako fitxategietan bidearen ezarpenak ez dira funtzionatuko.\nBidearen ezarpena shell konfigurazio fitxategian egin behar duzu.'; + + @override + String get download => 'Deskargatu'; + + @override + String get file_not_found => 'Fitxategia ez da aurkitu'; + + @override + String get custom => 'Pertsonalizatua'; + + @override + String get add_custom_url => 'Gehitu URL pertsonalizatua'; + + @override + String get edit_port => 'Editatu portua'; + + @override + String get port_helper_msg => + 'Lehenetsitako balioa -1 da, zenbaki aleatorioa adierazten duena. Su firewall konfiguratu baduzu, gomendatzen da hau ezartzea.'; + + @override + String connect_request(Object client) { + return '$client konektatzea baimendu?'; + } + + @override + String get connection_request_denied => + 'Konektatzea ukatu da. Erabiltzaileak sarbidea ukatu du.'; + + @override + String get an_error_occurred => 'Errore bat gertatu da'; + + @override + String get copy_to_clipboard => 'Hiztegiraino kopiatzea'; + + @override + String get view_logs => 'Erregistroak ikusi'; + + @override + String get retry => 'Berriro saiatu'; + + @override + String get no_default_metadata_provider_selected => + 'Ezarri ez duzu metadaten hornitzaile lehenetsirik'; + + @override + String get manage_metadata_providers => 'Metadaten hornitzaileak kudeatu'; + + @override + String get open_link_in_browser => 'Esteka nabigatzailean irekiko duzu?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Hurrengo esteka irekiko duzu?'; + + @override + String get unsafe_url_warning => + 'Iturri seguru gabeko estekak irekiz gero, ez da seguru suerta daiteke. Arduratu zaitez!\nEsteka ere hiztegirainokoan kopiatu dezakezu.'; + + @override + String get copy_link => 'Esteka kopiatu'; + + @override + String get building_your_timeline => + 'Zure entzuteen arabera zure kronologia eraikitzen…'; + + @override + String get official => 'Ofiziala'; + + @override + String author_name(Object author) { + return 'Egilea: $author'; + } + + @override + String get third_party => 'Hirugarrena'; + + @override + String get plugin_requires_authentication => + 'Pluginak autentifikazioa eskatzen du'; + + @override + String get update_available => 'Eguneratze bat dago eskuragarri'; + + @override + String get supports_scrobbling => 'Scrobbling-a onartzen du'; + + @override + String get plugin_scrobbling_info => + 'Plugin honek zure musika scrobbled egiten du zure entzuteen historia sortzeko.'; + + @override + String get default_metadata_source => 'Metadatu-iturburu lehenetsia'; + + @override + String get set_default_metadata_source => + 'Ezarri metadatu-iturburu lehenetsia'; + + @override + String get default_audio_source => 'Audio-iturburu lehenetsia'; + + @override + String get set_default_audio_source => 'Ezarri audio-iturburu lehenetsia'; + + @override + String get set_default => 'Lehenetsi gisa ezarri'; + + @override + String get support => 'Laguntza'; + + @override + String get support_plugin_development => 'Pluginaren garapena lagundu'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API-ra sar daiteke'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Plugin hau instalatu nahiko zenuke?'; + + @override + String get third_party_plugin_warning => + 'Plugin hau hirugarrenen biltegi batetik dator. Instalatu aurretik iturriari konfiantza behar diozu.'; + + @override + String get author => 'Egilea'; + + @override + String get this_plugin_can_do_following => + 'Plugin honek honako hau egin dezake:'; + + @override + String get install => 'Instalatu'; + + @override + String get install_a_metadata_provider => + 'Metadaten hornitzaile bat instalatu'; + + @override + String get no_tracks_playing => + 'Une honetan ez dago abestirik erreproduzitzen'; + + @override + String get synced_lyrics_not_available => + 'Abestiarentzako letra sinkronizatua ez dago erabilgarri. Mesedez, erabili'; + + @override + String get plain_lyrics => 'Letra arrunta'; + + @override + String get tab_instead => 'horren ordez, Tab teklatxaza erabili.'; + + @override + String get disclaimer => 'Aldez aurreko oharra'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube taldea ezin da arduratu (“hirugarrenen”) plugin-en>gatik (barne legala). Erabili zure arriskuarekin. Erroreak/ arazoak dituzu, jakinarazi pluginaren biltegiari.\n\nPlugin batek edozein zerbitzu/legalki entitate baten ToS/DMCA hautsi baditu, eska iezaiozu pluginaren egileari edo hosting plataformari (adibidez GitHub/Codeberg) neurriak har ditzaten. “Hirugarrena” etiketatutako plugin guztiak komunitate publikoaren bidez mantentzen dira; ez ditugu kuratoriatu, beraz ezin dugu inplikatu.\n\n'; + + @override + String get input_does_not_match_format => + 'Sarrera ezin da beharrezko formatutik desberdina izan'; + + @override + String get plugins => 'Pluginak'; + + @override + String get paste_plugin_download_url => + 'Kopiatu deskarga-URLa, GitHub/Codeberg biltegi-URLa edo .smplug fitxategiaren esteka zuzena'; + + @override + String get download_and_install_plugin_from_url => + 'Download eta instalatu plugin-a URL batetik'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Plugin gehitu ezin izan da: $error'; + } + + @override + String get upload_plugin_from_file => 'Plugin fitxategi batetik igo'; + + @override + String get installed => 'Instalatuta'; + + @override + String get available_plugins => 'Eskaintzen diren pluginak'; + + @override + String get configure_plugins => + 'Konfiguratu zure metadatu-hornitzaile eta audio-iturburu pluginak'; + + @override + String get audio_scrobblers => 'Audio scrobbler-ak'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Iturburua: '; + + @override + String get uncompressed => 'Konprimitu gabea'; + + @override + String get dab_music_source_description => + 'Audiozaleentzat. Kalitate handiko/galerarik gabeko audio-streamak eskaintzen ditu. ISRC oinarritutako pistaren parekatze zehatza.'; +} diff --git a/lib/l10n/generated/app_localizations_fa.dart b/lib/l10n/generated/app_localizations_fa.dart new file mode 100644 index 00000000..5c0b7c2b --- /dev/null +++ b/lib/l10n/generated/app_localizations_fa.dart @@ -0,0 +1,1564 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Persian (`fa`). +class AppLocalizationsFa extends AppLocalizations { + AppLocalizationsFa([String locale = 'fa']) : super(locale); + + @override + String get guest => 'مهمان'; + + @override + String get browse => 'مرور'; + + @override + String get search => 'جستجو'; + + @override + String get library => 'مجموعه'; + + @override + String get lyrics => 'متن'; + + @override + String get settings => 'تنظیمات'; + + @override + String get genre_categories_filter => 'دسته ها یا ژانر ها را فیلتر کنید'; + + @override + String get genre => 'ژانر'; + + @override + String get personalized => ' شخصی سازی شده'; + + @override + String get featured => 'ویژه'; + + @override + String get new_releases => 'آخرین انتشارات'; + + @override + String get songs => 'آهنگ ها'; + + @override + String playing_track(Object track) { + return 'درحال پخش $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'با این کار صف فعلی پاک می شود. $track_length آهنگ از صف حذف میشود\n؟آیا ادامه میدهید'; + } + + @override + String get load_more => 'بارگذاری بیشتر'; + + @override + String get playlists => 'لیست های پخش'; + + @override + String get artists => 'هنرمندان'; + + @override + String get albums => 'آلبوم ها'; + + @override + String get tracks => 'آهنگ ها'; + + @override + String get downloads => 'بارگیری شده ها'; + + @override + String get filter_playlists => 'لیست پخش خود را فیلتر کنید...'; + + @override + String get liked_tracks => 'آهنگ های مورد علاقه'; + + @override + String get liked_tracks_description => 'همه آهنگ های دوست داشتنی شما'; + + @override + String get playlist => 'لیست پخش'; + + @override + String get create_a_playlist => 'ساخت لیست پخش'; + + @override + String get update_playlist => 'بروز کردن لیست پخش'; + + @override + String get create => 'ساختن'; + + @override + String get cancel => 'لغو'; + + @override + String get update => 'بروز رسانی'; + + @override + String get playlist_name => 'نام لیست پخش'; + + @override + String get name_of_playlist => 'نام لیست پخش'; + + @override + String get description => 'توضیحات'; + + @override + String get public => 'عمومی'; + + @override + String get collaborative => 'مبتنی بر همکاری'; + + @override + String get search_local_tracks => 'جستجوی آهنگ های محلی...'; + + @override + String get play => 'پخش'; + + @override + String get delete => 'حذف'; + + @override + String get none => 'هیچ کدام'; + + @override + String get sort_a_z => 'مرتب سازی بر اساس حروف الفبا'; + + @override + String get sort_z_a => 'مرتب سازی برعکس حروف الفبا'; + + @override + String get sort_artist => 'مرتب سازی بر اساس هنرمند'; + + @override + String get sort_album => 'مرتب سازی بر اساس آلبوم'; + + @override + String get sort_duration => 'مرتب کردن بر اساس مدت زمان'; + + @override + String get sort_tracks => 'مرتب سازی آهنگ ها'; + + @override + String currently_downloading(Object tracks_length) { + return 'در حال بارگیری ($tracks_length)'; + } + + @override + String get cancel_all => 'لغو همه'; + + @override + String get filter_artist => 'فیلتر کردن هنرمند...'; + + @override + String followers(Object followers) { + return '$followers دنبال کننده'; + } + + @override + String get add_artist_to_blacklist => 'اضافه کردن هنرمند به لیست سیاه'; + + @override + String get top_tracks => 'بهترین آهنگ ها'; + + @override + String get fans_also_like => 'طرفداران هم دوست داشتند'; + + @override + String get loading => 'بارگزاری...'; + + @override + String get artist => 'هنرمند'; + + @override + String get blacklisted => 'در لیست سیاه قرار گرفته است'; + + @override + String get following => 'دنبال کننده'; + + @override + String get follow => 'دنبال کردن'; + + @override + String get artist_url_copied => 'لینک هنرمند در کلیپ بورد کپی شد'; + + @override + String added_to_queue(Object tracks) { + return 'تعداد $tracks آهنگ به صف اضافه شد'; + } + + @override + String get filter_albums => 'فیلتر کردن آلبوم...'; + + @override + String get synced => 'همگام سازی شد'; + + @override + String get plain => 'ساده'; + + @override + String get shuffle => 'تصادفی'; + + @override + String get search_tracks => 'جستجوی آهنگ ها...'; + + @override + String get released => 'منتشر شده'; + + @override + String error(Object error) { + return 'خطا $error'; + } + + @override + String get title => 'عنوان'; + + @override + String get time => 'زمان'; + + @override + String get more_actions => 'اقدامات بیشتر'; + + @override + String download_count(Object count) { + return 'دانلود ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'اضافه کردن ($count) به لیست پخش'; + } + + @override + String add_count_to_queue(Object count) { + return 'اضافه کردن ($count) به صف'; + } + + @override + String play_count_next(Object count) { + return 'پخش ($count) بعدی'; + } + + @override + String get album => 'آلبوم'; + + @override + String copied_to_clipboard(Object data) { + return '$data در کلیپ بورد کپی شد'; + } + + @override + String add_to_following_playlists(Object track) { + return 'اضافه کردن $track به لیست پخش زیر'; + } + + @override + String get add => 'اضافه کردن'; + + @override + String added_track_to_queue(Object track) { + return '$track به لیست پخش اضافه شد'; + } + + @override + String get add_to_queue => 'اضافه کردن به صف'; + + @override + String track_will_play_next(Object track) { + return '$track پخش خواهد شد'; + } + + @override + String get play_next => 'پخش آهنگ بعدی'; + + @override + String removed_track_from_queue(Object track) { + return '$track از لیست پخش حذف شد'; + } + + @override + String get remove_from_queue => 'از لیست پخش حذف شد'; + + @override + String get remove_from_favorites => 'از علاقمندی ها حدف شد'; + + @override + String get save_as_favorite => 'ذخیره به عنوان علاقمندی ها'; + + @override + String get add_to_playlist => 'به لیست پخش اضافه کردن'; + + @override + String get remove_from_playlist => 'از لیست پخش حذف کردن'; + + @override + String get add_to_blacklist => 'به لیست سیاه اضافه کردن'; + + @override + String get remove_from_blacklist => 'از لیست سیاه حذف کردن'; + + @override + String get share => 'اشتراک گذاری'; + + @override + String get mini_player => 'پخش کننده '; + + @override + String get slide_to_seek => 'برای جستجو عقب یا جلو بکشید'; + + @override + String get shuffle_playlist => 'پخش تصادفی'; + + @override + String get unshuffle_playlist => 'خاموش کردن پخش تصادفی'; + + @override + String get previous_track => 'آهنگ قبلی'; + + @override + String get next_track => 'آهنگ بعدی'; + + @override + String get pause_playback => 'توقف آهنگ'; + + @override + String get resume_playback => 'ادامه آهنگ'; + + @override + String get loop_track => 'تکرار آهنگ'; + + @override + String get no_loop => 'بدون حلقه'; + + @override + String get repeat_playlist => 'تکرار لیست پخش'; + + @override + String get queue => 'صف'; + + @override + String get alternative_track_sources => ' منبع آهنگ را جاگزین کردن '; + + @override + String get download_track => 'بارگیری آهنگ'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks آهنگ در صف'; + } + + @override + String get clear_all => 'همه را حدف کن'; + + @override + String get show_hide_ui_on_hover => 'نمایش/پنهان رابط کاربری در حالت شناور'; + + @override + String get always_on_top => 'همیشه روشن'; + + @override + String get exit_mini_player => 'از پخش کننده خارج شوید'; + + @override + String get download_location => 'محل بارگیری'; + + @override + String get local_library => 'کتابخانه محلی'; + + @override + String get add_library_location => 'اضافه کردن به کتابخانه'; + + @override + String get remove_library_location => 'حذف از کتابخانه'; + + @override + String get account => 'حساب کاربری'; + + @override + String get logout => 'خارج شدن'; + + @override + String get logout_of_this_account => 'از حساب کاربری خارج شوید'; + + @override + String get language_region => 'زبان و منطقه '; + + @override + String get language => 'زبان '; + + @override + String get system_default => 'پیش فرض سیستم'; + + @override + String get market_place_region => 'منطقه'; + + @override + String get recommendation_country => 'کشور های پیشنهادی'; + + @override + String get appearance => 'ظاهر'; + + @override + String get layout_mode => 'حالت چیدمان'; + + @override + String get override_layout_settings => + 'تنطیمات حالت واکنشگرای چیدمان را لغو کن'; + + @override + String get adaptive => 'قابل تطبیق'; + + @override + String get compact => 'فشرده'; + + @override + String get extended => 'گسترده'; + + @override + String get theme => 'تم'; + + @override + String get dark => 'تاریک'; + + @override + String get light => 'روشن'; + + @override + String get system => 'سیستم'; + + @override + String get accent_color => 'رنگ تاکیدی'; + + @override + String get sync_album_color => 'هنگام سازی رنگ البوم'; + + @override + String get sync_album_color_description => + 'از رنگ البوم هنرمند به عنوان رنگ تاکیدی استفاده میکند'; + + @override + String get playback => 'پخش'; + + @override + String get audio_quality => 'کیفیت صدا'; + + @override + String get high => 'زیاد'; + + @override + String get low => 'کم'; + + @override + String get pre_download_play => 'دانلود و پخش کنید'; + + @override + String get pre_download_play_description => + 'به جای پخش جریانی صدا، بایت ها را دانلود کنید و به جای آن پخش کنید (برای کاربران با پهنای باند بالاتر توصیه می شود)'; + + @override + String get skip_non_music => 'رد شدن از پخش های غیر موسیقی (SponsorBlock)'; + + @override + String get blacklist_description => 'آهنگ ها و هنرمند های در لیست سیاه'; + + @override + String get wait_for_download_to_finish => + 'لطفا صبر کنید تا دانلود آهنگ جاری تمام شود'; + + @override + String get desktop => 'میز کار'; + + @override + String get close_behavior => 'رفتار نزدیک'; + + @override + String get close => 'بستن'; + + @override + String get minimize_to_tray => 'پتجره را کوچک کنید'; + + @override + String get show_tray_icon => 'نماد را نمایش بده'; + + @override + String get about => 'درباره'; + + @override + String get u_love_spotube => 'دوست داریدSpotubeما میدانیم شما '; + + @override + String get check_for_updates => 'بروزرسانی را بررسی کنید'; + + @override + String get about_spotube => 'Spotube درباره'; + + @override + String get blacklist => 'لیست سیاه'; + + @override + String get please_sponsor => 'لطفا کمک/حمایت کنید'; + + @override + String get spotube_description => + 'یک برنامه سبک و مولتی پلتفرم و رایگان برای همه استSpotube'; + + @override + String get version => 'نسخه'; + + @override + String get build_number => 'شماره ساخت'; + + @override + String get founder => 'بنیانگذار'; + + @override + String get repository => 'مخزن'; + + @override + String get bug_issues => 'اشکال+مسایل'; + + @override + String get made_with => '🇧🇩ساخته شده با ❤️ در بنگلادش'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'مجوز'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'نگران نباشید هیچ کدوما از اعتبارات شما جمع اوری نمیشود یا با کسی اشتراک گزاشته نمیشود'; + + @override + String get know_how_to_login => 'نمیدانی چگونه این کار را انجام بدهی؟'; + + @override + String get follow_step_by_step_guide => 'راهنما را گام به گام دنبال کنید'; + + @override + String cookie_name_cookie(Object name) { + return '$name کوکی'; + } + + @override + String get fill_in_all_fields => 'لطفا تمام فلید ها را پر کنید'; + + @override + String get submit => 'ثبت'; + + @override + String get exit => 'خروج'; + + @override + String get previous => 'قبلی'; + + @override + String get next => 'بعدی '; + + @override + String get done => 'اتمام'; + + @override + String get step_1 => 'گام 1'; + + @override + String get first_go_to => 'اول برو داخل '; + + @override + String get something_went_wrong => 'اشتباهی رخ داده'; + + @override + String get piped_instance => 'مشکل در ارتباط با سرور'; + + @override + String get piped_description => 'مشکل در ارتباط با سرور در دریافت آهنگ ها'; + + @override + String get piped_warning => + 'برخی از آنها ممکن است خوب کارنکند.بنابراین با مسولیت خود استفاده کنید'; + + @override + String get invidious_instance => 'نمونه سرور Invidious'; + + @override + String get invidious_description => 'نمونه سرور Invidious برای تطبیق آهنگ'; + + @override + String get invidious_warning => + 'برخی از نمونه‌ها ممکن است به خوبی کار نکنند. با احتیاط استفاده کنید'; + + @override + String get generate => 'ایجاد'; + + @override + String track_exists(Object track) { + return 'آهنگ $track وجود دارد'; + } + + @override + String get replace_downloaded_tracks => + 'همه ی آهنگ های دانلود شده را جایگزین کنید'; + + @override + String get skip_download_tracks => 'همه ی آهنگ های دانلود شده را رد کنید'; + + @override + String get do_you_want_to_replace => + 'ایا میخواهید آهنگ های موجود جایگزین کنید؟'; + + @override + String get replace => 'جایگزین کردن'; + + @override + String get skip => 'رد کردن'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'انتخاب کنید تا $count $type'; + } + + @override + String get select_genres => 'ژانر ها را انتخاب کنید'; + + @override + String get add_genres => 'ژانر را اطافه کنید'; + + @override + String get country => 'کشور'; + + @override + String get number_of_tracks_generate => 'تعداد آهنگ های ساخته شده'; + + @override + String get acousticness => 'آکوستیک'; + + @override + String get danceability => 'رقصیدن'; + + @override + String get energy => 'انرژی'; + + @override + String get instrumentalness => 'بی کلام'; + + @override + String get liveness => 'حس زندگی'; + + @override + String get loudness => 'صدای بلند'; + + @override + String get speechiness => 'دکلمه'; + + @override + String get valence => 'ظرفیت'; + + @override + String get popularity => 'محبوبیت'; + + @override + String get key => 'کلید'; + + @override + String get duration => 'مدت زمان (ثانیه)'; + + @override + String get tempo => 'تمپو (BPM)'; + + @override + String get mode => 'حالت'; + + @override + String get time_signature => 'امضای زمان'; + + @override + String get short => 'کوتاه'; + + @override + String get medium => 'متوسط'; + + @override + String get long => 'بلند'; + + @override + String get min => 'حداقل'; + + @override + String get max => 'حداکثر'; + + @override + String get target => 'هدف'; + + @override + String get moderate => 'حد وسط'; + + @override + String get deselect_all => 'همه را لغو انتخاب کنید'; + + @override + String get select_all => 'همه را انتخاب کنید'; + + @override + String get are_you_sure => 'ایا مطمعن هستید؟'; + + @override + String get generating_playlist => ' درحال ایجاد لیست پخش سفارشی شما'; + + @override + String selected_count_tracks(Object count) { + return 'آهنگ انتخاب شده $count'; + } + + @override + String get download_warning => + 'اگر همه ی آهنگ ها را به صورت انبو دانلود کنید به وضوح در حال دزدی موسقی هستید و در حال اسیب وارد کردن به جامه ی خلاق هنری می باشید .امیدوارم که از این موضوع اگاه باشید .همیشه سعی کنید به کار سخت هنرمند اخترام بگذارید.'; + + @override + String get download_ip_ban_warning => + 'راستی آی پی شما می تواند در یوتوب به دلیل درخواست های دانلود بیش از حد معمول مسدود شود. بلوک آی پی به این معنی است که شما نمی توانید از یوتوب (حتی اگر وارد سیستم شده باشید) حداقل 2-3 ماه از آن دستگاه آی پی استفاده کنید. و Spotube هیچ مسئولیتی در صورت وقوع این اتفاق ندارد'; + + @override + String get by_clicking_accept_terms => + 'با کلیک بر روی قبول با شرایط زیر موافقت می کنید:'; + + @override + String get download_agreement_1 => 'من میدانم در حال دزدی هستم .من بد هستم'; + + @override + String get download_agreement_2 => + 'من هر کجا ک بتوانم از هنرمندان حمایت میکنم اما این کارا فقط به دلیل اینکه توانایی مالی ندارم انجام میدهم'; + + @override + String get download_agreement_3 => + 'من کاملا میدانم که از طرف یوتوب بلاک میشم و این برنامه و مالکان را مسول این حادثه نمیدانم.'; + + @override + String get decline => 'قبول نکردن'; + + @override + String get accept => 'قبول'; + + @override + String get details => 'جزئیات'; + + @override + String get youtube => 'یوتیوب'; + + @override + String get channel => 'کانال'; + + @override + String get likes => 'دوست داشتن'; + + @override + String get dislikes => 'دوست نداشتن'; + + @override + String get views => 'بازدید'; + + @override + String get streamUrl => 'لینک اثر'; + + @override + String get stop => 'توقف'; + + @override + String get sort_newest => 'مرتب سازی بر اساس جدید ترین اضافه شده'; + + @override + String get sort_oldest => 'مرتب سازی بر اساس قدیمی ترین اضافه شده'; + + @override + String get sleep_timer => 'زمان خواب'; + + @override + String mins(Object minutes) { + return '$minutes دقیقه'; + } + + @override + String hours(Object hours) { + return '$hours ساعت'; + } + + @override + String hour(Object hours) { + return '$hours ساعت'; + } + + @override + String get custom_hours => 'ساعت سفارشی'; + + @override + String get logs => 'رسید خطا'; + + @override + String get developers => 'توسعه دهنده ها'; + + @override + String get not_logged_in => 'شما وارد نشده اید '; + + @override + String get search_mode => 'حالت جستجو'; + + @override + String get audio_source => 'منبع صدا'; + + @override + String get ok => 'باشد'; + + @override + String get failed_to_encrypt => 'رمز گذاری نشده'; + + @override + String get encryption_failed_warning => + 'Spotube از رمزگذاری برای ذخیره ایمن داده های شما استفاده می کند. اما موفق به انجام این کار نشد. بنابراین به فضای ذخیره‌سازی ناامن تبدیل می‌شود\nاگر از لینوکس استفاده می‌کنید، لطفاً مطمئن شوید که سرویس مخفی (gnome-keyring، kde-wallet، keepassxc و غیره) را نصب کرده‌اید.'; + + @override + String get querying_info => 'جستجو درباره '; + + @override + String get piped_api_down => 'ایراد در سرور'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'به دلیل مشکل $pipedInstance ارتباط با سرور مقدور نیست\n\nنمونه را تغییر دهید یا «نوع API» را به API رسمی YouTube تغییر دهید\n\nحتماً پس از تغییر، برنامه را دوباره راه‌اندازی کنید'; + } + + @override + String get you_are_offline => 'شما در حال حاضر افلاین هستید '; + + @override + String get connection_restored => 'اتصال به اینترنت شما بازیابی شد '; + + @override + String get use_system_title_bar => 'از نوار عنوان سیستم استفاده کنید '; + + @override + String get crunching_results => 'نتایج خرد کردن...'; + + @override + String get search_to_get_results => 'جستجو کنید تا به نتیجه برسید'; + + @override + String get use_amoled_mode => 'استفاده از حالت AMOLED'; + + @override + String get pitch_dark_theme => 'تم تیره دارت'; + + @override + String get normalize_audio => 'نرمال کردن صدا'; + + @override + String get change_cover => 'تغییر جلد'; + + @override + String get add_cover => 'افزودن جلد'; + + @override + String get restore_defaults => 'بازیابی پیش فرض ها'; + + @override + String get download_music_format => 'فرمت دانلود موسیقی'; + + @override + String get streaming_music_format => 'فرمت پخش آنلاین موسیقی'; + + @override + String get download_music_quality => 'کیفیت دانلود موسیقی'; + + @override + String get streaming_music_quality => 'کیفیت پخش آنلاین موسیقی'; + + @override + String get login_with_lastfm => 'ورود با Last.fm'; + + @override + String get connect => 'اتصال'; + + @override + String get disconnect_lastfm => 'قطع ارتباط با Last.fm'; + + @override + String get disconnect => 'قطع ارتباط'; + + @override + String get username => 'نام کاربری'; + + @override + String get password => 'رمز عبور'; + + @override + String get login => 'ورود'; + + @override + String get login_with_your_lastfm => 'ورود با حساب کاربری Last.fm خود'; + + @override + String get scrobble_to_lastfm => 'Scrobble به Last.fm'; + + @override + String get go_to_album => 'رفتن به آلبوم'; + + @override + String get discord_rich_presence => 'حضور غنی دیسکورد'; + + @override + String get browse_all => 'مرور همه'; + + @override + String get genres => 'ژانرها'; + + @override + String get explore_genres => 'استکشاف ژانرها'; + + @override + String get friends => 'دوستان'; + + @override + String get no_lyrics_available => + 'متاسفیم، قادر به یافتن متن این قطعه نیستیم'; + + @override + String get start_a_radio => 'شروع یک رادیو'; + + @override + String get how_to_start_radio => 'چگونه می‌خواهید رادیو را شروع کنید؟'; + + @override + String get replace_queue_question => + 'آیا می‌خواهید لیست پخش فعلی را جایگزین کنید یا به آن اضافه کنید؟'; + + @override + String get endless_playback => 'پخش بی‌پایان'; + + @override + String get delete_playlist => 'حذف لیست پخش'; + + @override + String get delete_playlist_confirmation => + 'آیا مطمئن هستید که می‌خواهید این لیست پخش را حذف کنید؟'; + + @override + String get local_tracks => 'موسیقی‌های محلی'; + + @override + String get local_tab => 'محلی'; + + @override + String get song_link => 'پیوند آهنگ'; + + @override + String get skip_this_nonsense => 'این احمقانه را بگذرانید'; + + @override + String get freedom_of_music => '“آزادی موسیقی”'; + + @override + String get freedom_of_music_palm => '“آزادی موسیقی در دستان شما”'; + + @override + String get get_started => 'بیایید شروع کنیم'; + + @override + String get youtube_source_description => 'پیشنهاد شده و بهترین عمل می‌کند.'; + + @override + String get piped_source_description => + 'احساس آزادی می‌کنید؟ مانند یوتیوب اما بیشتر آزاد.'; + + @override + String get jiosaavn_source_description => 'بهترین برای منطقه جنوب آسیا.'; + + @override + String get invidious_source_description => + 'شبیه Piped اما با در دسترس بودن بیشتر'; + + @override + String highest_quality(Object quality) { + return 'بالاترین کیفیت: $quality'; + } + + @override + String get select_audio_source => 'انتخاب منبع صوتی'; + + @override + String get endless_playback_description => + 'خودکار اضافه کردن آهنگ‌های جدید\nبه انتهای صف'; + + @override + String get choose_your_region => 'منطقه خود را انتخاب کنید'; + + @override + String get choose_your_region_description => + 'این به Spotube کمک می‌کند تا محتوای مناسبی را برای موقعیت شما نشان دهد.'; + + @override + String get choose_your_language => 'زبان خود را انتخاب کنید'; + + @override + String get help_project_grow => 'کمک به رشد این پروژه'; + + @override + String get help_project_grow_description => + 'Spotube یک پروژه متن باز است. شما می‌توانید با به پروژه کمک کردن، گزارش دادن اشکالات یا پیشنهاد ویژگی‌های جدید، به این پروژه کمک کنید.'; + + @override + String get contribute_on_github => 'مشارکت در GitHub'; + + @override + String get donate_on_open_collective => 'کمک مالی در Open Collective'; + + @override + String get browse_anonymously => 'مرور به صورت ناشناس'; + + @override + String get enable_connect => 'فعال‌سازی اتصال'; + + @override + String get enable_connect_description => 'کنترل Spotube از دیگر دستگاه‌ها'; + + @override + String get devices => 'دستگاه‌ها'; + + @override + String get select => 'انتخاب'; + + @override + String connect_client_alert(Object client) { + return 'شما توسط $client کنترل می‌شوید'; + } + + @override + String get this_device => 'این دستگاه'; + + @override + String get remote => 'راه‌دور'; + + @override + String get stats => 'آمار'; + + @override + String and_n_more(Object count) { + return 'و $count بیشتر'; + } + + @override + String get recently_played => 'اخیراً پخش شده'; + + @override + String get browse_more => 'بیشتر مرور کنید'; + + @override + String get no_title => 'بدون عنوان'; + + @override + String get not_playing => 'در حال پخش نیست'; + + @override + String get epic_failure => 'شکست حماسی!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length ترک به صف اضافه شد'; + } + + @override + String get spotube_has_an_update => 'Spotube یک بروزرسانی دارد'; + + @override + String get download_now => 'اکنون دانلود کنید'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'نسخه شبانه Spotube $nightlyBuildNum منتشر شد'; + } + + @override + String release_version(Object version) { + return 'نسخه Spotube v$version منتشر شد'; + } + + @override + String get read_the_latest => 'آخرین‌ها را بخوانید'; + + @override + String get release_notes => 'یادداشت‌های انتشار'; + + @override + String get pick_color_scheme => 'طرح رنگ را انتخاب کنید'; + + @override + String get save => 'ذخیره'; + + @override + String get choose_the_device => 'دستگاه را انتخاب کنید:'; + + @override + String get multiple_device_connected => + 'چندین دستگاه متصل هستند.\nدستگاهی را انتخاب کنید که می‌خواهید این عملیات بر روی آن انجام شود'; + + @override + String get nothing_found => 'چیزی پیدا نشد'; + + @override + String get the_box_is_empty => 'جعبه خالی است'; + + @override + String get top_artists => 'بهترین هنرمندان'; + + @override + String get top_albums => 'بهترین آلبوم‌ها'; + + @override + String get this_week => 'این هفته'; + + @override + String get this_month => 'این ماه'; + + @override + String get last_6_months => '۶ ماه گذشته'; + + @override + String get this_year => 'امسال'; + + @override + String get last_2_years => '۲ سال گذشته'; + + @override + String get all_time => 'همیشه'; + + @override + String powered_by_provider(Object providerName) { + return 'توسط $providerName پشتیبانی شده است'; + } + + @override + String get email => 'ایمیل'; + + @override + String get profile_followers => 'دنبال‌کنندگان'; + + @override + String get birthday => 'تولد'; + + @override + String get subscription => 'اشتراک'; + + @override + String get not_born => 'متولد نشده'; + + @override + String get hacker => 'هکر'; + + @override + String get profile => 'پروفایل'; + + @override + String get no_name => 'بدون نام'; + + @override + String get edit => 'ویرایش'; + + @override + String get user_profile => 'پروفایل کاربر'; + + @override + String count_plays(Object count) { + return '$count پخش'; + } + + @override + String get streaming_fees_hypothetical => 'هزینه‌های پخش (فرضی)'; + + @override + String get minutes_listened => 'دقایق گوش داده شده'; + + @override + String get streamed_songs => 'ترانه‌های پخش شده'; + + @override + String count_streams(Object count) { + return '$count پخش'; + } + + @override + String get owned_by_you => 'توسط شما مالکیت شده'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl به کلیپ‌بورد کپی شد'; + } + + @override + String get hipotetical_calculation => + '*این محاسبه بر اساس میانگین پرداخت به ازای هر پخش (0.003 تا 0.005 دلار) در پلتفرم‌های استریم موزیک آنلاین انجام شده است. این یک محاسبه فرضی است که به کاربر دیدی از مقدار پرداختی به هنرمندان در صورت گوش دادن به آهنگ آن‌ها در پلتفرم‌های مختلف ارائه می‌دهد.'; + + @override + String count_mins(Object minutes) { + return '$minutes دقیقه'; + } + + @override + String get summary_minutes => 'دقیقه‌ها'; + + @override + String get summary_listened_to_music => 'به موسیقی گوش داده شده'; + + @override + String get summary_songs => 'ترانه‌ها'; + + @override + String get summary_streamed_overall => 'پخش شده به طور کلی'; + + @override + String get summary_owed_to_artists => 'به هنرمندان بدهکار است\nاین ماه'; + + @override + String get summary_artists => 'هنرمندان'; + + @override + String get summary_music_reached_you => 'موسیقی به شما رسیده است'; + + @override + String get summary_full_albums => 'آلبوم‌های کامل'; + + @override + String get summary_got_your_love => 'عشق شما را به دست آورد'; + + @override + String get summary_playlists => 'لیست‌های پخش'; + + @override + String get summary_were_on_repeat => 'در تکرار بودند'; + + @override + String total_money(Object money) { + return 'مجموع $money'; + } + + @override + String get webview_not_found => 'وب‌ویو پیدا نشد'; + + @override + String get webview_not_found_description => + 'هیچ اجرای وب‌ویو روی دستگاه شما نصب نشده است.\nدر صورت نصب، مطمئن شوید که در environment PATH قرار دارد\n\nپس از نصب، برنامه را مجدداً راه‌اندازی کنید'; + + @override + String get unsupported_platform => 'پلتفرم پشتیبانی نمی‌شود'; + + @override + String get cache_music => 'موسیقی در حافظه موقت'; + + @override + String get open => 'باز کردن'; + + @override + String get cache_folder => 'پوشه حافظه موقت'; + + @override + String get export => 'صادر کردن'; + + @override + String get clear_cache => 'پاک کردن حافظه موقت'; + + @override + String get clear_cache_confirmation => + 'آیا می‌خواهید حافظه موقت را پاک کنید؟'; + + @override + String get export_cache_files => 'صادر کردن فایل‌های حافظه موقت'; + + @override + String found_n_files(Object count) { + return '$count فایل یافت شد'; + } + + @override + String get export_cache_confirmation => + 'آیا می‌خواهید این فایل‌ها را صادر کنید به'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported از $files فایل صادر شد'; + } + + @override + String get undo => 'بازگشت'; + + @override + String get download_all => 'دانلود همه'; + + @override + String get add_all_to_playlist => 'افزودن همه به لیست پخش'; + + @override + String get add_all_to_queue => 'افزودن همه به صف'; + + @override + String get play_all_next => 'پخش همه بعدی'; + + @override + String get pause => 'مکث'; + + @override + String get view_all => 'مشاهده همه'; + + @override + String get no_tracks_added_yet => + 'به نظر می‌رسد هنوز هیچ آهنگی اضافه نکرده‌اید.'; + + @override + String get no_tracks => 'به نظر می‌رسد هیچ آهنگی در اینجا وجود ندارد.'; + + @override + String get no_tracks_listened_yet => 'به نظر می‌رسد هنوز چیزی نشنیده‌اید.'; + + @override + String get not_following_artists => 'شما هیچ هنرمندی را دنبال نمی‌کنید.'; + + @override + String get no_favorite_albums_yet => + 'به نظر می‌رسد هنوز هیچ آلبومی را به علاقه‌مندی‌هایتان اضافه نکرده‌اید.'; + + @override + String get no_logs_found => 'هیچ لاگی پیدا نشد'; + + @override + String get youtube_engine => 'موتور YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine نصب نشده است'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine در سیستم شما نصب نشده است.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'اطمینان حاصل کنید که در متغیر PATH موجود است یا\nآدرس مطلق فایل اجرایی $engine را در زیر تنظیم کنید.'; + } + + @override + String get youtube_engine_unix_issue_message => + 'در macOS/Linux/سیستم‌عامل‌های مشابه Unix، تنظیم مسیر در .zshrc/.bashrc/.bash_profile و غیره کار نمی‌کند.\nباید مسیر را در فایل پیکربندی شل تنظیم کنید.'; + + @override + String get download => 'دانلود'; + + @override + String get file_not_found => 'فایل پیدا نشد'; + + @override + String get custom => 'شخصی‌سازی شده'; + + @override + String get add_custom_url => 'اضافه کردن URL سفارشی'; + + @override + String get edit_port => 'ویرایش پورت'; + + @override + String get port_helper_msg => + 'پیش‌فرض -1 است که نشان‌دهنده یک عدد تصادفی است. اگر فایروال شما پیکربندی شده است، توصیه می‌شود این را تنظیم کنید.'; + + @override + String connect_request(Object client) { + return 'آیا اجازه می‌دهید $client متصل شود؟'; + } + + @override + String get connection_request_denied => + 'اتصال رد شد. کاربر دسترسی را رد کرد.'; + + @override + String get an_error_occurred => 'خطایی رخ داد'; + + @override + String get copy_to_clipboard => 'کپی به کلیپ‌بورد'; + + @override + String get view_logs => 'مشاهده لاگ‌ها'; + + @override + String get retry => 'دوباره تلاش کن'; + + @override + String get no_default_metadata_provider_selected => + 'هیچ ارائه‌دهندهٔ پیش‌فرض متادیتا تعیین نکرده‌اید'; + + @override + String get manage_metadata_providers => 'مدیریت ارائه‌دهندگان متادیتا'; + + @override + String get open_link_in_browser => 'باز کردن لینک در مرورگر؟'; + + @override + String get do_you_want_to_open_the_following_link => + 'آیا می‌خواهید لینک زیر را باز کنید؟'; + + @override + String get unsafe_url_warning => + 'باز کردن لینک از منابع نامطمئن می‌تواند ناامن باشد. مراقب باشید!\nهمچنین می‌توانید لینک را در کلیپ‌بورد خود کپی کنید.'; + + @override + String get copy_link => 'کپی لینک'; + + @override + String get building_your_timeline => + 'در حال ساخت جدول زمانی بر اساس شنیده‌هایتان…'; + + @override + String get official => 'رسمی'; + + @override + String author_name(Object author) { + return 'نویسنده: $author'; + } + + @override + String get third_party => 'سوم‌شخص'; + + @override + String get plugin_requires_authentication => 'افزونه نیاز به احراز هویت دارد'; + + @override + String get update_available => 'به‌روزرسانی در دسترس است'; + + @override + String get supports_scrobbling => 'پشتیبانی از اسکراب‌بلینگ'; + + @override + String get plugin_scrobbling_info => + 'این افزونه موسیقی شما را اسکراب می‌کند تا تاریخچهٔ شنیداری‌تان را تولید کند.'; + + @override + String get default_metadata_source => 'منبع پیش‌فرض فراداده'; + + @override + String get set_default_metadata_source => 'تنظیم منبع پیش‌فرض فراداده'; + + @override + String get default_audio_source => 'منبع پیش‌فرض صوت'; + + @override + String get set_default_audio_source => 'تنظیم منبع پیش‌فرض صوت'; + + @override + String get set_default => 'تنظیم به عنوان پیش‌فرض'; + + @override + String get support => 'پشتیبانی'; + + @override + String get support_plugin_development => 'حمایت از توسعهٔ افزونه'; + + @override + String can_access_name_api(Object name) { + return '- می‌تواند به API **$name** دسترسی پیدا کند'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'می‌خواهید این افزونه را نصب کنید؟'; + + @override + String get third_party_plugin_warning => + 'این افزونه از مخزن شخص ثالث آمده است. لطفاً قبل از نصب از منابع آن مطمئن شوید.'; + + @override + String get author => 'نویسنده'; + + @override + String get this_plugin_can_do_following => + 'این افزونه می‌تواند موارد زیر را انجام دهد'; + + @override + String get install => 'نصب'; + + @override + String get install_a_metadata_provider => 'نصب یک ارائه‌دهندهٔ متادیتا'; + + @override + String get no_tracks_playing => 'در حال‌ حاضر هیچ تراکی در حال پخش نیست'; + + @override + String get synced_lyrics_not_available => + 'متن هم‌زمان‌شده برای این آهنگ در دسترس نیست. لطفاً از'; + + @override + String get plain_lyrics => 'متن ساده'; + + @override + String get tab_instead => 'به‌جای آن از کلید Tab استفاده کنید.'; + + @override + String get disclaimer => 'سلب مسئولیت'; + + @override + String get third_party_plugin_dmca_notice => + 'تیم Spotube هیچ مسئولیتی (حتی قانونی) در قبال افزونه‌های \"شخص ثالث\" ندارد. از آن‌ها به‌خاطر خود استفاده کنید. برای خطاها/مشکلات، لطفاً در مخزن افزونه گزارش دهید.\n\nاگر هر افزونهٔ \"شخص ثالث\" قوانین ToS/DMCA سرویس یا نهاد قانونی را نقض کند، لطفاً از نویسندهٔ افزونه یا پلتفرم میزبانی (مثل GitHub/Codeberg) درخواست اقدام کنید. افزونه‌هایی که با برچسب \"شخص ثالث\" مشخص شده‌اند، عمومی هستند و توسط جامعه نگهداری می‌شوند؛ ما آن‌ها را تغییر یا مدیریت نمی‌کنیم و نمی‌توانیم دخالت کنیم.\n\n'; + + @override + String get input_does_not_match_format => + 'ورودی با قالب مورد نیاز تطابق ندارد'; + + @override + String get plugins => 'افزونه‌ها'; + + @override + String get paste_plugin_download_url => + 'URL دانلود یا مخزن GitHub/Codeberg یا لینک مستقیم فایل .smplug را الصاق کنید'; + + @override + String get download_and_install_plugin_from_url => + 'دانلود و نصب افزونه از طریق لینک'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'افزونه اضافه نشد: $error'; + } + + @override + String get upload_plugin_from_file => 'بارگذاری افزونه از فایل'; + + @override + String get installed => 'نصب شد'; + + @override + String get available_plugins => 'افزونه‌های موجود'; + + @override + String get configure_plugins => + 'افزونه‌های منبع صوت و ارائه‌دهنده فراداده خود را پیکربندی کنید'; + + @override + String get audio_scrobblers => 'اسکراب‌بلرهای صوتی'; + + @override + String get scrobbling => 'اسکراب‌بلینگ'; + + @override + String get source => 'منبع: '; + + @override + String get uncompressed => 'بدون فشرده‌سازی'; + + @override + String get dab_music_source_description => + 'مخصوص علاقه‌مندان صدا. ارائه‌دهنده استریم‌های باکیفیت/بدون افت. تطبیق دقیق آهنگ بر اساس ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_fi.dart b/lib/l10n/generated/app_localizations_fi.dart new file mode 100644 index 00000000..3f616849 --- /dev/null +++ b/lib/l10n/generated/app_localizations_fi.dart @@ -0,0 +1,1564 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Finnish (`fi`). +class AppLocalizationsFi extends AppLocalizations { + AppLocalizationsFi([String locale = 'fi']) : super(locale); + + @override + String get guest => 'Vieras'; + + @override + String get browse => 'Selaa'; + + @override + String get search => 'Hae'; + + @override + String get library => 'Kirjasto'; + + @override + String get lyrics => 'Lyriikat'; + + @override + String get settings => 'Asetukset'; + + @override + String get genre_categories_filter => 'Suodata kategorioita tai genrejä'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Personoidut'; + + @override + String get featured => 'Esittelyssä'; + + @override + String get new_releases => 'Uusi julkaisu'; + + @override + String get songs => 'Laulut'; + + @override + String playing_track(Object track) { + return 'Soitetaan $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Tämä tulee tyhjentämään jonon. $track_length Kappaleita poistetaan\nHaluatko jatkaa?'; + } + + @override + String get load_more => 'Lataa lisää'; + + @override + String get playlists => 'Soittolistat'; + + @override + String get artists => 'Artistit'; + + @override + String get albums => 'Albumit'; + + @override + String get tracks => 'Kappaleet'; + + @override + String get downloads => 'Lataukset'; + + @override + String get filter_playlists => 'Suodata soittolistasi...'; + + @override + String get liked_tracks => 'Tykätyt kappaleet'; + + @override + String get liked_tracks_description => 'Kaikki tykättysi kappaleet'; + + @override + String get playlist => 'Soittolista'; + + @override + String get create_a_playlist => 'Luo soittolista'; + + @override + String get update_playlist => 'Päivitä soittolista'; + + @override + String get create => 'Luo'; + + @override + String get cancel => 'Peruuta'; + + @override + String get update => 'Päivitä'; + + @override + String get playlist_name => 'Soittolistan nimi'; + + @override + String get name_of_playlist => 'Soittolistan nimi'; + + @override + String get description => 'Kuvaus'; + + @override + String get public => 'Julkinen'; + + @override + String get collaborative => 'Collaborative'; + + @override + String get search_local_tracks => 'Hae paikallisia lauluja...'; + + @override + String get play => 'Soita'; + + @override + String get delete => 'Poista'; + + @override + String get none => 'Ei mitään'; + + @override + String get sort_a_z => 'Suodata A-Z'; + + @override + String get sort_z_a => 'Suodata Z-A'; + + @override + String get sort_artist => 'Suodata Artistilta'; + + @override + String get sort_album => 'Suodata Albumilta'; + + @override + String get sort_duration => 'Suodata Pituudelta'; + + @override + String get sort_tracks => 'Suodata Kappaleet'; + + @override + String currently_downloading(Object tracks_length) { + return 'Ladataan ($tracks_length)'; + } + + @override + String get cancel_all => 'Peru kaikki'; + + @override + String get filter_artist => 'Suodata artistit...'; + + @override + String followers(Object followers) { + return '$followers Seuraajaa'; + } + + @override + String get add_artist_to_blacklist => 'Lisää artisti mustalle listalle'; + + @override + String get top_tracks => 'Suosituimmat kappaleet'; + + @override + String get fans_also_like => 'Fanit myös tykkäsivät'; + + @override + String get loading => 'Ladataan...'; + + @override + String get artist => 'Artisti'; + + @override + String get blacklisted => 'Mustalistattu'; + + @override + String get following => 'Seurataan'; + + @override + String get follow => 'Seuraa'; + + @override + String get artist_url_copied => 'Aristin URL kopioitiin leikepöytään'; + + @override + String added_to_queue(Object tracks) { + return 'Lisättiin $tracks kappaletta jonoon'; + } + + @override + String get filter_albums => 'Suodata albumit...'; + + @override + String get synced => 'Synkronoitu'; + + @override + String get plain => 'Tavallinen'; + + @override + String get shuffle => 'Sekoita'; + + @override + String get search_tracks => 'Hae kappaleita...'; + + @override + String get released => 'Julkaistu'; + + @override + String error(Object error) { + return 'Virhe $error'; + } + + @override + String get title => 'Otsikko'; + + @override + String get time => 'Aika'; + + @override + String get more_actions => 'Lisää toimintoja'; + + @override + String download_count(Object count) { + return 'Lataa ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Lisää ($count) Soittolistaasi'; + } + + @override + String add_count_to_queue(Object count) { + return 'Lisää ($count) Jonoon'; + } + + @override + String play_count_next(Object count) { + return 'Soita ($count) seuraavaksi'; + } + + @override + String get album => 'Albumi'; + + @override + String copied_to_clipboard(Object data) { + return 'Kopioitiin $data leikepöytään'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Lisää $track seuraaviin soittolistoihin'; + } + + @override + String get add => 'Lisää'; + + @override + String added_track_to_queue(Object track) { + return 'Lisättiin $track jonoon'; + } + + @override + String get add_to_queue => 'Lisää jonoon'; + + @override + String track_will_play_next(Object track) { + return '$track Soitetaan seuraavaksi'; + } + + @override + String get play_next => 'Soita seuraavaksi'; + + @override + String removed_track_from_queue(Object track) { + return 'Poistettiin $track jonosta'; + } + + @override + String get remove_from_queue => 'Poista jonosta'; + + @override + String get remove_from_favorites => 'Poista suosikeista'; + + @override + String get save_as_favorite => 'Tallenna soittolistana'; + + @override + String get add_to_playlist => 'Lisää soittolistaan'; + + @override + String get remove_from_playlist => 'Poista soittolistasta'; + + @override + String get add_to_blacklist => 'Lisää mustalle listalle'; + + @override + String get remove_from_blacklist => 'Poista mustalistalta'; + + @override + String get share => 'Jaa'; + + @override + String get mini_player => 'Minisoitin'; + + @override + String get slide_to_seek => 'Liu\'uta mennäkseen eteenpäin tai taaksepäin'; + + @override + String get shuffle_playlist => 'Sekoita soittolista'; + + @override + String get unshuffle_playlist => 'Poista sekoitus soittolistasta'; + + @override + String get previous_track => 'Äskeinen kappale'; + + @override + String get next_track => 'Seuraava kappale'; + + @override + String get pause_playback => 'Pysäytä soittolistan toisto'; + + @override + String get resume_playback => 'Jatka soittolistan toistoa'; + + @override + String get loop_track => 'Uudelleentoista kappale'; + + @override + String get no_loop => 'Ei silmukkaa'; + + @override + String get repeat_playlist => 'Toista soittolista uudelleen'; + + @override + String get queue => 'Jono'; + + @override + String get alternative_track_sources => 'Toinen kappale lähde'; + + @override + String get download_track => 'Lataa kappale'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks kappaletta jonossa'; + } + + @override + String get clear_all => 'Tyhjennä kaikki'; + + @override + String get show_hide_ui_on_hover => 'Näytä/Piilota UI leijumalla'; + + @override + String get always_on_top => 'Aina päällimmäisenä'; + + @override + String get exit_mini_player => 'Lähde minisoittimesta'; + + @override + String get download_location => 'Lataus sijainti'; + + @override + String get local_library => 'Paikallinen kirjasto'; + + @override + String get add_library_location => 'Lisää kirjastoon'; + + @override + String get remove_library_location => 'Poista kirjastosta'; + + @override + String get account => 'Käyttäjä'; + + @override + String get logout => 'Kirjaudu ulos'; + + @override + String get logout_of_this_account => 'Kirjaudu ulos tältä käyttäjältä'; + + @override + String get language_region => 'Kieli ja Maa'; + + @override + String get language => 'Kieli'; + + @override + String get system_default => 'Järjestelmän oletus'; + + @override + String get market_place_region => 'Markkina-alue'; + + @override + String get recommendation_country => 'Suositeltu maa'; + + @override + String get appearance => 'Ulkomuto'; + + @override + String get layout_mode => 'Asettelutila'; + + @override + String get override_layout_settings => + 'Jätä reagoiva asettelutila huomioimatta'; + + @override + String get adaptive => 'Mukautuva'; + + @override + String get compact => 'Kompakti'; + + @override + String get extended => 'Laajennettu'; + + @override + String get theme => 'Teema'; + + @override + String get dark => 'Tumma'; + + @override + String get light => 'Vaalea'; + + @override + String get system => 'Järjestelmä'; + + @override + String get accent_color => 'Korostusväri'; + + @override + String get sync_album_color => 'Synkronoi albumin väri'; + + @override + String get sync_album_color_description => + 'Käyttää albumin kansitaiteen vallitsevaa väirä korostuvärinä'; + + @override + String get playback => 'Toisto'; + + @override + String get audio_quality => 'Äänenlaatu'; + + @override + String get high => 'Korkea'; + + @override + String get low => 'Matala'; + + @override + String get pre_download_play => 'Esilataa ja soita'; + + @override + String get pre_download_play_description => + 'Audion suoratoiston sijaan, lataa tavut ja soita ne (Suositeltu korkeamman kaistanleveyden käyttäjille)'; + + @override + String get skip_non_music => 'Ohita ei-musiikki kohdat (SponsorBlock)'; + + @override + String get blacklist_description => 'Mustalistat kappaleet aja artistit'; + + @override + String get wait_for_download_to_finish => + 'Odota nykyisen latauksen lopetteluun'; + + @override + String get desktop => 'Työpöytä'; + + @override + String get close_behavior => 'Sulkemisen käyttäytyminen'; + + @override + String get close => 'Sulje'; + + @override + String get minimize_to_tray => 'Minimisoi tehtäväpalkkiin'; + + @override + String get show_tray_icon => 'Näytä järjestelmäkuvake'; + + @override + String get about => 'Tietoa'; + + @override + String get u_love_spotube => 'Tiedämme että rakastat Spotubea'; + + @override + String get check_for_updates => 'Tarkista päivitykset'; + + @override + String get about_spotube => 'Tietoa Spotube:sta'; + + @override + String get blacklist => 'Mustalista'; + + @override + String get please_sponsor => 'Sponsoroi/Lahjoita, kiitos'; + + @override + String get spotube_description => + 'Spotube, kevyt, cross-platform, vapaa-kaikille spotify clientti'; + + @override + String get version => 'Versio'; + + @override + String get build_number => 'Rakennusnumero'; + + @override + String get founder => 'Perustaja'; + + @override + String get repository => 'Arkisto'; + + @override + String get bug_issues => 'Bugit+Ongelmat'; + + @override + String get made_with => 'Tehty ❤️ Bangladeshista 🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Lisenssi'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Älä huoli, tunnuksiasi ei talleteta tai jaeta kenenkään kanssa'; + + @override + String get know_how_to_login => 'Etkö tiedä miten tehdä tämä?'; + + @override + String get follow_step_by_step_guide => 'Seuraa askel askeleelta opasta'; + + @override + String cookie_name_cookie(Object name) { + return '$name Keksi'; + } + + @override + String get fill_in_all_fields => 'Täytä kaikki kentät'; + + @override + String get submit => 'Lähetä'; + + @override + String get exit => 'Poistu'; + + @override + String get previous => 'Edellinen'; + + @override + String get next => 'Seuraava'; + + @override + String get done => 'Tehty'; + + @override + String get step_1 => 'Vaihe 1'; + + @override + String get first_go_to => 'Ensiksi, mene'; + + @override + String get something_went_wrong => 'Jotain meni pieleen'; + + @override + String get piped_instance => 'Johdettu palvelinesiintymä'; + + @override + String get piped_description => + 'Johdettu palvelinesiintymä Kappale täsmäyksiin'; + + @override + String get piped_warning => + 'Jotkut niistä eivät toimi hyvin, käytä siis omalla vastuullasi'; + + @override + String get invidious_instance => 'Invidious-palvelinesiintymä'; + + @override + String get invidious_description => + 'Invidious-palvelinesiintymä raitojen yhteensovittamiseen'; + + @override + String get invidious_warning => + 'Jotkin esiintymät eivät välttämättä toimi hyvin. Käytä omalla vastuullasi'; + + @override + String get generate => 'Luo'; + + @override + String track_exists(Object track) { + return 'Kappale $track on jo olemassa!'; + } + + @override + String get replace_downloaded_tracks => 'Korvaa kaikki ladatut kappaleet'; + + @override + String get skip_download_tracks => 'Ohita ladattujen laulujen lataaminen'; + + @override + String get do_you_want_to_replace => + 'Haluatko korvata olemassa olevan kappaleen??'; + + @override + String get replace => 'Korvaa'; + + @override + String get skip => 'Ohita'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Valitse enintään $count $type'; + } + + @override + String get select_genres => 'Valitse Genret'; + + @override + String get add_genres => 'Lisää Genrejä'; + + @override + String get country => 'Maa'; + + @override + String get number_of_tracks_generate => 'Numero tuotettavia kappaleita'; + + @override + String get acousticness => 'Akustisuus'; + + @override + String get danceability => 'Tanssittavuus'; + + @override + String get energy => 'Energia'; + + @override + String get instrumentalness => 'Instrumentaalisuus'; + + @override + String get liveness => 'Elävyyttä'; + + @override + String get loudness => 'Äänekkyys'; + + @override + String get speechiness => 'Puheisuus'; + + @override + String get valence => 'Valenssi'; + + @override + String get popularity => 'Suosio'; + + @override + String get key => 'Sävellaji'; + + @override + String get duration => 'Pituus (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Tila'; + + @override + String get time_signature => 'Aikamerkki'; + + @override + String get short => 'Lyhyt'; + + @override + String get medium => 'Keskikokoinen'; + + @override + String get long => 'Pitkä'; + + @override + String get min => 'Minimi'; + + @override + String get max => 'Maximi'; + + @override + String get target => 'Kohde'; + + @override + String get moderate => 'Kohtalainen'; + + @override + String get deselect_all => 'Poista kaikki valinnat'; + + @override + String get select_all => 'Valitse kaikki'; + + @override + String get are_you_sure => 'Oletko varma?'; + + @override + String get generating_playlist => 'Luodaan mukautettua soittolistoa...'; + + @override + String selected_count_tracks(Object count) { + return 'Valittu $count kappaletta'; + } + + @override + String get download_warning => + 'Jos lataat kaikki laulut kerrällä olet selkeästi Piratoimassa ja aiheuttamassa vahinkoa musiikin luovaan yhteiskuntaan. Toivottavasti olet tietoinen tästä. Yritä aina kunnioittaa ja tukea Artistin kovaa työtä.'; + + @override + String get download_ip_ban_warning => + 'BTW, YouTube voi estää IP-Osoitteesi tavallista liiallisten latauspyyntöjen takia. IP-Osoitteen esto tarkoittaa sitä, ettet voi käyttää YouTubea (vaikka olisit kirjautunut) vähintään 2-3kk aikana kyseiseltä laitteelta. Spotube ei kanna yhtään vastuuta jos se tapahtuu.'; + + @override + String get by_clicking_accept_terms => + 'Painamalla \'hyväksy\' hyväksyt seuraaviin ehtoihin:'; + + @override + String get download_agreement_1 => + 'Tiedän että Piratoin musiikkia. Olen paha.'; + + @override + String get download_agreement_2 => + 'Tuen Artisteja silloin kun pystyn, ja teen tämän vain koska minulla ei ole rahaa ostaa heidän taidetta'; + + @override + String get download_agreement_3 => + 'Ymmärrän että minun YouTube voi estää IP-Osoitteeni ja en pidä Spotubea tai omistajiinsa/avustajia vastuullisena mistään omista teoistsani'; + + @override + String get decline => 'Hylkää'; + + @override + String get accept => 'Hyväksy'; + + @override + String get details => 'Yksityiskohdat'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanava'; + + @override + String get likes => 'Tykkäykset'; + + @override + String get dislikes => 'Epä-tykkäykset'; + + @override + String get views => 'Näyttökerrat'; + + @override + String get streamUrl => 'Suoratoiston URL'; + + @override + String get stop => 'Lopeta'; + + @override + String get sort_newest => 'Suodata uusimmista'; + + @override + String get sort_oldest => 'Suodata vanhimmista'; + + @override + String get sleep_timer => 'Uniajastin'; + + @override + String mins(Object minutes) { + return '$minutes Minuuttia'; + } + + @override + String hours(Object hours) { + return '$hours Tuntia'; + } + + @override + String hour(Object hours) { + return '$hours Tunti'; + } + + @override + String get custom_hours => 'Mukautetut tunnit'; + + @override + String get logs => 'Lokit'; + + @override + String get developers => 'Kehittäjät'; + + @override + String get not_logged_in => 'Et ole kirjautunut sisään.'; + + @override + String get search_mode => 'Hakutila'; + + @override + String get audio_source => 'Äänilähde'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Salaaminen epäonnistui'; + + @override + String get encryption_failed_warning => + 'Spotube käyttää salausta tallentaakseen tietosi, mutta epäonnistui, joten se palaa epäturvalliseen tallennukseen\nJos käytät Linuxia, varmista että sinulla on turvallisuuspalvelu (gnome-keyring, kde-wallet, keepassxc jne) asennettu'; + + @override + String get querying_info => 'Hankitaan tietoa...'; + + @override + String get piped_api_down => 'Johdettu palvelinesiintymä on alhaalla'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Johdettu palvelinesiintymä $pipedInstance on alhaalla.\n\nVaihda joko ilmeytymä tia vahda \'API tyyppi\' YouTuben viralliseen API\n\nKäynnistä sovellus uudestaan vaihdon jälkeen'; + } + + @override + String get you_are_offline => 'Et ole yhdistetty verkkoon'; + + @override + String get connection_restored => 'Verkkoyhteys palautettu'; + + @override + String get use_system_title_bar => 'Käytä järjestelmäpalkkia'; + + @override + String get crunching_results => 'Paloitellaan tuloksia...'; + + @override + String get search_to_get_results => 'Hae saadakseen tuloksia'; + + @override + String get use_amoled_mode => 'Pilkkopimeä tumma teema'; + + @override + String get pitch_dark_theme => 'AMOLED Tila'; + + @override + String get normalize_audio => 'Normalisoi audio'; + + @override + String get change_cover => 'Vaihda koveri'; + + @override + String get add_cover => 'Lisää koveri'; + + @override + String get restore_defaults => 'Palauta oletukset'; + + @override + String get download_music_format => 'Musiikin latausmuoto'; + + @override + String get streaming_music_format => 'Musiikin suoratoistomuoto'; + + @override + String get download_music_quality => 'Musiikin latauslaatu'; + + @override + String get streaming_music_quality => 'Musiikin suoratoistolaadun'; + + @override + String get login_with_lastfm => 'Kirjaudu sisään Last.fm:llä'; + + @override + String get connect => 'Yhdistä'; + + @override + String get disconnect_lastfm => 'Katkaise Last.fm'; + + @override + String get disconnect => 'Katkaise'; + + @override + String get username => 'Käyttäjänimi'; + + @override + String get password => 'Salasana'; + + @override + String get login => 'Kirjaudu'; + + @override + String get login_with_your_lastfm => 'Kirjaudu Last.fm käyttäjälläsi'; + + @override + String get scrobble_to_lastfm => 'Scrobble Last.fm:ään'; + + @override + String get go_to_album => 'Mene albumiin'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'Selaa kaikki'; + + @override + String get genres => 'Genret'; + + @override + String get explore_genres => 'Seikkaile genrejä'; + + @override + String get friends => 'Kaverit'; + + @override + String get no_lyrics_available => + 'Anteeksi, emme löytäneet lyriikoita tälle laululle'; + + @override + String get start_a_radio => 'Aloita Radio'; + + @override + String get how_to_start_radio => 'Kuinka haluat aloittaa radion?'; + + @override + String get replace_queue_question => + 'Haluatko korvata nykyisen jonon vai lisätä siihen?'; + + @override + String get endless_playback => 'Loputon toisto'; + + @override + String get delete_playlist => 'Poista soittolista'; + + @override + String get delete_playlist_confirmation => + 'Oletko varma että haluat poistaa tämän soittolistan?'; + + @override + String get local_tracks => 'Paikalliset kappaleet'; + + @override + String get local_tab => 'Paikallinen'; + + @override + String get song_link => 'Laulun linkki'; + + @override + String get skip_this_nonsense => 'Ohita tämä hölynpöly'; + + @override + String get freedom_of_music => '“Musiikin vapaus”'; + + @override + String get freedom_of_music_palm => '“Musiikin vapaus käsissäsi”'; + + @override + String get get_started => 'Aloitetaan'; + + @override + String get youtube_source_description => 'Suositeltu ja toimii parhaiten.'; + + @override + String get piped_source_description => + 'Tuntuuko vapaalta? Sama kuin YouTube mutta paljon vapautta'; + + @override + String get jiosaavn_source_description => 'Paras Etelä-Aasian alueelle.'; + + @override + String get invidious_source_description => + 'Samankaltainen kuin Piped, mutta korkeammalla saatavuudella'; + + @override + String highest_quality(Object quality) { + return 'Korkein laatu: $quality'; + } + + @override + String get select_audio_source => 'Valitse äänilähde'; + + @override + String get endless_playback_description => + 'Lisää automaattisesti uusia lauluja\njonon perään'; + + @override + String get choose_your_region => 'Valitse alueesi'; + + @override + String get choose_your_region_description => + 'Tämä auttaa Spotube näyttämään sinulle oikeaa sisältöä\nsijaintiasi varten.'; + + @override + String get choose_your_language => 'Valitse kielesi'; + + @override + String get help_project_grow => 'Auta tätä projektia kasvamaan'; + + @override + String get help_project_grow_description => + 'Spotube projekti minkä lähdekoodi on julkisesti saatavilla. Voit autta tätä projektia kasvamaan muutoksilla, ilmoittamalla bugeista, tai ehdottamalla uusia ominaisuuksia.'; + + @override + String get contribute_on_github => 'Auta GitHub:ssa'; + + @override + String get donate_on_open_collective => 'Lahjoita avoimessa kollektiivissa'; + + @override + String get browse_anonymously => 'Selaa anonyyminä'; + + @override + String get enable_connect => 'Ota käyttöön yhdistäminen'; + + @override + String get enable_connect_description => 'Ohjaa Spotubea toiselta laitteelta'; + + @override + String get devices => 'Laitteet'; + + @override + String get select => 'Valitse'; + + @override + String connect_client_alert(Object client) { + return '$client ohjaa sinua'; + } + + @override + String get this_device => 'Tämä laite'; + + @override + String get remote => 'Etä'; + + @override + String get stats => 'Tilastot'; + + @override + String and_n_more(Object count) { + return 'ja $count lisää'; + } + + @override + String get recently_played => 'Äskettäin soitetut'; + + @override + String get browse_more => 'Selaa lisää'; + + @override + String get no_title => 'Ei otsikkoa'; + + @override + String get not_playing => 'Ei soi'; + + @override + String get epic_failure => 'Epäonnistuminen!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Lisätty $tracks_length kappaletta jonoon'; + } + + @override + String get spotube_has_an_update => 'Spotubella on päivitys'; + + @override + String get download_now => 'Lataa nyt'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum on julkaistu'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version on julkaistu'; + } + + @override + String get read_the_latest => 'Lue viimeisimmät'; + + @override + String get release_notes => 'julkaisumuistiinpanot'; + + @override + String get pick_color_scheme => 'Valitse värimaailma'; + + @override + String get save => 'Tallenna'; + + @override + String get choose_the_device => 'Valitse laite:'; + + @override + String get multiple_device_connected => + 'Useita laitteita on kytketty.\nValitse laite, jossa haluat toiminnon suorittaa'; + + @override + String get nothing_found => 'Ei tuloksia'; + + @override + String get the_box_is_empty => 'Laatikko on tyhjä'; + + @override + String get top_artists => 'Suosituimmat artistit'; + + @override + String get top_albums => 'Suosituimmat albumit'; + + @override + String get this_week => 'Tällä viikolla'; + + @override + String get this_month => 'Tässä kuussa'; + + @override + String get last_6_months => 'Viimeiset 6 kuukautta'; + + @override + String get this_year => 'Tänä vuonna'; + + @override + String get last_2_years => 'Viimeiset 2 vuotta'; + + @override + String get all_time => 'Kaikki ajat'; + + @override + String powered_by_provider(Object providerName) { + return 'Tuottanut $providerName'; + } + + @override + String get email => 'Sähköposti'; + + @override + String get profile_followers => 'Seuraajat'; + + @override + String get birthday => 'Syntymäpäivä'; + + @override + String get subscription => 'Tilaus'; + + @override + String get not_born => 'Ei syntynyt'; + + @override + String get hacker => 'Hakkeri'; + + @override + String get profile => 'Profiili'; + + @override + String get no_name => 'Ei nimeä'; + + @override + String get edit => 'Muokkaa'; + + @override + String get user_profile => 'Käyttäjäprofiili'; + + @override + String count_plays(Object count) { + return '$count toistoa'; + } + + @override + String get streaming_fees_hypothetical => + 'Suoratoiston maksut (hypoteettinen)'; + + @override + String get minutes_listened => 'Kuunneltuja minuutteja'; + + @override + String get streamed_songs => 'Suoratoistettuja kappaleita'; + + @override + String count_streams(Object count) { + return '$count suoratoistoa'; + } + + @override + String get owned_by_you => 'Sinun omistama'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl kopioitu leikepöydälle'; + } + + @override + String get hipotetical_calculation => + '*Tämä on laskettu keskimääräisen musiikin suoratoistopalvelun 0,003–0,005 dollarin kappalekohtaisen maksun perusteella. Tämä on hypoteettinen laskelma, joka antaa käyttäjälle käsityksen siitä, kuinka paljon he olisivat maksaneet artisteille, jos he kuuntelisivat heidän kappaleitaan eri musiikin suoratoistopalveluissa.'; + + @override + String count_mins(Object minutes) { + return '$minutes min'; + } + + @override + String get summary_minutes => 'minuuttia'; + + @override + String get summary_listened_to_music => 'Kuunneltu musiikkia'; + + @override + String get summary_songs => 'kappaletta'; + + @override + String get summary_streamed_overall => 'Suoratoistettu yhteensä'; + + @override + String get summary_owed_to_artists => 'Maksettava artisteille\nTässä kuussa'; + + @override + String get summary_artists => 'artisti'; + + @override + String get summary_music_reached_you => 'Musiikki saavutti sinut'; + + @override + String get summary_full_albums => 'täydet albumit'; + + @override + String get summary_got_your_love => 'Sai rakkautesi'; + + @override + String get summary_playlists => 'soittolistat'; + + @override + String get summary_were_on_repeat => 'Olivat toistossa'; + + @override + String total_money(Object money) { + return 'Yhteensä $money'; + } + + @override + String get webview_not_found => 'Webview ei löydy'; + + @override + String get webview_not_found_description => + 'Laitteellasi ei ole asennettua Webview-ajonaikaa.\nJos se on asennettu, varmista, että se on environment PATH:ssa\n\nAsennuksen jälkeen käynnistä sovellus uudelleen'; + + @override + String get unsupported_platform => 'Ei tuettu alusta'; + + @override + String get cache_music => 'Musiikki välimuistissa'; + + @override + String get open => 'Avaa'; + + @override + String get cache_folder => 'Välimuistikansio'; + + @override + String get export => 'Vie'; + + @override + String get clear_cache => 'Tyhjennä välimuisti'; + + @override + String get clear_cache_confirmation => 'Haluatko tyhjentää välimuistin?'; + + @override + String get export_cache_files => 'Vie välimuistitiedostot'; + + @override + String found_n_files(Object count) { + return 'Löydettiin $count tiedostoa'; + } + + @override + String get export_cache_confirmation => 'Haluatko viedä nämä tiedostot'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Vietiin $filesExported/$files tiedostoa'; + } + + @override + String get undo => 'Peruuta'; + + @override + String get download_all => 'Lataa kaikki'; + + @override + String get add_all_to_playlist => 'Lisää kaikki soittolistalle'; + + @override + String get add_all_to_queue => 'Lisää kaikki jonoon'; + + @override + String get play_all_next => 'Toista kaikki seuraavaksi'; + + @override + String get pause => 'Pysäytä'; + + @override + String get view_all => 'Näytä kaikki'; + + @override + String get no_tracks_added_yet => + 'Näyttää siltä, että et ole lisännyt vielä mitään kappaleita.'; + + @override + String get no_tracks => 'Näyttää siltä, että täällä ei ole kappaleita.'; + + @override + String get no_tracks_listened_yet => + 'Näyttää siltä, että et ole kuunnellut mitään vielä.'; + + @override + String get not_following_artists => 'Et seuraa yhtään artistia.'; + + @override + String get no_favorite_albums_yet => + 'Näyttää siltä, että et ole lisännyt yhtään albumia suosikkeihisi.'; + + @override + String get no_logs_found => 'Ei lokitietoja löydetty'; + + @override + String get youtube_engine => 'YouTube-moottori'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine ei ole asennettu'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine ei ole asennettu järjestelmääsi.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Varmista, että se on saatavilla PATH-muuttujassa tai\nasetetaan $engine suoritettavan tiedoston absoluuttinen polku alla.'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/unix-tyyppisissä käyttöjärjestelmissä polun asettaminen .zshrc/.bashrc/.bash_profile jne. ei toimi.\nSinun täytyy asettaa polku shellin asetustiedostoon.'; + + @override + String get download => 'Lataa'; + + @override + String get file_not_found => 'Tiedostoa ei löydy'; + + @override + String get custom => 'Mukautettu'; + + @override + String get add_custom_url => 'Lisää mukautettu URL'; + + @override + String get edit_port => 'Muokkaa porttia'; + + @override + String get port_helper_msg => + 'Oletusarvo on -1, mikä tarkoittaa satunnaista numeroa. Jos sinulla on palomuuri määritetty, tämän asettamista suositellaan.'; + + @override + String connect_request(Object client) { + return 'Salli $client yhdistää?'; + } + + @override + String get connection_request_denied => + 'Yhteys evätty. Käyttäjä eväsi pääsyn.'; + + @override + String get an_error_occurred => 'Tapahtui virhe'; + + @override + String get copy_to_clipboard => 'Kopioi leikepöydälle'; + + @override + String get view_logs => 'Näytä lokit'; + + @override + String get retry => 'Yritä uudelleen'; + + @override + String get no_default_metadata_provider_selected => + 'Et ole asettanut oletusmetatietojen tarjoajaa'; + + @override + String get manage_metadata_providers => 'Hallinnoi metatietojen tarjoajia'; + + @override + String get open_link_in_browser => 'Avaa linkki selaimessa?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Haluatko avata seuraavan linkin'; + + @override + String get unsafe_url_warning => + 'Linkkien avaaminen epäluotettavista lähteistä voi olla vaarallista. Ole varovainen!\nVoit myös kopioida linkin leikepöydälle.'; + + @override + String get copy_link => 'Kopioi linkki'; + + @override + String get building_your_timeline => + 'Rakennetaan aikajanaasi kuuntelujesi perusteella...'; + + @override + String get official => 'Virallinen'; + + @override + String author_name(Object author) { + return 'Tekijä: $author'; + } + + @override + String get third_party => 'Kolmannen osapuolen'; + + @override + String get plugin_requires_authentication => 'Lisäosa vaatii todentamisen'; + + @override + String get update_available => 'Päivitys saatavilla'; + + @override + String get supports_scrobbling => 'Tukee scrobblingia'; + + @override + String get plugin_scrobbling_info => + 'Tämä lisäosa scrobblaa musiikkisi luodakseen kuunteluhistoriasi.'; + + @override + String get default_metadata_source => 'Oletusarvoinen metatietolähde'; + + @override + String get set_default_metadata_source => 'Aseta oletusmetatietolähde'; + + @override + String get default_audio_source => 'Oletusarvoinen äänilähde'; + + @override + String get set_default_audio_source => 'Aseta oletusäänilähde'; + + @override + String get set_default => 'Aseta oletukseksi'; + + @override + String get support => 'Tuki'; + + @override + String get support_plugin_development => 'Tue lisäosan kehitystä'; + + @override + String can_access_name_api(Object name) { + return '- Voi käyttää **$name** APIa'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Haluatko asentaa tämän lisäosan?'; + + @override + String get third_party_plugin_warning => + 'Tämä lisäosa on kolmannen osapuolen arkistosta. Varmista, että luotat lähteeseen ennen asennusta.'; + + @override + String get author => 'Tekijä'; + + @override + String get this_plugin_can_do_following => 'Tämä lisäosa voi tehdä seuraavaa'; + + @override + String get install => 'Asenna'; + + @override + String get install_a_metadata_provider => 'Asenna metatietojen tarjoaja'; + + @override + String get no_tracks_playing => 'Ei kappaletta toistossa tällä hetkellä'; + + @override + String get synced_lyrics_not_available => + 'Synkronoidut sanoitukset eivät ole saatavilla tälle kappaleelle. Käytä sen sijaan'; + + @override + String get plain_lyrics => 'Pelkät sanoitukset'; + + @override + String get tab_instead => 'välilehteä.'; + + @override + String get disclaimer => 'Vastuuvapauslauseke'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube-tiimi ei ota mitään vastuuta (mukaan lukien oikeudellinen) mistään \"kolmannen osapuolen\" lisäosista.\nKäytä niitä omalla vastuullasi. Ilmoita kaikista virheistä/ongelmista lisäosan arkistoon.\n\nJos jokin \"kolmannen osapuolen\" lisäosa rikkoo jonkin palvelun/oikeushenkilön käyttöehtoja/DMCA:ta, pyydä \"kolmannen osapuolen\" lisäosan tekijää tai isännöintialustaa, esim. GitHubia/Codebergiä, ryhtymään toimiin. Yllä luetellut (\"kolmannen osapuolen\" merkityt) ovat kaikki julkisia/yhteisön ylläpitämiä lisäosia. Emme kuratoi niitä, joten emme voi ryhtyä niihin toimiin.\n\n'; + + @override + String get input_does_not_match_format => 'Syöte ei vastaa vaadittua muotoa'; + + @override + String get plugins => 'Laajennukset'; + + @override + String get paste_plugin_download_url => + 'Liitä lataus-URL-osoite tai GitHub/Codeberg-arkiston URL-osoite tai suora linkki .smplug-tiedostoon'; + + @override + String get download_and_install_plugin_from_url => + 'Lataa ja asenna lisäosa URL-osoitteesta'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Lisäosan lisääminen epäonnistui: $error'; + } + + @override + String get upload_plugin_from_file => 'Lataa lisäosa tiedostosta'; + + @override + String get installed => 'Asennettu'; + + @override + String get available_plugins => 'Saatavilla olevat lisäosat'; + + @override + String get configure_plugins => + 'Määritä omat metatietojen tarjoaja- ja äänilähdelaajennukset'; + + @override + String get audio_scrobblers => 'Äänen scrobblerit'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Lähde: '; + + @override + String get uncompressed => 'Pakkaamaton'; + + @override + String get dab_music_source_description => + 'Audiofiileille. Tarjoaa korkealaatuisia/häviöttömiä äänivirtoja. Tarkka ISRC-pohjainen kappaleiden tunnistus.'; +} diff --git a/lib/l10n/generated/app_localizations_fr.dart b/lib/l10n/generated/app_localizations_fr.dart new file mode 100644 index 00000000..3637391b --- /dev/null +++ b/lib/l10n/generated/app_localizations_fr.dart @@ -0,0 +1,1584 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for French (`fr`). +class AppLocalizationsFr extends AppLocalizations { + AppLocalizationsFr([String locale = 'fr']) : super(locale); + + @override + String get guest => 'Invité'; + + @override + String get browse => 'Explorer'; + + @override + String get search => 'Rechercher'; + + @override + String get library => 'Bibliothèque'; + + @override + String get lyrics => 'Paroles'; + + @override + String get settings => 'Paramètres'; + + @override + String get genre_categories_filter => + 'Filtrer les catégories ou les genres...'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Personnalisé'; + + @override + String get featured => 'En vedette'; + + @override + String get new_releases => 'Nouvelles sorties'; + + @override + String get songs => 'Chansons'; + + @override + String playing_track(Object track) { + return 'Lecture de $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Cela effacera la file d\'attente actuelle. $track_length pistes seront supprimées\nVoulez-vous continuer?'; + } + + @override + String get load_more => 'Charger plus'; + + @override + String get playlists => 'Listes de lecture'; + + @override + String get artists => 'Artistes'; + + @override + String get albums => 'Albums'; + + @override + String get tracks => 'Pistes'; + + @override + String get downloads => 'Téléchargements'; + + @override + String get filter_playlists => 'Filtrer vos listes de lecture...'; + + @override + String get liked_tracks => 'Pistes aimées'; + + @override + String get liked_tracks_description => 'Toutes vos pistes aimées'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Créer une liste de lecture'; + + @override + String get update_playlist => 'Mettre à jour la playlist'; + + @override + String get create => 'Créer'; + + @override + String get cancel => 'Annuler'; + + @override + String get update => 'Mettre à jour'; + + @override + String get playlist_name => 'Nom de la liste de lecture'; + + @override + String get name_of_playlist => 'Nom de la liste de lecture'; + + @override + String get description => 'Description'; + + @override + String get public => 'Public'; + + @override + String get collaborative => 'Collaborative'; + + @override + String get search_local_tracks => 'Rechercher des pistes locales...'; + + @override + String get play => 'Lecture'; + + @override + String get delete => 'Supprimer'; + + @override + String get none => 'Aucun'; + + @override + String get sort_a_z => 'Trier par ordre alphabétique'; + + @override + String get sort_z_a => 'Trier par ordre alphabétique inverse'; + + @override + String get sort_artist => 'Trier par artiste'; + + @override + String get sort_album => 'Trier par album'; + + @override + String get sort_duration => 'Trier par durée'; + + @override + String get sort_tracks => 'Trier les pistes'; + + @override + String currently_downloading(Object tracks_length) { + return 'Téléchargement en cours ($tracks_length)'; + } + + @override + String get cancel_all => 'Tout annuler'; + + @override + String get filter_artist => 'Filtrer les artistes...'; + + @override + String followers(Object followers) { + return '$followers abonnés'; + } + + @override + String get add_artist_to_blacklist => 'Ajouter l\'artiste à la liste noire'; + + @override + String get top_tracks => 'Meilleures pistes'; + + @override + String get fans_also_like => 'Les fans aiment aussi'; + + @override + String get loading => 'Chargement...'; + + @override + String get artist => 'Artiste'; + + @override + String get blacklisted => 'Liste noire'; + + @override + String get following => 'Abonné'; + + @override + String get follow => 'S\'abonner'; + + @override + String get artist_url_copied => + 'URL de l\'artiste copiée dans le presse-papiers'; + + @override + String added_to_queue(Object tracks) { + return '$tracks pistes ajoutées à la file d\'attente'; + } + + @override + String get filter_albums => 'Filtrer les albums...'; + + @override + String get synced => 'Synchronisé'; + + @override + String get plain => 'Simple'; + + @override + String get shuffle => 'Lecture aléatoire'; + + @override + String get search_tracks => 'Rechercher des pistes...'; + + @override + String get released => 'Sorti'; + + @override + String error(Object error) { + return 'Erreur $error'; + } + + @override + String get title => 'Titre'; + + @override + String get time => 'Durée'; + + @override + String get more_actions => 'Plus d\'actions'; + + @override + String download_count(Object count) { + return 'Téléchargement ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Ajouter ($count) à la liste de lecture'; + } + + @override + String add_count_to_queue(Object count) { + return 'Ajouter ($count) à la file d\'attente'; + } + + @override + String play_count_next(Object count) { + return 'Lire ($count) ensuite'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return '$data copié dans le presse-papiers'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Ajouter $track aux listes de lecture suivantes'; + } + + @override + String get add => 'Ajouter'; + + @override + String added_track_to_queue(Object track) { + return '$track ajouté à la file d\'attente'; + } + + @override + String get add_to_queue => 'Ajouter à la file d\'attente'; + + @override + String track_will_play_next(Object track) { + return '$track sera joué ensuite'; + } + + @override + String get play_next => 'Lire ensuite'; + + @override + String removed_track_from_queue(Object track) { + return '$track retiré de la file d\'attente'; + } + + @override + String get remove_from_queue => 'Retirer de la file d\'attente'; + + @override + String get remove_from_favorites => 'Retirer des favoris'; + + @override + String get save_as_favorite => 'Enregistrer comme favori'; + + @override + String get add_to_playlist => 'Ajouter à la liste de lecture'; + + @override + String get remove_from_playlist => 'Retirer de la liste de lecture'; + + @override + String get add_to_blacklist => 'Ajouter à la liste noire'; + + @override + String get remove_from_blacklist => 'Retirer de la liste noire'; + + @override + String get share => 'Partager'; + + @override + String get mini_player => 'Lecteur mini'; + + @override + String get slide_to_seek => 'Faites glisser pour avancer ou reculer'; + + @override + String get shuffle_playlist => 'Lecture aléatoire de la liste de lecture'; + + @override + String get unshuffle_playlist => + 'Annuler la lecture aléatoire de la liste de lecture'; + + @override + String get previous_track => 'Piste précédente'; + + @override + String get next_track => 'Piste suivante'; + + @override + String get pause_playback => 'Mettre en pause la lecture'; + + @override + String get resume_playback => 'Reprendre la lecture'; + + @override + String get loop_track => 'Lecture en boucle de la piste'; + + @override + String get no_loop => 'Pas de boucle'; + + @override + String get repeat_playlist => 'Répéter la liste de lecture'; + + @override + String get queue => 'File d\'attente'; + + @override + String get alternative_track_sources => 'Sources alternatives de pistes'; + + @override + String get download_track => 'Télécharger la piste'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks pistes dans la file d\'attente'; + } + + @override + String get clear_all => 'Tout effacer'; + + @override + String get show_hide_ui_on_hover => + 'Afficher/Masquer l\'interface utilisateur au survol'; + + @override + String get always_on_top => 'Toujours au-dessus'; + + @override + String get exit_mini_player => 'Quitter le lecteur mini'; + + @override + String get download_location => 'Emplacement de téléchargement'; + + @override + String get local_library => 'Bibliothèque locale'; + + @override + String get add_library_location => 'Ajouter à la bibliothèque'; + + @override + String get remove_library_location => 'Retirer de la bibliothèque'; + + @override + String get account => 'Compte'; + + @override + String get logout => 'Se déconnecter'; + + @override + String get logout_of_this_account => 'Se déconnecter de ce compte'; + + @override + String get language_region => 'Langue et région'; + + @override + String get language => 'Langue'; + + @override + String get system_default => 'Paramètres par défaut du système'; + + @override + String get market_place_region => 'Région du marché'; + + @override + String get recommendation_country => 'Pays de recommandation'; + + @override + String get appearance => 'Apparence'; + + @override + String get layout_mode => 'Mode de mise en page'; + + @override + String get override_layout_settings => + 'Remplacer les paramètres de mise en page adaptative'; + + @override + String get adaptive => 'Adaptatif'; + + @override + String get compact => 'Compact'; + + @override + String get extended => 'Étendu'; + + @override + String get theme => 'Thème'; + + @override + String get dark => 'Sombre'; + + @override + String get light => 'Clair'; + + @override + String get system => 'Système'; + + @override + String get accent_color => 'Couleur d\'accentuation'; + + @override + String get sync_album_color => 'Synchroniser la couleur de l\'album'; + + @override + String get sync_album_color_description => + 'Utilise la couleur dominante de l\'art de l\'album comme couleur d\'accentuation'; + + @override + String get playback => 'Lecture'; + + @override + String get audio_quality => 'Qualité audio'; + + @override + String get high => 'Haute'; + + @override + String get low => 'Basse'; + + @override + String get pre_download_play => 'Pré-télécharger et lire'; + + @override + String get pre_download_play_description => + 'Au lieu de diffuser de l\'audio, téléchargez les octets et lisez-les à la place (recommandé pour les utilisateurs à bande passante élevée)'; + + @override + String get skip_non_music => + 'Ignorer les segments non musicaux (SponsorBlock)'; + + @override + String get blacklist_description => 'Pistes et artistes en liste noire'; + + @override + String get wait_for_download_to_finish => + 'Veuillez attendre la fin du téléchargement en cours'; + + @override + String get desktop => 'Bureau'; + + @override + String get close_behavior => 'Comportement de fermeture'; + + @override + String get close => 'Fermer'; + + @override + String get minimize_to_tray => 'Réduire dans la zone de notification'; + + @override + String get show_tray_icon => 'Afficher l\'icône de la zone de notification'; + + @override + String get about => 'À propos'; + + @override + String get u_love_spotube => 'Nous savons que vous aimez Spotube'; + + @override + String get check_for_updates => 'Vérifier les mises à jour'; + + @override + String get about_spotube => 'À propos de Spotube'; + + @override + String get blacklist => 'Liste noire'; + + @override + String get please_sponsor => 'S\'il vous plaît Sponsoriser/Donner'; + + @override + String get spotube_description => + 'Spotube, un client Spotify léger, multiplateforme et gratuit pour tous'; + + @override + String get version => 'Version'; + + @override + String get build_number => 'Numéro de version'; + + @override + String get founder => 'Fondateur'; + + @override + String get repository => 'Dépôt'; + + @override + String get bug_issues => 'Bugs + Problèmes'; + + @override + String get made_with => 'Fabriqué avec ❤️ au Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licence'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Ne vous inquiétez pas, vos identifiants ne seront ni collectés ni partagés avec qui que ce soit'; + + @override + String get know_how_to_login => 'Vous ne savez pas comment faire?'; + + @override + String get follow_step_by_step_guide => 'Suivez le guide étape par étape'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Veuillez remplir tous les champs'; + + @override + String get submit => 'Soumettre'; + + @override + String get exit => 'Quitter'; + + @override + String get previous => 'Précédent'; + + @override + String get next => 'Suivant'; + + @override + String get done => 'Terminé'; + + @override + String get step_1 => 'Étape 1'; + + @override + String get first_go_to => 'Tout d\'abord, allez sur'; + + @override + String get something_went_wrong => 'Quelque chose s\'est mal passé'; + + @override + String get piped_instance => 'Instance pipée'; + + @override + String get piped_description => + 'L\'instance de serveur Piped à utiliser pour la correspondance des pistes'; + + @override + String get piped_warning => + 'Certaines d\'entre elles peuvent ne pas fonctionner correctement. Alors utilisez à vos risques et périls'; + + @override + String get invidious_instance => 'Instance de serveur Invidious'; + + @override + String get invidious_description => + 'L\'instance de serveur Invidious à utiliser pour la correspondance de pistes'; + + @override + String get invidious_warning => + 'Certaines instances pourraient ne pas bien fonctionner. À utiliser à vos risques et périls'; + + @override + String get generate => 'Générer'; + + @override + String track_exists(Object track) { + return 'La piste $track existe déjà'; + } + + @override + String get replace_downloaded_tracks => + 'Remplacer toutes les pistes téléchargées'; + + @override + String get skip_download_tracks => + 'Ignorer le téléchargement de toutes les pistes téléchargées'; + + @override + String get do_you_want_to_replace => + 'Voulez-vous remplacer la piste existante ?'; + + @override + String get replace => 'Remplacer'; + + @override + String get skip => 'Passer'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Sélectionnez jusqu\'à $count $type'; + } + + @override + String get select_genres => 'Sélectionner les genres'; + + @override + String get add_genres => 'Ajouter des genres'; + + @override + String get country => 'Pays'; + + @override + String get number_of_tracks_generate => 'Nombre de pistes à générer'; + + @override + String get acousticness => 'Acoustique'; + + @override + String get danceability => 'Dansabilité'; + + @override + String get energy => 'Énergie'; + + @override + String get instrumentalness => 'Instrumentalité'; + + @override + String get liveness => 'Interprétation en direct'; + + @override + String get loudness => 'Sonorité'; + + @override + String get speechiness => 'Parlé'; + + @override + String get valence => 'Valeur émotionnelle'; + + @override + String get popularity => 'Popularité'; + + @override + String get key => 'Clé'; + + @override + String get duration => 'Durée (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'Signature rythmique'; + + @override + String get short => 'Court'; + + @override + String get medium => 'Moyen'; + + @override + String get long => 'Long'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Cible'; + + @override + String get moderate => 'Modéré'; + + @override + String get deselect_all => 'Tout désélectionner'; + + @override + String get select_all => 'Tout sélectionner'; + + @override + String get are_you_sure => 'Êtes-vous sûr(e) ?'; + + @override + String get generating_playlist => + 'Génération de votre playlist personnalisée en cours...'; + + @override + String selected_count_tracks(Object count) { + return '$count pistes sélectionnées'; + } + + @override + String get download_warning => + 'Si vous téléchargez toutes les pistes en vrac, vous violez clairement les droits d\'auteur de la musique et vous causez des dommages à la société créative de la musique. J\'espère que vous en êtes conscient. Essayez toujours de respecter et de soutenir le travail acharné des artistes.'; + + @override + String get download_ip_ban_warning => + 'Au fait, votre adresse IP peut être bloquée sur YouTube en raison d\'une demande excessive de téléchargements par rapport à la normale. Le blocage de l\'IP signifie que vous ne pourrez pas utiliser YouTube (même si vous êtes connecté) pendant au moins 2 à 3 mois à partir de cet appareil IP. Et Spotube ne peut être tenu responsable si cela se produit.'; + + @override + String get by_clicking_accept_terms => + 'En cliquant sur \'accepter\', vous acceptez les conditions suivantes :'; + + @override + String get download_agreement_1 => + 'Je sais que je pirate de la musique. Je suis méchant(e).'; + + @override + String get download_agreement_2 => + 'Je soutiendrai l\'artiste autant que possible et je ne fais cela que parce que je n\'ai pas d\'argent pour acheter leur art.'; + + @override + String get download_agreement_3 => + 'Je suis parfaitement conscient(e) que mon adresse IP peut être bloquée sur YouTube et je ne tiens pas Spotube ni ses propriétaires/contributeurs responsables de tout accident causé par mon action actuelle.'; + + @override + String get decline => 'Refuser'; + + @override + String get accept => 'Accepter'; + + @override + String get details => 'Détails'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Chaîne'; + + @override + String get likes => 'J\'aime'; + + @override + String get dislikes => 'Je n\'aime pas'; + + @override + String get views => 'Vues'; + + @override + String get streamUrl => 'URL de diffusion'; + + @override + String get stop => 'Arrêter'; + + @override + String get sort_newest => 'Trier par les plus récents'; + + @override + String get sort_oldest => 'Trier par les plus anciens'; + + @override + String get sleep_timer => 'Minuteur de veille'; + + @override + String mins(Object minutes) { + return '$minutes minutes'; + } + + @override + String hours(Object hours) { + return '$hours heures'; + } + + @override + String hour(Object hours) { + return '$hours heure'; + } + + @override + String get custom_hours => 'Heures personnalisées'; + + @override + String get logs => 'Journaux'; + + @override + String get developers => 'Développeurs'; + + @override + String get not_logged_in => 'Vous n\'êtes pas connecté(e)'; + + @override + String get search_mode => 'Mode de recherche'; + + @override + String get audio_source => 'Source audio'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => 'Échec de la cryptage'; + + @override + String get encryption_failed_warning => + 'Spotube utilise le cryptage pour stocker vos données en toute sécurité. Mais cela a échoué. Il basculera donc vers un stockage non sécurisé\nSi vous utilisez Linux, assurez-vous d\'avoir installé des services secrets tels que gnome-keyring, kde-wallet et keepassxc'; + + @override + String get querying_info => 'Interrogation des info...'; + + @override + String get piped_api_down => 'L\'API Piped est hors service'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'L\'instance Piped $pipedInstance est actuellement indisponible\n\nChangez soit l\'instance, soit le \'Type d\'API\' pour utiliser l\'API officielle de YouTube\n\nN\'oubliez pas de redémarrer l\'application après la modification'; + } + + @override + String get you_are_offline => 'Vous êtes actuellement hors ligne'; + + @override + String get connection_restored => 'Votre connexion internet a été rétablie'; + + @override + String get use_system_title_bar => 'Utiliser la barre de titre système'; + + @override + String get crunching_results => 'Traitement des résultats...'; + + @override + String get search_to_get_results => 'Recherche pour obtenir des résultats'; + + @override + String get use_amoled_mode => 'Utiliser le mode AMOLED'; + + @override + String get pitch_dark_theme => 'Thème Dart noir intense'; + + @override + String get normalize_audio => 'Normaliser l\'audio'; + + @override + String get change_cover => 'Changer de couverture'; + + @override + String get add_cover => 'Ajouter une couverture'; + + @override + String get restore_defaults => 'Restaurer les valeurs par défaut'; + + @override + String get download_music_format => 'Format de téléchargement de musique'; + + @override + String get streaming_music_format => 'Format de streaming de musique'; + + @override + String get download_music_quality => 'Qualité de téléchargement de musique'; + + @override + String get streaming_music_quality => 'Qualité de streaming de musique'; + + @override + String get login_with_lastfm => 'Se connecter avec Last.fm'; + + @override + String get connect => 'Connecter'; + + @override + String get disconnect_lastfm => 'Déconnecter de Last.fm'; + + @override + String get disconnect => 'Déconnecter'; + + @override + String get username => 'Nom d\'utilisateur'; + + @override + String get password => 'Mot de passe'; + + @override + String get login => 'Se connecter'; + + @override + String get login_with_your_lastfm => 'Se connecter avec votre compte Last.fm'; + + @override + String get scrobble_to_lastfm => 'Scrobble à Last.fm'; + + @override + String get go_to_album => 'Aller à l\'album'; + + @override + String get discord_rich_presence => 'Présence riche de Discord'; + + @override + String get browse_all => 'Parcourir tout'; + + @override + String get genres => 'Genres'; + + @override + String get explore_genres => 'Explorer les genres'; + + @override + String get friends => 'Amis'; + + @override + String get no_lyrics_available => + 'Désolé, impossible de trouver les paroles de cette piste'; + + @override + String get start_a_radio => 'Démarrer une radio'; + + @override + String get how_to_start_radio => 'Comment voulez-vous démarrer la radio ?'; + + @override + String get replace_queue_question => + 'Voulez-vous remplacer la file d\'attente actuelle ou y ajouter ?'; + + @override + String get endless_playback => 'Lecture sans fin'; + + @override + String get delete_playlist => 'Supprimer la playlist'; + + @override + String get delete_playlist_confirmation => + 'Êtes-vous sûr de vouloir supprimer cette playlist ?'; + + @override + String get local_tracks => 'Titres locaux'; + + @override + String get local_tab => 'Local'; + + @override + String get song_link => 'Lien de la chanson'; + + @override + String get skip_this_nonsense => 'Passer cette absurdité'; + + @override + String get freedom_of_music => '“Liberté de la musique”'; + + @override + String get freedom_of_music_palm => + '“Liberté de la musique dans la paume de votre main”'; + + @override + String get get_started => 'Commençons'; + + @override + String get youtube_source_description => 'Recommandé et fonctionne mieux.'; + + @override + String get piped_source_description => + 'Vous vous sentez libre ? Comme YouTube mais beaucoup plus gratuit.'; + + @override + String get jiosaavn_source_description => + 'Le meilleur pour la région d\'Asie du Sud.'; + + @override + String get invidious_source_description => + 'Similaire à Piped mais avec une meilleure disponibilité'; + + @override + String highest_quality(Object quality) { + return 'Meilleure qualité : $quality'; + } + + @override + String get select_audio_source => 'Sélectionner la source audio'; + + @override + String get endless_playback_description => + 'Ajouter automatiquement de nouvelles chansons à la fin de la file d\'attente'; + + @override + String get choose_your_region => 'Choisissez votre région'; + + @override + String get choose_your_region_description => + 'Cela aidera Spotube à vous montrer le bon contenu pour votre emplacement.'; + + @override + String get choose_your_language => 'Choisissez votre langue'; + + @override + String get help_project_grow => 'Aidez ce projet à grandir'; + + @override + String get help_project_grow_description => + 'Spotube est un projet open-source. Vous pouvez aider ce projet à grandir en contribuant au projet, en signalant des bugs ou en suggérant de nouvelles fonctionnalités.'; + + @override + String get contribute_on_github => 'Contribuer sur GitHub'; + + @override + String get donate_on_open_collective => 'Faire un don sur Open Collective'; + + @override + String get browse_anonymously => 'Naviguer anonymement'; + + @override + String get enable_connect => 'Activer la connexion'; + + @override + String get enable_connect_description => + 'Contrôlez Spotube depuis d\'autres appareils'; + + @override + String get devices => 'Appareils'; + + @override + String get select => 'Sélectionner'; + + @override + String connect_client_alert(Object client) { + return 'Vous êtes contrôlé par $client'; + } + + @override + String get this_device => 'Cet appareil'; + + @override + String get remote => 'À distance'; + + @override + String get stats => 'Statistiques'; + + @override + String and_n_more(Object count) { + return 'et $count de plus'; + } + + @override + String get recently_played => 'Récemment joué'; + + @override + String get browse_more => 'Parcourir plus'; + + @override + String get no_title => 'Sans titre'; + + @override + String get not_playing => 'Non joué'; + + @override + String get epic_failure => 'Échec épique!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length morceaux ajoutés à la file d\'attente'; + } + + @override + String get spotube_has_an_update => 'Spotube a une mise à jour'; + + @override + String get download_now => 'Télécharger maintenant'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum a été publié'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version a été publié'; + } + + @override + String get read_the_latest => 'Lisez les dernières '; + + @override + String get release_notes => 'notes de version'; + + @override + String get pick_color_scheme => 'Choisissez le schéma de couleurs'; + + @override + String get save => 'Sauvegarder'; + + @override + String get choose_the_device => 'Choisissez l\'appareil:'; + + @override + String get multiple_device_connected => + 'Plusieurs appareils sont connectés.\nChoisissez l\'appareil sur lequel vous souhaitez effectuer cette action'; + + @override + String get nothing_found => 'Rien trouvé'; + + @override + String get the_box_is_empty => 'La boîte est vide'; + + @override + String get top_artists => 'Meilleurs artistes'; + + @override + String get top_albums => 'Meilleurs albums'; + + @override + String get this_week => 'Cette semaine'; + + @override + String get this_month => 'Ce mois-ci'; + + @override + String get last_6_months => 'Les 6 derniers mois'; + + @override + String get this_year => 'Cette année'; + + @override + String get last_2_years => 'Les 2 dernières années'; + + @override + String get all_time => 'De tous les temps'; + + @override + String powered_by_provider(Object providerName) { + return 'Propulsé par $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Abonnés'; + + @override + String get birthday => 'Anniversaire'; + + @override + String get subscription => 'Abonnement'; + + @override + String get not_born => 'Non né'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profil'; + + @override + String get no_name => 'Sans nom'; + + @override + String get edit => 'Modifier'; + + @override + String get user_profile => 'Profil utilisateur'; + + @override + String count_plays(Object count) { + return '$count lectures'; + } + + @override + String get streaming_fees_hypothetical => + 'Frais de streaming (hypothétiques)'; + + @override + String get minutes_listened => 'Minutes écoutées'; + + @override + String get streamed_songs => 'Morceaux diffusés'; + + @override + String count_streams(Object count) { + return '$count streams'; + } + + @override + String get owned_by_you => 'Possédé par vous'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl copié dans le presse-papier'; + } + + @override + String get hipotetical_calculation => + '*Ce calcul est basé sur le paiement moyen par lecture des plateformes de streaming musical en ligne, de 0,003 \$ à 0,005 \$. Il s\'agit d\'un calcul hypothétique pour donner à l\'utilisateur un aperçu de ce qu\'il aurait payé aux artistes s\'il écoutait leur chanson sur différentes plateformes de streaming musical.'; + + @override + String count_mins(Object minutes) { + return '$minutes minutes'; + } + + @override + String get summary_minutes => 'minutes'; + + @override + String get summary_listened_to_music => 'A écouté de la musique'; + + @override + String get summary_songs => 'morceaux'; + + @override + String get summary_streamed_overall => 'Diffusé en général'; + + @override + String get summary_owed_to_artists => 'Dû aux artistes\nCe mois-ci'; + + @override + String get summary_artists => 'artistes'; + + @override + String get summary_music_reached_you => 'La musique vous a atteint'; + + @override + String get summary_full_albums => 'albums complets'; + + @override + String get summary_got_your_love => 'A obtenu votre amour'; + + @override + String get summary_playlists => 'playlists'; + + @override + String get summary_were_on_repeat => 'Était en répétition'; + + @override + String total_money(Object money) { + return 'Total $money'; + } + + @override + String get webview_not_found => 'Webview non trouvé'; + + @override + String get webview_not_found_description => + 'Aucun environnement d\'exécution Webview installé sur votre appareil.\nSi c\'est installé, assurez-vous qu\'il soit dans le environment PATH\n\nAprès l\'installation, redémarrez l\'application'; + + @override + String get unsupported_platform => 'Plateforme non prise en charge'; + + @override + String get cache_music => 'Mettre la musique en cache'; + + @override + String get open => 'Ouvrir'; + + @override + String get cache_folder => 'Dossier du cache'; + + @override + String get export => 'Exporter'; + + @override + String get clear_cache => 'Effacer le cache'; + + @override + String get clear_cache_confirmation => 'Voulez-vous effacer le cache ?'; + + @override + String get export_cache_files => 'Exporter les fichiers en cache'; + + @override + String found_n_files(Object count) { + return '$count fichiers trouvés'; + } + + @override + String get export_cache_confirmation => + 'Voulez-vous exporter ces fichiers vers'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported fichiers exportés sur $files'; + } + + @override + String get undo => 'Annuler'; + + @override + String get download_all => 'Télécharger tout'; + + @override + String get add_all_to_playlist => 'Ajouter tout à la playlist'; + + @override + String get add_all_to_queue => 'Ajouter tout à la file d\'attente'; + + @override + String get play_all_next => 'Lire tout suivant'; + + @override + String get pause => 'Pause'; + + @override + String get view_all => 'Voir tout'; + + @override + String get no_tracks_added_yet => + 'Il semble que vous n\'avez encore ajouté aucun morceau.'; + + @override + String get no_tracks => 'Il semble qu\'il n\'y ait pas de morceaux ici.'; + + @override + String get no_tracks_listened_yet => + 'Il semble que vous n\'avez encore rien écouté.'; + + @override + String get not_following_artists => 'Vous ne suivez aucun artiste.'; + + @override + String get no_favorite_albums_yet => + 'Il semble que vous n\'ayez encore ajouté aucun album à vos favoris.'; + + @override + String get no_logs_found => 'Aucun log trouvé'; + + @override + String get youtube_engine => 'Moteur YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine n\'est pas installé'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine n\'est pas installé sur votre système.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Assurez-vous qu\'il est disponible dans la variable PATH ou\nfixez le chemin absolu du fichier exécutable $engine ci-dessous.'; + } + + @override + String get youtube_engine_unix_issue_message => + 'Dans macOS/Linux/les systèmes d\'exploitation similaires à Unix, définir le chemin dans .zshrc/.bashrc/.bash_profile etc. ne fonctionnera pas.\nVous devez définir le chemin dans le fichier de configuration du shell.'; + + @override + String get download => 'Télécharger'; + + @override + String get file_not_found => 'Fichier non trouvé'; + + @override + String get custom => 'Personnalisé'; + + @override + String get add_custom_url => 'Ajouter une URL personnalisée'; + + @override + String get edit_port => 'Modifier le port'; + + @override + String get port_helper_msg => + 'La valeur par défaut est -1, ce qui indique un nombre aléatoire. Si vous avez configuré un pare-feu, il est recommandé de le définir.'; + + @override + String connect_request(Object client) { + return 'Autoriser $client à se connecter ?'; + } + + @override + String get connection_request_denied => + 'Connexion refusée. L\'utilisateur a refusé l\'accès.'; + + @override + String get an_error_occurred => 'Une erreur est survenue'; + + @override + String get copy_to_clipboard => 'Copier dans le presse-papiers'; + + @override + String get view_logs => 'Afficher les journaux'; + + @override + String get retry => 'Réessayer'; + + @override + String get no_default_metadata_provider_selected => + 'Vous n\'avez pas de fournisseur de métadonnées par défaut'; + + @override + String get manage_metadata_providers => + 'Gérer les fournisseurs de métadonnées'; + + @override + String get open_link_in_browser => 'Ouvrir le lien dans le navigateur ?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Voulez-vous ouvrir le lien suivant'; + + @override + String get unsafe_url_warning => + 'L\'ouverture de liens provenant de sources non fiables peut être dangereuse. Soyez prudent !\nVous pouvez également copier le lien dans votre presse-papiers.'; + + @override + String get copy_link => 'Copier le lien'; + + @override + String get building_your_timeline => + 'Construction de votre chronologie en fonction de vos écoutes...'; + + @override + String get official => 'Officiel'; + + @override + String author_name(Object author) { + return 'Auteur : $author'; + } + + @override + String get third_party => 'Tiers'; + + @override + String get plugin_requires_authentication => + 'Le plugin nécessite une authentification'; + + @override + String get update_available => 'Mise à jour disponible'; + + @override + String get supports_scrobbling => 'Supporte le scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Ce plugin scrobble votre musique pour générer votre historique d\'écoute.'; + + @override + String get default_metadata_source => 'Source de métadonnées par défaut'; + + @override + String get set_default_metadata_source => + 'Définir la source de métadonnées par défaut'; + + @override + String get default_audio_source => 'Source audio par défaut'; + + @override + String get set_default_audio_source => 'Définir la source audio par défaut'; + + @override + String get set_default => 'Définir par défaut'; + + @override + String get support => 'Soutien'; + + @override + String get support_plugin_development => + 'Soutenir le développement de plugins'; + + @override + String can_access_name_api(Object name) { + return '- Peut accéder à l\'API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Voulez-vous installer ce plugin ?'; + + @override + String get third_party_plugin_warning => + 'Ce plugin provient d\'un dépôt tiers. Assurez-vous de faire confiance à la source avant de l\'installer.'; + + @override + String get author => 'Auteur'; + + @override + String get this_plugin_can_do_following => 'Ce plugin peut faire ce qui suit'; + + @override + String get install => 'Installer'; + + @override + String get install_a_metadata_provider => + 'Installer un fournisseur de métadonnées'; + + @override + String get no_tracks_playing => + 'Aucune piste n\'est en cours de lecture actuellement'; + + @override + String get synced_lyrics_not_available => + 'Les paroles synchronisées ne sont pas disponibles pour cette chanson. Veuillez utiliser l\'onglet'; + + @override + String get plain_lyrics => 'Paroles simples'; + + @override + String get tab_instead => 'à la place.'; + + @override + String get disclaimer => 'Avertissement'; + + @override + String get third_party_plugin_dmca_notice => + 'L\'équipe de Spotube n\'assume aucune responsabilité (y compris juridique) pour les plugins \"tiers\".\nVeuillez les utiliser à vos propres risques. Pour tout bug/problème, veuillez le signaler au dépôt du plugin.\n\nSi un plugin \"tiers\" enfreint les conditions d\'utilisation/DMCA d\'un service/entité juridique, veuillez demander à l\'auteur du plugin \"tiers\" ou à la plateforme d\'hébergement (par exemple GitHub/Codeberg) de prendre des mesures. Les plugins listés ci-dessus (étiquetés \"tiers\") sont tous des plugins publics/maintenus par la communauté. Nous ne les gérons pas, nous ne pouvons donc prendre aucune mesure à leur sujet.\n\n'; + + @override + String get input_does_not_match_format => + 'L\'entrée ne correspond pas au format requis'; + + @override + String get plugins => 'Plugins'; + + @override + String get paste_plugin_download_url => + 'Collez l\'URL de téléchargement ou l\'URL du dépôt GitHub/Codeberg ou un lien direct vers le fichier .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Télécharger et installer le plugin à partir de l\'URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Échec de l\'ajout du plugin : $error'; + } + + @override + String get upload_plugin_from_file => + 'Télécharger le plugin à partir d\'un fichier'; + + @override + String get installed => 'Installé'; + + @override + String get available_plugins => 'Plugins disponibles'; + + @override + String get configure_plugins => + 'Configurez vos propres plugins de fournisseur de métadonnées et de source audio'; + + @override + String get audio_scrobblers => 'Scrobblers audio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Source : '; + + @override + String get uncompressed => 'Non compressé'; + + @override + String get dab_music_source_description => + 'Pour les audiophiles. Fournit des flux audio de haute qualité/sans perte. Correspondance précise des pistes basée sur ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_hi.dart b/lib/l10n/generated/app_localizations_hi.dart new file mode 100644 index 00000000..0434e8db --- /dev/null +++ b/lib/l10n/generated/app_localizations_hi.dart @@ -0,0 +1,1570 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Hindi (`hi`). +class AppLocalizationsHi extends AppLocalizations { + AppLocalizationsHi([String locale = 'hi']) : super(locale); + + @override + String get guest => 'अतिथि'; + + @override + String get browse => 'ब्राउज़ करें'; + + @override + String get search => 'खोजें'; + + @override + String get library => 'लाइब्रेरी'; + + @override + String get lyrics => 'गीतों के बोल'; + + @override + String get settings => 'सेटिंग्स'; + + @override + String get genre_categories_filter => 'श्रेणियों या जानरों को फिल्टर करें...'; + + @override + String get genre => 'जानर'; + + @override + String get personalized => 'व्यक्तिगत'; + + @override + String get featured => 'विशेष रुप से प्रदर्शित'; + + @override + String get new_releases => 'नई रिलीज़'; + + @override + String get songs => 'गाने'; + + @override + String playing_track(Object track) { + return '$track चल रहा है'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'यह मौजूदा कतार को साफ़ कर देगा। $track_length ट्रैक हटा दिए जाएंगे\nक्या आप जारी रखना चाहते हैं?'; + } + + @override + String get load_more => 'और लोड करें'; + + @override + String get playlists => 'प्लेलिस्ट'; + + @override + String get artists => 'कलाकार'; + + @override + String get albums => 'एल्बम'; + + @override + String get tracks => 'ट्रैक'; + + @override + String get downloads => 'डाउनलोड'; + + @override + String get filter_playlists => 'अपनी प्लेलिस्टों को फ़िल्टर करें...'; + + @override + String get liked_tracks => 'पसंदीदा ट्रैक'; + + @override + String get liked_tracks_description => 'आपके सभी पसंदीदा ट्रैक'; + + @override + String get playlist => 'प्लेलिस्ट'; + + @override + String get create_a_playlist => 'एक प्लेलिस्ट बनाएं'; + + @override + String get update_playlist => 'प्लेलिस्ट अपडेट करें'; + + @override + String get create => 'बनाएं'; + + @override + String get cancel => 'रद्द करें'; + + @override + String get update => 'अपडेट करें'; + + @override + String get playlist_name => 'प्लेलिस्ट का नाम'; + + @override + String get name_of_playlist => 'प्लेलिस्ट का नाम'; + + @override + String get description => 'विवरण'; + + @override + String get public => 'सार्वजनिक'; + + @override + String get collaborative => 'सहयोगी'; + + @override + String get search_local_tracks => 'स्थानीय ट्रैक खोजें...'; + + @override + String get play => 'चलाएँ'; + + @override + String get delete => 'हटाएँ'; + + @override + String get none => 'कोई नहीं'; + + @override + String get sort_a_z => 'A-Z सॉर्ट करें'; + + @override + String get sort_z_a => 'Z-A सॉर्ट करें'; + + @override + String get sort_artist => 'कलाकार के अनुसार सॉर्ट करें'; + + @override + String get sort_album => 'एल्बम के अनुसार सॉर्ट करें'; + + @override + String get sort_duration => 'समय के आधार पर क्रमबद्ध करें'; + + @override + String get sort_tracks => 'ट्रैक को सॉर्ट करें'; + + @override + String currently_downloading(Object tracks_length) { + return 'वर्तमान में डाउनलोड हो रहा है ($tracks_length)'; + } + + @override + String get cancel_all => 'सभी को रद्द करें'; + + @override + String get filter_artist => 'कलाकारों को फ़िल्टर करें...'; + + @override + String followers(Object followers) { + return '$followers फॉलोअर्स'; + } + + @override + String get add_artist_to_blacklist => 'काल सूची में कलाकार जोड़ें'; + + @override + String get top_tracks => 'शीर्ष ट्रैक'; + + @override + String get fans_also_like => 'फैंस भी पसंद करते हैं'; + + @override + String get loading => 'लोड हो रहा है...'; + + @override + String get artist => 'कलाकार'; + + @override + String get blacklisted => 'काल सूची में है'; + + @override + String get following => 'फॉलो करना'; + + @override + String get follow => 'फॉलो करें'; + + @override + String get artist_url_copied => 'कलाकार URL क्लिपबोर्ड पर कॉपी हुआ'; + + @override + String added_to_queue(Object tracks) { + return '$tracks ट्रैक कतार में जोड़े गए'; + } + + @override + String get filter_albums => 'एल्बमों को फ़िल्टर करें...'; + + @override + String get synced => 'सिंक किया गया'; + + @override + String get plain => 'सादा'; + + @override + String get shuffle => 'शफल'; + + @override + String get search_tracks => 'ट्रैक खोजें...'; + + @override + String get released => 'जारी हुआ'; + + @override + String error(Object error) { + return 'त्रुटि $error'; + } + + @override + String get title => 'शीर्षक'; + + @override + String get time => 'समय'; + + @override + String get more_actions => 'अधिक कार्रवाई'; + + @override + String download_count(Object count) { + return 'डाउनलोड ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return '($count) को प्लेलिस्ट में जोड़ें'; + } + + @override + String add_count_to_queue(Object count) { + return '($count) को कतार में जोड़ें'; + } + + @override + String play_count_next(Object count) { + return '($count) अगले में चलाएँ'; + } + + @override + String get album => 'एल्बम'; + + @override + String copied_to_clipboard(Object data) { + return '$data क्लिपबोर्ड पर कॉपी किया गया'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track को निम्नलिखित प्लेलिस्ट में जोड़ें'; + } + + @override + String get add => 'जोड़ें'; + + @override + String added_track_to_queue(Object track) { + return '$track को कतार में जोड़ दिया गया'; + } + + @override + String get add_to_queue => 'कतार में जोड़ें'; + + @override + String track_will_play_next(Object track) { + return '$track अगले में चलेगा'; + } + + @override + String get play_next => 'अगले में चलाएँ'; + + @override + String removed_track_from_queue(Object track) { + return '$track को कतार से हटा दिया गया'; + } + + @override + String get remove_from_queue => 'कतार से हटाएँ'; + + @override + String get remove_from_favorites => 'पसंदीदा से हटाएँ'; + + @override + String get save_as_favorite => 'पसंदीदा के रूप में सहेजें'; + + @override + String get add_to_playlist => 'प्लेलिस्ट में जोड़ें'; + + @override + String get remove_from_playlist => 'प्लेलिस्ट से हटाएँ'; + + @override + String get add_to_blacklist => 'ब्लैकलिस्ट में जोड़ें'; + + @override + String get remove_from_blacklist => 'ब्लैकलिस्ट से हटाएँ'; + + @override + String get share => 'साझा करें'; + + @override + String get mini_player => 'मिनी प्लेयर'; + + @override + String get slide_to_seek => 'आगे या पीछे खोजने के लिए स्लाइड करें'; + + @override + String get shuffle_playlist => 'प्लेलिस्ट शफल करें'; + + @override + String get unshuffle_playlist => 'अनशफल प्लेलिस्ट'; + + @override + String get previous_track => 'पिछला ट्रैक'; + + @override + String get next_track => 'अगला ट्रैक'; + + @override + String get pause_playback => 'वापसी बंद करें'; + + @override + String get resume_playback => 'पुनः चलाना'; + + @override + String get loop_track => 'लूप ट्रैक'; + + @override + String get no_loop => 'कोई लूप नहीं'; + + @override + String get repeat_playlist => 'प्लेलिस्ट दोहराएं'; + + @override + String get queue => 'कतार'; + + @override + String get alternative_track_sources => 'वैकल्पिक ट्रैक स्रोत'; + + @override + String get download_track => 'ट्रैक डाउनलोड करें'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks ट्रैक कतार में हैं'; + } + + @override + String get clear_all => 'सभी हटाएं'; + + @override + String get show_hide_ui_on_hover => 'होवर पर यूआई दिखाएँ/छिपाएँ'; + + @override + String get always_on_top => 'हमेशा ऊपर हो'; + + @override + String get exit_mini_player => 'मिनी प्लेयर से बाहर निकलें'; + + @override + String get download_location => 'डाउनलोड स्थान'; + + @override + String get local_library => 'स्थानीय पुस्तकालय'; + + @override + String get add_library_location => 'पुस्तकालय में जोड़ें'; + + @override + String get remove_library_location => 'पुस्तकालय से हटाएं'; + + @override + String get account => 'खाता'; + + @override + String get logout => 'लॉगआउट'; + + @override + String get logout_of_this_account => 'इस खाते से लॉगआउट करें'; + + @override + String get language_region => 'भाषा और क्षेत्र'; + + @override + String get language => 'भाषा'; + + @override + String get system_default => 'सिस्टम डिफ़ॉल्ट'; + + @override + String get market_place_region => 'मार्केटप्लेस क्षेत्र'; + + @override + String get recommendation_country => 'सिफ़ारिश देने वाला देश'; + + @override + String get appearance => 'दिखने में'; + + @override + String get layout_mode => 'लेआउट मोड'; + + @override + String get override_layout_settings => + 'ओवरराइड रेस्पॉन्सिव लेआउट मोड सेटिंग्स'; + + @override + String get adaptive => 'अनुकूल'; + + @override + String get compact => 'कॉम्पैक्ट'; + + @override + String get extended => 'विस्तृत'; + + @override + String get theme => 'थीम'; + + @override + String get dark => 'डार्क'; + + @override + String get light => 'लाइट'; + + @override + String get system => 'सिस्टम'; + + @override + String get accent_color => 'अक्षरशैली का रंग'; + + @override + String get sync_album_color => 'एल्बम का रंग सिंक करें'; + + @override + String get sync_album_color_description => + 'एल्बम कला का प्रधान रंग एक्सेंट रंग के रूप में उपयोग किया जाता है'; + + @override + String get playback => 'प्लेबैक'; + + @override + String get audio_quality => 'ऑडियो क्वालिटी'; + + @override + String get high => 'उच्च'; + + @override + String get low => 'निम्न'; + + @override + String get pre_download_play => 'पूर्वावत डाउनलोड और प्ले करें'; + + @override + String get pre_download_play_description => + 'ऑडियो स्ट्रीमिंग की बजाय बाइट्स डाउनलोड करें और बजाय में प्ले करें (उच्च बैंडविड्थ उपयोगकर्ताओं के लिए सिफारिश किया जाता है)'; + + @override + String get skip_non_music => + 'गाने के अलावा सेगमेंट्स को छोड़ें (स्पॉन्सरब्लॉक)'; + + @override + String get blacklist_description => 'ब्लैकलिस्ट में शामिल ट्रैक और कलाकार'; + + @override + String get wait_for_download_to_finish => + 'वर्तमान डाउनलोड समाप्त होने तक कृपया प्रतीक्षा करें'; + + @override + String get desktop => 'डेस्कटॉप'; + + @override + String get close_behavior => 'बंद करने का व्यवहार'; + + @override + String get close => 'बंद करें'; + + @override + String get minimize_to_tray => 'ट्रे में कम करें'; + + @override + String get show_tray_icon => 'सिस्टम ट्रे आइकन दिखाएं'; + + @override + String get about => 'के बारे में'; + + @override + String get u_love_spotube => 'हम जानते हैं कि आप Spotube से प्यार करते हैं'; + + @override + String get check_for_updates => 'अपडेट के लिए जाँच करें'; + + @override + String get about_spotube => 'Spotube के बारे में'; + + @override + String get blacklist => 'ब्लैकलिस्ट'; + + @override + String get please_sponsor => 'कृपया स्पॉन्सर / डोनेट करें'; + + @override + String get spotube_description => + 'Spotube, एक हल्का, सभी प्लेटफॉर्मों पर चलने वाला, मुफ्त स्पॉटिफाई क्लाइंट'; + + @override + String get version => 'संस्करण'; + + @override + String get build_number => 'बिल्ड नंबर'; + + @override + String get founder => 'संस्थापक'; + + @override + String get repository => 'भण्डार'; + + @override + String get bug_issues => 'बग+मुद्दे'; + + @override + String get made_with => 'बांग्लादेश🇧🇩 में दिल से बनाया गया'; + + @override + String get kingkor_roy_tirtho => 'किंगकोर रॉय तिर्थो'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year किंगकोर रॉय तिर्थो'; + } + + @override + String get license => 'लाइसेंस'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'चिंता न करें, आपके क्रेडेंशियल किसी भी तरह से नहीं एकत्रित या साझा किए जाएंगे'; + + @override + String get know_how_to_login => 'इसे कैसे करें पता नहीं?'; + + @override + String get follow_step_by_step_guide => 'कदम से कदम गाइड के साथ चलें'; + + @override + String cookie_name_cookie(Object name) { + return '$name कुकी'; + } + + @override + String get fill_in_all_fields => 'कृपया सभी फ़ील्ड भरें'; + + @override + String get submit => 'सबमिट'; + + @override + String get exit => 'बाहर निकलें'; + + @override + String get previous => 'पिछला'; + + @override + String get next => 'अगला'; + + @override + String get done => 'किया हुआ'; + + @override + String get step_1 => '1 चरण'; + + @override + String get first_go_to => 'पहले, जाएं'; + + @override + String get something_went_wrong => 'कुछ गलत हो गया'; + + @override + String get piped_instance => 'पाइप्ड सर्वर'; + + @override + String get piped_description => 'पाइप किए गए सर्वर'; + + @override + String get piped_warning => + 'गानों का मिलान करने के लिए उपयोग किए जाते हैं, हो सकता है कि उनमें से कुछ के साथ ठीक से काम न करें इसलिए अपने जोखिम पर उपयोग करें'; + + @override + String get invidious_instance => 'इन्विडियस सर्वर इंस्टेंस'; + + @override + String get invidious_description => + 'ट्रैक मिलान के लिए इन्विडियस सर्वर इंस्टेंस'; + + @override + String get invidious_warning => + 'कुछ इंस्टेंस अच्छी तरह से काम नहीं कर सकते। अपने जोखिम पर उपयोग करें'; + + @override + String get generate => 'उत्पन्न करें'; + + @override + String track_exists(Object track) { + return 'ट्रैक $track पहले से मौजूद है'; + } + + @override + String get replace_downloaded_tracks => 'सभी डाउनलोड किए गए ट्रैक्स को बदलें'; + + @override + String get skip_download_tracks => 'सभी डाउनलोड किए गए ट्रैक्स को छोड़ें'; + + @override + String get do_you_want_to_replace => + 'क्या आप मौजूदा ट्रैक को बदलना चाहते हैं?'; + + @override + String get replace => 'बदलें'; + + @override + String get skip => 'छोड़ें'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '$count $type तक चुनें'; + } + + @override + String get select_genres => 'जान्र चुनें'; + + @override + String get add_genres => 'जान्र जोड़ें'; + + @override + String get country => 'देश'; + + @override + String get number_of_tracks_generate => 'उत्पन्न करने के लिए ट्रैक की संख्या'; + + @override + String get acousticness => 'ध्वनिकता'; + + @override + String get danceability => 'नृत्यता'; + + @override + String get energy => 'ऊर्जा'; + + @override + String get instrumentalness => 'आलापिकता'; + + @override + String get liveness => 'जीवंतता'; + + @override + String get loudness => 'शोर'; + + @override + String get speechiness => 'बोलचालता'; + + @override + String get valence => 'मनोदयता'; + + @override + String get popularity => 'लोकप्रियता'; + + @override + String get key => 'कुंजी'; + + @override + String get duration => 'अवधि (सेकंड)'; + + @override + String get tempo => 'गति (BPM)'; + + @override + String get mode => 'मोड'; + + @override + String get time_signature => 'समय छाप'; + + @override + String get short => 'संक्षेप'; + + @override + String get medium => 'मध्यम'; + + @override + String get long => 'लंबा'; + + @override + String get min => 'न्यूनतम'; + + @override + String get max => 'अधिकतम'; + + @override + String get target => 'लक्ष्य'; + + @override + String get moderate => 'मध्यम'; + + @override + String get deselect_all => 'सभी को अचयनित करें'; + + @override + String get select_all => 'सभी को चुनें'; + + @override + String get are_you_sure => 'क्या आपको यकीन है?'; + + @override + String get generating_playlist => 'आपकी कस्टम प्लेलिस्ट बनाई जा रही है...'; + + @override + String selected_count_tracks(Object count) { + return '$count ट्रैक्स चयनित हैं'; + } + + @override + String get download_warning => + 'यदि आप सभी ट्रैक्स को बल्क में डाउनलोड करते हैं, तो आप स्पष्ट रूप से संगीत की अवैध नकली बना रहे हैं और संगीत के रचनात्मक समाज को क्षति पहुंचा रहे हैं। मुझे आशा है कि आप इसके बारे में जागरूक हैं। हमेशा कोशिश करें कि कलाकार के मेहनत का सम्मान और समर्थन करें।'; + + @override + String get download_ip_ban_warning => + 'बाहरी डाउनलोड अनुरोधों के कारण आपका आईपी YouTube पर अधिक से अधिक ब्लॉक हो सकता है। आईपी ब्लॉक का अर्थ है कि आप उसी आईपी उपकरण से कम से कम 2-3 महीनों तक YouTube का उपयोग नहीं कर सकेंगे (यदि आप लॉग इन हैं तो भी)। और स्पोट्यूब किसी भी जिम्मेदारी को नहीं उठाता है अगर ऐसा कभी होता है।'; + + @override + String get by_clicking_accept_terms => + '\'स्वीकार\' पर क्लिक करके आप निम्नलिखित शर्तों से सहमत होते हैं:'; + + @override + String get download_agreement_1 => + 'मुझे पता है कि मैं संगीत की अवैध नकली बना रहा हूं। मैं बुरा हूं'; + + @override + String get download_agreement_2 => + 'मैं कलाकार का समर्थन करूंगा जहां भी मुझे संभव हो और मैं केवल इसल िए ऐसा कर रहा हूं क्योंकि मेरे पास उनकी कला खरीदने के लिए पैसे नहीं हैं।'; + + @override + String get download_agreement_3 => + 'मैं पूरी तरह से जागरूक हूं कि मेरा आईपी YouTube पर ब्लॉक हो सकता है और मैं स्पोट्यूब या उसके मालिकों / सहयोगियों को किसी भी दुर्घटना के लिए जिम्मेदार नहीं मानता।'; + + @override + String get decline => 'इनकार करें'; + + @override + String get accept => 'स्वीकार करें'; + + @override + String get details => 'विवरण'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'चैनल'; + + @override + String get likes => 'पसंद'; + + @override + String get dislikes => 'अप्रिय'; + + @override + String get views => 'दृश्य'; + + @override + String get streamUrl => 'स्ट्रीम URL'; + + @override + String get stop => 'रोकें'; + + @override + String get sort_newest => 'नवीनतम जोड़े गए के अनुसार क्रमबद्ध करें'; + + @override + String get sort_oldest => 'सबसे पुराने जोड़े गए के अनुसार क्रमबद्ध करें'; + + @override + String get sleep_timer => 'स्लीप टाइमर'; + + @override + String mins(Object minutes) { + return '$minutes मिनट'; + } + + @override + String hours(Object hours) { + return '$hours घंटे'; + } + + @override + String hour(Object hours) { + return '$hours घंटा'; + } + + @override + String get custom_hours => 'कस्टम घंटे'; + + @override + String get logs => 'लॉग'; + + @override + String get developers => 'डेवलपर्स'; + + @override + String get not_logged_in => 'आप लॉग इन नहीं हैं'; + + @override + String get search_mode => 'खोज मोड'; + + @override + String get audio_source => 'ऑडियो स्रोत'; + + @override + String get ok => 'ठीक है'; + + @override + String get failed_to_encrypt => 'एन्क्रिप्ट करने में विफल रहा'; + + @override + String get encryption_failed_warning => + 'Spotube आपके डेटा को सुरक्षित रूप से स्टोर करने के लिए एन्क्रिप्शन का उपयोग करता है। लेकिन इसमें विफल रहा। इसलिए, यह असुरक्षित स्टोरेज पर फॉलबैक करेगा\nयदि आप Linux का उपयोग कर रहे हैं, तो कृपया सुनिश्चित करें कि आपके पास gnome-keyring, kde-wallet, keepassxc आदि जैसी कोई सीक्रेट-सर्विस इंस्टॉल की गई है'; + + @override + String get querying_info => 'जानकारी प्राप्त करना'; + + @override + String get piped_api_down => 'पाइप्ड एपीआई डाउन है'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'पाइप्ड इंस्टेंस $pipedInstance वर्तमान में डाउन है\n\nइंस्टेंस बदलें या \'एपीआई प्रकार\' को आधिकृत YouTube एपीआई में बदलें\n\nपरिवर्तन के बाद ऐप को फिर से चालने की सुनिश्चित करें'; + } + + @override + String get you_are_offline => 'आप वर्तमान में ऑफ़लाइन हैं'; + + @override + String get connection_restored => 'आपका इंटरनेट कनेक्शन बहाल हो गया है'; + + @override + String get use_system_title_bar => 'सिस्टम शीर्षक पट्टी का उपयोग करें'; + + @override + String get crunching_results => 'परिणाम को प्रसंस्कृत किया जा रहा है...'; + + @override + String get search_to_get_results => 'परिणाम प्राप्त करने के लिए खोजें'; + + @override + String get use_amoled_mode => 'AMOLED मोड का उपयोग करें'; + + @override + String get pitch_dark_theme => 'पिच ब्लैक डार्ट थीम'; + + @override + String get normalize_audio => 'ऑडियो को सामान्य करें'; + + @override + String get change_cover => 'कवर बदलें'; + + @override + String get add_cover => 'कवर जोड़ें'; + + @override + String get restore_defaults => 'डिफ़ॉल्ट सेटिंग्स को बहाल करें'; + + @override + String get download_music_format => 'संगीत डाउनलोड प्रारूप'; + + @override + String get streaming_music_format => 'संगीत स्ट्रीमिंग प्रारूप'; + + @override + String get download_music_quality => 'संगीत डाउनलोड गुणवत्ता'; + + @override + String get streaming_music_quality => 'संगीत स्ट्रीमिंग गुणवत्ता'; + + @override + String get login_with_lastfm => 'Last.fm से लॉगिन करें'; + + @override + String get connect => 'कनेक्ट करें'; + + @override + String get disconnect_lastfm => 'Last.fm से डिस्कनेक्ट करें'; + + @override + String get disconnect => 'डिस्कनेक्ट करें'; + + @override + String get username => 'उपयोगकर्ता नाम'; + + @override + String get password => 'पासवर्ड'; + + @override + String get login => 'लॉग इन करें'; + + @override + String get login_with_your_lastfm => 'अपने Last.fm अकाउंट से लॉगिन करें'; + + @override + String get scrobble_to_lastfm => 'Last.fm पर स्क्रॉबल करें'; + + @override + String get go_to_album => 'एल्बम पर जाएं'; + + @override + String get discord_rich_presence => 'डिस्कॉर्ड रिच प्रेजेंस'; + + @override + String get browse_all => 'सभी को ब्राउज़ करें'; + + @override + String get genres => 'शैलियाँ'; + + @override + String get explore_genres => 'शैलियों का अन्वेषण करें'; + + @override + String get friends => 'दोस्त'; + + @override + String get no_lyrics_available => + 'क्षमा करें, इस ट्रैक के लिए गाने नहीं मिल सके'; + + @override + String get start_a_radio => 'रेडियो शुरू करें'; + + @override + String get how_to_start_radio => 'रेडियो कैसे शुरू करना चाहते हैं?'; + + @override + String get replace_queue_question => + 'क्या आप वर्तमान कतार को बदलना चाहते हैं या इसे जोड़ना चाहते हैं?'; + + @override + String get endless_playback => 'अंतहीन प्लेबैक'; + + @override + String get delete_playlist => 'प्लेलिस्ट हटाएं'; + + @override + String get delete_playlist_confirmation => + 'क्या आप वाकई इस प्लेलिस्ट को हटाना चाहते हैं?'; + + @override + String get local_tracks => 'स्थानीय ट्रैक्स'; + + @override + String get local_tab => 'स्थानीय'; + + @override + String get song_link => 'गाने का लिंक'; + + @override + String get skip_this_nonsense => 'इस माया को छोड़ें'; + + @override + String get freedom_of_music => '“संगीत की स्वतंत्रता”'; + + @override + String get freedom_of_music_palm => '“हाथ में संगीत की स्वतंत्रता”'; + + @override + String get get_started => 'आइए शुरू करें'; + + @override + String get youtube_source_description => + 'सिफारिश किया गया और सबसे अच्छा काम करता है।'; + + @override + String get piped_source_description => + 'मुफ्त महसूस कर रहे हैं? YouTube के समान लेकिन काफी अधिक मुफ्त।'; + + @override + String get jiosaavn_source_description => + 'दक्षिण एशियाई क्षेत्र के लिए सर्वोत्तम।'; + + @override + String get invidious_source_description => + 'पाइप्ड के समान, लेकिन अधिक उपलब्धता के साथ'; + + @override + String highest_quality(Object quality) { + return 'सर्वोत्तम गुणवत्ता: $quality'; + } + + @override + String get select_audio_source => 'ऑडियो स्रोत चुनें'; + + @override + String get endless_playback_description => + 'क्रमबद्ध कतार के अंत में नए गाने स्वचालित रूप से जोड़ें'; + + @override + String get choose_your_region => 'अपना क्षेत्र चुनें'; + + @override + String get choose_your_region_description => + 'यह Spotube को आपके स्थान के लिए सही सामग्री दिखाने में मदद करेगा।'; + + @override + String get choose_your_language => 'अपनी भाषा चुनें'; + + @override + String get help_project_grow => 'इस परियोजना को बढ़ावा दें'; + + @override + String get help_project_grow_description => + 'Spotube एक ओपन सोर्स परियोजना है। आप इस परियोजना को योगदान देकर, बग रिपोर्ट करके या नई विशेषताओं का सुझाव देकर इस परियोजना को बढ़ा सकते हैं।'; + + @override + String get contribute_on_github => 'GitHub पर योगदान करें'; + + @override + String get donate_on_open_collective => 'ओपन कलेक्टिव पर दान करें'; + + @override + String get browse_anonymously => 'बिना नाम के ब्राउज़ करें'; + + @override + String get enable_connect => 'कनेक्ट सक्षम करें'; + + @override + String get enable_connect_description => + 'अन्य उपकरणों से Spotube को नियंत्रित करें'; + + @override + String get devices => 'उपकरण'; + + @override + String get select => 'चयन करें'; + + @override + String connect_client_alert(Object client) { + return 'आप $client द्वारा नियंत्रित हो रहे हैं'; + } + + @override + String get this_device => 'यह उपकरण'; + + @override + String get remote => 'रिमोट'; + + @override + String get stats => 'आंकड़े'; + + @override + String and_n_more(Object count) { + return 'और $count और'; + } + + @override + String get recently_played => 'हाल ही में खेले गए'; + + @override + String get browse_more => 'अधिक ब्राउज़ करें'; + + @override + String get no_title => 'कोई शीर्षक नहीं'; + + @override + String get not_playing => 'नहीं चल रहा'; + + @override + String get epic_failure => 'महान असफलता!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length ट्रैक्स कतार में जोड़े गए'; + } + + @override + String get spotube_has_an_update => 'Spotube में एक अपडेट है'; + + @override + String get download_now => 'अभी डाउनलोड करें'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum जारी किया गया है'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version जारी किया गया है'; + } + + @override + String get read_the_latest => 'नवीनतम पढ़ें'; + + @override + String get release_notes => 'रिलीज़ नोट्स'; + + @override + String get pick_color_scheme => 'रंग योजना चुनें'; + + @override + String get save => 'सहेजें'; + + @override + String get choose_the_device => 'उपकरण चुनें:'; + + @override + String get multiple_device_connected => + 'कई उपकरण जुड़े हुए हैं।\nउस उपकरण को चुनें जिस पर आप यह क्रिया करना चाहते हैं'; + + @override + String get nothing_found => 'कुछ भी नहीं मिला'; + + @override + String get the_box_is_empty => 'बॉक्स खाली है'; + + @override + String get top_artists => 'शीर्ष कलाकार'; + + @override + String get top_albums => 'शीर्ष एल्बम'; + + @override + String get this_week => 'इस हफ्ते'; + + @override + String get this_month => 'इस महीने'; + + @override + String get last_6_months => 'पिछले 6 महीने'; + + @override + String get this_year => 'इस साल'; + + @override + String get last_2_years => 'पिछले 2 साल'; + + @override + String get all_time => 'सभी समय'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName द्वारा संचालित'; + } + + @override + String get email => 'ईमेल'; + + @override + String get profile_followers => 'अनुयायी'; + + @override + String get birthday => 'जन्मदिन'; + + @override + String get subscription => 'सदस्यता'; + + @override + String get not_born => 'अभी पैदा नहीं हुआ'; + + @override + String get hacker => 'हैकर'; + + @override + String get profile => 'प्रोफ़ाइल'; + + @override + String get no_name => 'कोई नाम नहीं'; + + @override + String get edit => 'संपादित करें'; + + @override + String get user_profile => 'उपयोगकर्ता प्रोफ़ाइल'; + + @override + String count_plays(Object count) { + return '$count प्ले'; + } + + @override + String get streaming_fees_hypothetical => + '*Spotify की प्रति स्ट्रीम भुगतान के आधार पर\n\$0.003 से \$0.005 तक गणना की गई है। यह एक काल्पनिक\nगणना है जो उपयोगकर्ता को यह जानकारी देती है कि वे कितना भुगतान\nकरते यदि वे Spotify पर गाने सुनते।'; + + @override + String get minutes_listened => 'सुनिएका मिनेटहरू'; + + @override + String get streamed_songs => 'स्ट्रीम गरिएका गीतहरू'; + + @override + String count_streams(Object count) { + return '$count स्ट्रिम'; + } + + @override + String get owned_by_you => 'तपाईंले स्वामित्व गरेको'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl क्लिपबोर्डमा कपी गरियो'; + } + + @override + String get hipotetical_calculation => + '*यह औसत ऑनलाइन संगीत स्ट्रीमिंग प्लेटफ़ॉर्म के प्रति स्ट्रीम भुगतान (\$0.003 से \$0.005) के आधार पर गणना की गई है। यह एक काल्पनिक गणना है जो उपयोगकर्ता को यह जानकारी देने के लिए है कि यदि वे विभिन्न संगीत स्ट्रीमिंग प्लेटफ़ॉर्म पर अपने गाने सुनते हैं तो उन्होंने कलाकारों को कितना भुगतान किया होगा।'; + + @override + String count_mins(Object minutes) { + return '$minutes मिनट'; + } + + @override + String get summary_minutes => 'मिनट'; + + @override + String get summary_listened_to_music => 'सुनी गई संगीत'; + + @override + String get summary_songs => 'गाने'; + + @override + String get summary_streamed_overall => 'कुल स्ट्रीम'; + + @override + String get summary_owed_to_artists => 'कलाकारों को देनदार\nइस महीने'; + + @override + String get summary_artists => 'कलाकार'; + + @override + String get summary_music_reached_you => 'संगीत आपके पास पहुंच गया'; + + @override + String get summary_full_albums => 'पूरा एल्बम'; + + @override + String get summary_got_your_love => 'आपका प्यार मिला'; + + @override + String get summary_playlists => 'प्लेलिस्ट'; + + @override + String get summary_were_on_repeat => 'दोहराया गया'; + + @override + String total_money(Object money) { + return 'कुल $money'; + } + + @override + String get webview_not_found => 'वेबव्यू नहीं मिला'; + + @override + String get webview_not_found_description => + 'आपके डिवाइस पर वेबव्यू रनटाइम इंस्टॉल नहीं है।\nअगर इंस्टॉल है, तो सुनिश्चित करें कि यह environment PATH में है\n\nइंस्टॉल करने के बाद, ऐप को पुनः शुरू करें'; + + @override + String get unsupported_platform => 'असमर्थित प्लेटफार्म'; + + @override + String get cache_music => 'संगीत को कैश करें'; + + @override + String get open => 'खोलें'; + + @override + String get cache_folder => 'कैश फ़ोल्डर'; + + @override + String get export => 'निर्यात करें'; + + @override + String get clear_cache => 'कैश साफ़ करें'; + + @override + String get clear_cache_confirmation => 'क्या आप कैश साफ़ करना चाहते हैं?'; + + @override + String get export_cache_files => 'कैश फ़ाइलें निर्यात करें'; + + @override + String found_n_files(Object count) { + return '$count फ़ाइलें मिलीं'; + } + + @override + String get export_cache_confirmation => + 'क्या आप इन फ़ाइलों को निर्यात करना चाहते हैं'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported फ़ाइलें निर्यात की गईं $files में से'; + } + + @override + String get undo => 'पूर्ववत करें'; + + @override + String get download_all => 'सभी डाउनलोड करें'; + + @override + String get add_all_to_playlist => 'सभी को प्लेलिस्ट में जोड़ें'; + + @override + String get add_all_to_queue => 'सभी को कतार में जोड़ें'; + + @override + String get play_all_next => 'सभी को अगले खेलने के लिए'; + + @override + String get pause => 'रोकें'; + + @override + String get view_all => 'सभी देखें'; + + @override + String get no_tracks_added_yet => + 'लगता है आपने अभी तक कोई ट्रैक नहीं जोड़ा है।'; + + @override + String get no_tracks => 'लगता है यहाँ कोई ट्रैक नहीं है।'; + + @override + String get no_tracks_listened_yet => 'लगता है आपने अभी तक कुछ नहीं सुना है।'; + + @override + String get not_following_artists => + 'आप किसी भी कलाकार को फॉलो नहीं कर रहे हैं।'; + + @override + String get no_favorite_albums_yet => + 'लगता है आपने अभी तक कोई एल्बम अपनी पसंदीदा सूची में नहीं जोड़ा है।'; + + @override + String get no_logs_found => 'कोई लॉग नहीं मिला'; + + @override + String get youtube_engine => 'YouTube इंजन'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine स्थापित नहीं है'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine आपके सिस्टम में स्थापित नहीं है।'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'यह सुनिश्चित करें कि यह PATH वेरिएबल में उपलब्ध हो या\nनीचे $engine निष्पादन योग्य फ़ाइल का पूर्ण पथ सेट करें।'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/यूनिक्स जैसे OS में, .zshrc/.bashrc/.bash_profile आदि में पथ सेट करना काम नहीं करेगा।\nआपको पथ को शेल कॉन्फ़िगरेशन फ़ाइल में सेट करना होगा।'; + + @override + String get download => 'डाउनलोड करें'; + + @override + String get file_not_found => 'फाइल नहीं मिली'; + + @override + String get custom => 'कस्टम'; + + @override + String get add_custom_url => 'कस्टम URL जोड़ें'; + + @override + String get edit_port => 'पोर्ट संपादित करें'; + + @override + String get port_helper_msg => + 'डिफ़ॉल्ट -1 है जो यादृच्छिक संख्या को दर्शाता है। यदि आपने फ़ायरवॉल कॉन्फ़िगर किया है, तो इसे सेट करना अनुशंसित है।'; + + @override + String connect_request(Object client) { + return '$client को कनेक्ट करने की अनुमति दें?'; + } + + @override + String get connection_request_denied => + 'कनेक्शन अस्वीकृत। उपयोगकर्ता ने पहुंच अस्वीकृत कर दी।'; + + @override + String get an_error_occurred => 'एक त्रुटि हुई'; + + @override + String get copy_to_clipboard => 'क्लिपबोर्ड पर कॉपी करें'; + + @override + String get view_logs => 'लॉग देखें'; + + @override + String get retry => 'पुनः प्रयास करें'; + + @override + String get no_default_metadata_provider_selected => + 'आपने कोई डिफ़ॉल्ट मेटाडेटा प्रदाता सेट नहीं किया है'; + + @override + String get manage_metadata_providers => 'मेटाडेटा प्रदाताओं को प्रबंधित करें'; + + @override + String get open_link_in_browser => 'ब्राउज़र में लिंक खोलें?'; + + @override + String get do_you_want_to_open_the_following_link => + 'क्या आप निम्नलिखित लिंक खोलना चाहते हैं'; + + @override + String get unsafe_url_warning => + 'अविश्वसनीय स्रोतों से लिंक खोलना असुरक्षित हो सकता है। सावधान रहें!\nआप लिंक को अपने क्लिपबोर्ड पर भी कॉपी कर सकते हैं।'; + + @override + String get copy_link => 'लिंक कॉपी करें'; + + @override + String get building_your_timeline => + 'आपकी सुनने की आदतों के आधार पर आपकी टाइमलाइन बनाई जा रही है...'; + + @override + String get official => 'आधिकारिक'; + + @override + String author_name(Object author) { + return 'लेखक: $author'; + } + + @override + String get third_party => 'तृतीय-पक्ष'; + + @override + String get plugin_requires_authentication => + 'प्लगइन को प्रमाणीकरण की आवश्यकता है'; + + @override + String get update_available => 'अपडेट उपलब्ध है'; + + @override + String get supports_scrobbling => 'स्क्रॉबलिंग का समर्थन करता है'; + + @override + String get plugin_scrobbling_info => + 'यह प्लगइन आपके सुनने के इतिहास को उत्पन्न करने के लिए आपके संगीत को स्क्रॉबल करता है।'; + + @override + String get default_metadata_source => 'डिफ़ॉल्ट मेटाडेटा स्रोत'; + + @override + String get set_default_metadata_source => 'डिफ़ॉल्ट मेटाडेटा स्रोत सेट करें'; + + @override + String get default_audio_source => 'डिफ़ॉल्ट ऑडियो स्रोत'; + + @override + String get set_default_audio_source => 'डिफ़ॉल्ट ऑडियो स्रोत सेट करें'; + + @override + String get set_default => 'डिफ़ॉल्ट सेट करें'; + + @override + String get support => 'समर्थन'; + + @override + String get support_plugin_development => 'प्लगइन विकास का समर्थन करें'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API तक पहुंच सकता है'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'क्या आप इस प्लगइन को स्थापित करना चाहते हैं?'; + + @override + String get third_party_plugin_warning => + 'यह प्लगइन एक तृतीय-पक्ष रिपॉजिटरी से है। कृपया सुनिश्चित करें कि आप इसे स्थापित करने से पहले स्रोत पर भरोसा करते हैं।'; + + @override + String get author => 'लेखक'; + + @override + String get this_plugin_can_do_following => 'यह प्लगइन निम्नलिखित कर सकता है'; + + @override + String get install => 'स्थापित करें'; + + @override + String get install_a_metadata_provider => 'एक मेटाडेटा प्रदाता स्थापित करें'; + + @override + String get no_tracks_playing => 'वर्तमान में कोई ट्रैक नहीं चल रहा है'; + + @override + String get synced_lyrics_not_available => + 'इस गाने के लिए सिंक्रनाइज़ किए गए बोल उपलब्ध नहीं हैं। कृपया'; + + @override + String get plain_lyrics => 'सादे बोल'; + + @override + String get tab_instead => 'टैब का उपयोग करें।'; + + @override + String get disclaimer => 'अस्वीकरण'; + + @override + String get third_party_plugin_dmca_notice => + 'स्पॉट्यूब टीम किसी भी \"तृतीय-पक्ष\" प्लगइन के लिए कोई जिम्मेदारी (कानूनी सहित) नहीं लेती है।\nकृपया उन्हें अपने जोखिम पर उपयोग करें। किसी भी बग/समस्या के लिए, कृपया उन्हें प्लगइन रिपॉजिटरी को रिपोर्ट करें।\n\nयदि कोई \"तृतीय-पक्ष\" प्लगइन किसी सेवा/कानूनी इकाई के ToS/DMCA को तोड़ रहा है, तो कृपया \"तृतीय-पक्ष\" प्लगइन लेखक या होस्टिंग प्लेटफ़ॉर्म जैसे GitHub/Codeberg से कार्रवाई करने के लिए कहें। ऊपर सूचीबद्ध (\"तृतीय-पक्ष\" लेबल वाले) सभी सार्वजनिक/समुदाय-द्वारा-रखरखाव किए गए प्लगइन हैं। हम उन्हें क्यूरेट नहीं कर रहे हैं, इसलिए हम उन पर कोई कार्रवाई नहीं कर सकते हैं।\n\n'; + + @override + String get input_does_not_match_format => + 'इनपुट आवश्यक प्रारूप से मेल नहीं खाता है'; + + @override + String get plugins => 'प्लगइन्स'; + + @override + String get paste_plugin_download_url => + 'डाउनलोड यूआरएल या गिटहब/कोडबर्ग रेपो यूआरएल या .smplug फ़ाइल का सीधा लिंक पेस्ट करें'; + + @override + String get download_and_install_plugin_from_url => + 'यूआरएल से प्लगइन डाउनलोड और स्थापित करें'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'प्लगइन जोड़ने में विफल: $error'; + } + + @override + String get upload_plugin_from_file => 'फ़ाइल से प्लगइन अपलोड करें'; + + @override + String get installed => 'स्थापित'; + + @override + String get available_plugins => 'उपलब्ध प्लगइन'; + + @override + String get configure_plugins => + 'अपने स्वयं के मेटाडेटा प्रदाता और ऑडियो स्रोत प्लगइन्स कॉन्फ़िगर करें'; + + @override + String get audio_scrobblers => 'ऑडियो स्क्रॉबलर्स'; + + @override + String get scrobbling => 'स्क्रॉबलिंग'; + + @override + String get source => 'स्रोत: '; + + @override + String get uncompressed => 'असंपीड़ित'; + + @override + String get dab_music_source_description => + 'ऑडियोफाइलों के लिए। उच्च-गुणवत्ता/बिना हानि वाले ऑडियो स्ट्रीम प्रदान करता है। सटीक ISRC आधारित ट्रैक मिलान।'; +} diff --git a/lib/l10n/generated/app_localizations_id.dart b/lib/l10n/generated/app_localizations_id.dart new file mode 100644 index 00000000..ce250425 --- /dev/null +++ b/lib/l10n/generated/app_localizations_id.dart @@ -0,0 +1,1572 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Indonesian (`id`). +class AppLocalizationsId extends AppLocalizations { + AppLocalizationsId([String locale = 'id']) : super(locale); + + @override + String get guest => 'Tamu'; + + @override + String get browse => 'Jelajahi'; + + @override + String get search => 'Cari'; + + @override + String get library => 'Pustaka'; + + @override + String get lyrics => 'Lirik'; + + @override + String get settings => 'Pengaturan'; + + @override + String get genre_categories_filter => 'Urutkan kategori atau genre...'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Dipersonalisasi'; + + @override + String get featured => 'Unggulan'; + + @override + String get new_releases => 'Rilis Terbaru'; + + @override + String get songs => 'Lagu'; + + @override + String playing_track(Object track) { + return 'Memutar $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Ini akan menghapus antrian saat ini This will clear the current queue. $track_length trek akan dihapus\nAnda ingin melanjutkan?'; + } + + @override + String get load_more => 'Lebih Banyak'; + + @override + String get playlists => 'Daftar Putar'; + + @override + String get artists => 'Artis'; + + @override + String get albums => 'Album'; + + @override + String get tracks => 'Trek'; + + @override + String get downloads => 'Unduhan'; + + @override + String get filter_playlists => 'Urutkan daftar putar Anda...'; + + @override + String get liked_tracks => 'Lagu Yang Disukai'; + + @override + String get liked_tracks_description => 'Semua lagu yang Anda sukai'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Buat daftar putar'; + + @override + String get update_playlist => 'Ubah daftar putar'; + + @override + String get create => 'Buat'; + + @override + String get cancel => 'Batal'; + + @override + String get update => 'Ubah'; + + @override + String get playlist_name => 'Nama Daftar Putar'; + + @override + String get name_of_playlist => 'Nama daftar putar'; + + @override + String get description => 'Deskripsi'; + + @override + String get public => 'Publik'; + + @override + String get collaborative => 'Kolaboratif'; + + @override + String get search_local_tracks => 'Cari trek lokal...'; + + @override + String get play => 'Putar'; + + @override + String get delete => 'Hapus'; + + @override + String get none => 'Tidak Ada'; + + @override + String get sort_a_z => 'Urutkan berdasarkan A-Z'; + + @override + String get sort_z_a => 'Urutkan berdasarkan Z-A'; + + @override + String get sort_artist => 'Urutkan berdasarkan Artis'; + + @override + String get sort_album => 'Urutkan berdasarkan Album'; + + @override + String get sort_duration => 'Urutkan berdasarkan Durasi'; + + @override + String get sort_tracks => 'Urutkan trek'; + + @override + String currently_downloading(Object tracks_length) { + return 'Sedang Mengunduh ($tracks_length)'; + } + + @override + String get cancel_all => 'Batalkan Semua'; + + @override + String get filter_artist => 'Urutkan artis...'; + + @override + String followers(Object followers) { + return '$followers Pengikut'; + } + + @override + String get add_artist_to_blacklist => 'Tambah artis ke daftar hitam'; + + @override + String get top_tracks => 'Lagu Teratas'; + + @override + String get fans_also_like => 'Penggemar juga menyukainya'; + + @override + String get loading => 'Memuat...'; + + @override + String get artist => 'Artis'; + + @override + String get blacklisted => 'Masuk Daftar Hitam'; + + @override + String get following => 'Mengikuti'; + + @override + String get follow => 'Ikuti'; + + @override + String get artist_url_copied => 'URL artis telah disalin'; + + @override + String added_to_queue(Object tracks) { + return 'Menambah trek $tracks ke antrean'; + } + + @override + String get filter_albums => 'Urutkan album...'; + + @override + String get synced => 'Disinkronkan'; + + @override + String get plain => 'Normal'; + + @override + String get shuffle => 'Acak'; + + @override + String get search_tracks => 'Cari trek...'; + + @override + String get released => 'Dirilis'; + + @override + String error(Object error) { + return 'Kesalahan $error'; + } + + @override + String get title => 'Judul'; + + @override + String get time => 'Waktu'; + + @override + String get more_actions => 'Tindakan Lainnya'; + + @override + String download_count(Object count) { + return 'Unduhan ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Menambah ($count) ke Daftar Putar'; + } + + @override + String add_count_to_queue(Object count) { + return 'Menambah ($count) ke Antrian'; + } + + @override + String play_count_next(Object count) { + return 'Mainkan ($count) selanjutnya'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return '$data telah disalin'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Menambah $track ke Daftar Putar berikut'; + } + + @override + String get add => 'Tambah'; + + @override + String added_track_to_queue(Object track) { + return 'Menambah $track ke antrian'; + } + + @override + String get add_to_queue => 'Tambah ke antrian'; + + @override + String track_will_play_next(Object track) { + return '$track akan diputar berikutnya'; + } + + @override + String get play_next => 'Mainkan selanjutnya'; + + @override + String removed_track_from_queue(Object track) { + return 'Menghapus $track dari antrian'; + } + + @override + String get remove_from_queue => 'Hapus dari antrian'; + + @override + String get remove_from_favorites => 'Hapus dari favorit'; + + @override + String get save_as_favorite => 'Simpan sebagai favorit'; + + @override + String get add_to_playlist => 'Tambah ke daftar putar'; + + @override + String get remove_from_playlist => 'Hapus dari daftar putar'; + + @override + String get add_to_blacklist => 'Tambah ke daftar hitam'; + + @override + String get remove_from_blacklist => 'Hapus dari daftar hitam'; + + @override + String get share => 'Bagikan'; + + @override + String get mini_player => 'Pemutar Mini'; + + @override + String get slide_to_seek => 'Geser untuk maju atau mundur'; + + @override + String get shuffle_playlist => 'Acak daftar putar'; + + @override + String get unshuffle_playlist => 'Batalkan pengacakan daftar putar'; + + @override + String get previous_track => 'Lagu sebelumnya'; + + @override + String get next_track => 'Lagu berikutnya'; + + @override + String get pause_playback => 'Jeda Pemutaran'; + + @override + String get resume_playback => 'Lanjutkan Pemutaran'; + + @override + String get loop_track => 'Ulangi Pemutaran'; + + @override + String get no_loop => 'No loop'; + + @override + String get repeat_playlist => 'Ulangi daftar putar'; + + @override + String get queue => 'Antrian'; + + @override + String get alternative_track_sources => 'Sumber trek alternatif'; + + @override + String get download_track => 'Unduh lagu'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks trek dalam antrian'; + } + + @override + String get clear_all => 'Bersihkan semua'; + + @override + String get show_hide_ui_on_hover => + 'Tampil/Sembunyikan UI saat mengarahkan kursor'; + + @override + String get always_on_top => 'Selalu di atas'; + + @override + String get exit_mini_player => 'Keluar Pemutar Mini'; + + @override + String get download_location => 'Lokasi unduhan'; + + @override + String get local_library => 'Perpustakaan lokal'; + + @override + String get add_library_location => 'Tambahkan ke perpustakaan'; + + @override + String get remove_library_location => 'Hapus dari perpustakaan'; + + @override + String get account => 'Akun'; + + @override + String get logout => 'Keluar'; + + @override + String get logout_of_this_account => 'Keluar dari akun'; + + @override + String get language_region => 'Bahasa & Wilayah'; + + @override + String get language => 'Bahasa'; + + @override + String get system_default => 'Bawaan Sistem'; + + @override + String get market_place_region => 'Wilayah Pasar'; + + @override + String get recommendation_country => 'Negara Rekomendasi'; + + @override + String get appearance => 'Tampilan'; + + @override + String get layout_mode => 'Mode Tata Letak'; + + @override + String get override_layout_settings => + 'Ganti pengaturan mode tata letak responsif'; + + @override + String get adaptive => 'Adaptif'; + + @override + String get compact => 'Ringkas'; + + @override + String get extended => 'Diperluas'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Gelap'; + + @override + String get light => 'Terang'; + + @override + String get system => 'Sistem'; + + @override + String get accent_color => 'Warna Aksen'; + + @override + String get sync_album_color => 'Sinkronkan warna album'; + + @override + String get sync_album_color_description => + 'Menggunakan warna dominan sampul album sebagai warna aksen'; + + @override + String get playback => 'Pemutaran'; + + @override + String get audio_quality => 'Kualitas Suara'; + + @override + String get high => 'Tinggi'; + + @override + String get low => 'Rendah'; + + @override + String get pre_download_play => 'Unduh dan putar'; + + @override + String get pre_download_play_description => + 'Daripada streaming audio, unduh byte dan mainkan (Direkomendasikan untuk pengguna bandwidth yang lebih tinggi)'; + + @override + String get skip_non_music => 'Lewati segmen non-musik (SponsorBlock)'; + + @override + String get blacklist_description => 'Lagu dan artis di daftar hitam'; + + @override + String get wait_for_download_to_finish => + 'Tunggu hingga unduhan saat ini selesai'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Tutup Perilaku'; + + @override + String get close => 'Tutup'; + + @override + String get minimize_to_tray => 'Perkecil ke tray'; + + @override + String get show_tray_icon => 'Tampilkan tray ikon sistem'; + + @override + String get about => 'Tentang'; + + @override + String get u_love_spotube => 'Kami tahu Anda menyukai Spotube'; + + @override + String get check_for_updates => 'Periksa pembaruan'; + + @override + String get about_spotube => 'Tentang Spotube'; + + @override + String get blacklist => 'Daftar Hitam'; + + @override + String get please_sponsor => 'Silakan Sponsor/Menyumbang'; + + @override + String get spotube_description => + 'Spotube, klien Spotify yang ringan, lintas platform, dan gratis untuk semua'; + + @override + String get version => 'Versi'; + + @override + String get build_number => 'Nomor Pembuatan'; + + @override + String get founder => 'Pendiri'; + + @override + String get repository => 'Repositori'; + + @override + String get bug_issues => 'Bug+Masalah'; + + @override + String get made_with => 'Dibuat dengan ❤️ di Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Lisensi'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Jangan khawatir, kredensial Anda tidak akan dikumpulkan atau dibagikan kepada siapa pun'; + + @override + String get know_how_to_login => 'Tidak tahu bagaimana melakukan ini?'; + + @override + String get follow_step_by_step_guide => 'Ikuti panduan Langkah demi Langkah'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => 'Silakan isi semua kolom'; + + @override + String get submit => 'Kirim'; + + @override + String get exit => 'Keluar'; + + @override + String get previous => 'Sebelumnya'; + + @override + String get next => 'Berikutnya'; + + @override + String get done => 'Selesai'; + + @override + String get step_1 => 'Langkah 1'; + + @override + String get first_go_to => 'Pertama, Pergi ke'; + + @override + String get something_went_wrong => 'Terjadi kesalahan'; + + @override + String get piped_instance => 'Piped Server Instance'; + + @override + String get piped_description => + 'The Piped server instance untuk digunakan sebagai pencocokan trek'; + + @override + String get piped_warning => + 'Beberapa di antaranya mungkin tidak berfungsi dengan baik. Jadi gunakan dengan risiko Anda sendiri'; + + @override + String get invidious_instance => 'Invidious Server Instance'; + + @override + String get invidious_description => + 'The Invidious server instance to use for track matching'; + + @override + String get invidious_warning => + 'Some of them might not work well. So use at your own risk'; + + @override + String get generate => 'Generate'; + + @override + String track_exists(Object track) { + return 'Lagu $track sudah ada'; + } + + @override + String get replace_downloaded_tracks => 'Ganti semua trek yang diunduh'; + + @override + String get skip_download_tracks => + 'Lewati pengunduhan semua trek yang diunduh'; + + @override + String get do_you_want_to_replace => + 'Apakah Anda ingin mengganti track yang ada?'; + + @override + String get replace => 'Ganti'; + + @override + String get skip => 'Lewati'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Pilih hingga $count $type'; + } + + @override + String get select_genres => 'Pilih Genre'; + + @override + String get add_genres => 'Tambah Genre'; + + @override + String get country => 'Negara'; + + @override + String get number_of_tracks_generate => 'Jumlah trek yang akan dihasilkan'; + + @override + String get acousticness => 'Akustik'; + + @override + String get danceability => 'Menari'; + + @override + String get energy => 'Energi'; + + @override + String get instrumentalness => 'Instrumentalitas'; + + @override + String get liveness => 'Kehidupan'; + + @override + String get loudness => 'Kekerasan'; + + @override + String get speechiness => 'Berbicara'; + + @override + String get valence => 'Valensi'; + + @override + String get popularity => 'Popularitas'; + + @override + String get key => 'Kunci'; + + @override + String get duration => 'Durasi (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'Tanda Tangan Waktu'; + + @override + String get short => 'Pendek'; + + @override + String get medium => 'Sedang'; + + @override + String get long => 'Panjang'; + + @override + String get min => 'Minimal'; + + @override + String get max => 'Maksimal'; + + @override + String get target => 'Target'; + + @override + String get moderate => 'Sedang'; + + @override + String get deselect_all => 'Batalkan Semua'; + + @override + String get select_all => 'Pilih Semua'; + + @override + String get are_you_sure => 'Anda yakin?'; + + @override + String get generating_playlist => 'Menghasilkan daftar putar khusus Anda...'; + + @override + String selected_count_tracks(Object count) { + return '$count lagu yang dipilih'; + } + + @override + String get download_warning => + 'Jika Anda mengunduh semua Lagu secara massal, Anda jelas membajak Musik & menyebabkan kerusakan pada masyarakat kreatif Musik. Saya harap Anda menyadari hal ini. Selalu berusaha menghormati & mendukung kerja keras Artis'; + + @override + String get download_ip_ban_warning => + 'BTW, IP Anda bisa diblokir di YouTube karena permintaan unduhan yang berlebihan dari biasanya. Blokir IP berarti Anda tidak dapat menggunakan YouTube (meskipun Anda masuk) setidaknya selama 2-3 bulan dari perangkat IP tersebut. Dan Spotube tidak bertanggung jawab jika hal ini terjadi'; + + @override + String get by_clicking_accept_terms => + 'Dengan mengklik \'terima\' Anda menyetujui ketentuan berikut:'; + + @override + String get download_agreement_1 => + 'Saya tahu saya membajak Musik. Saya buruk'; + + @override + String get download_agreement_2 => + 'Saya akan mendukung Artis di mana pun saya bisa dan saya melakukan ini hanya karena saya tidak punya uang untuk membeli karya seni mereka'; + + @override + String get download_agreement_3 => + 'Saya sepenuhnya menyadari bahwa IP saya dapat diblokir di YouTube & saya tidak menganggap Spotube atau pemilik/kontributornya bertanggung jawab atas kecelakaan apa pun yang disebabkan oleh tindakan saya saat ini'; + + @override + String get decline => 'Menolak'; + + @override + String get accept => 'Setuju'; + + @override + String get details => 'Detail'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Channel'; + + @override + String get likes => 'Suka'; + + @override + String get dislikes => 'Tidak Suka'; + + @override + String get views => 'Dilihat'; + + @override + String get streamUrl => 'URL Stream'; + + @override + String get stop => 'Berhenti'; + + @override + String get sort_newest => 'Urutkan yang baru ditambah'; + + @override + String get sort_oldest => 'Urutkan yang paling lama ditambah'; + + @override + String get sleep_timer => 'Pengatur Waktu Tidur'; + + @override + String mins(Object minutes) { + return '$minutes Menit'; + } + + @override + String hours(Object hours) { + return '$hours Jam'; + } + + @override + String hour(Object hours) { + return '$hours Jam'; + } + + @override + String get custom_hours => 'Jam Kostum'; + + @override + String get logs => 'Log'; + + @override + String get developers => 'Pengembang'; + + @override + String get not_logged_in => 'Anda belum masuk'; + + @override + String get search_mode => 'Mode Pencarian'; + + @override + String get audio_source => 'Sumber Suara'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => 'Gagal mengenkripsi'; + + @override + String get encryption_failed_warning => + 'Spotube menggunakan enkripsi untuk menyimpan data Anda dengan aman. Namun gagal melakukannya. Jadi itu akan kembali ke penyimpanan yang tidak aman\nJika Anda menggunakan linux, pastikan Anda telah menginstal layanan rahasia (gnome-keyring, kde-wallet, keepassxc, dll)'; + + @override + String get querying_info => 'Mencari informasi...'; + + @override + String get piped_api_down => 'Piped API tidak aktif'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Piped Instance $pipedInstance saat ini tidak aktif\n\nUbah instance atau ubah \'jenis API\' menjadi API YouTube resmi\n\nPastikan untuk memulai ulang aplikasi setelah perubahan'; + } + + @override + String get you_are_offline => 'Anda sedang offline'; + + @override + String get connection_restored => 'Koneksi internet Anda telah pulih'; + + @override + String get use_system_title_bar => 'Gunakan bilah judul sistem'; + + @override + String get crunching_results => 'Mengolah hasil...'; + + @override + String get search_to_get_results => 'Cari untuk mendapatkan hasil'; + + @override + String get use_amoled_mode => 'Tema gelap gulita'; + + @override + String get pitch_dark_theme => 'Mode AMOLED'; + + @override + String get normalize_audio => 'Normalisasi audio'; + + @override + String get change_cover => 'Ganti sampul'; + + @override + String get add_cover => 'Tambah sampul'; + + @override + String get restore_defaults => 'Kembalikan semula'; + + @override + String get download_music_format => 'Format unduh musik'; + + @override + String get streaming_music_format => 'Format streaming musik'; + + @override + String get download_music_quality => 'Kualitas unduh musik'; + + @override + String get streaming_music_quality => 'Kualitas streaming musik'; + + @override + String get login_with_lastfm => 'Masuk dengan Last.fm'; + + @override + String get connect => 'Hubungkan'; + + @override + String get disconnect_lastfm => 'Memutuskan Last.fm'; + + @override + String get disconnect => 'Memutuskan'; + + @override + String get username => 'Username'; + + @override + String get password => 'Password'; + + @override + String get login => 'Masuk'; + + @override + String get login_with_your_lastfm => 'Masuk dengan Last.fm Anda'; + + @override + String get scrobble_to_lastfm => 'Scrobble ke Last.fm'; + + @override + String get go_to_album => 'Pergi ke Album'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'Lihat Semua'; + + @override + String get genres => 'Genre'; + + @override + String get explore_genres => 'Jelajahi Genre'; + + @override + String get friends => 'Daftar Teman'; + + @override + String get no_lyrics_available => + 'Maaf, tidak dapat menemukan lirik untuk lagu ini'; + + @override + String get start_a_radio => 'Putar Radio'; + + @override + String get how_to_start_radio => 'Bagaimana Anda ingin memutar radio?'; + + @override + String get replace_queue_question => + 'Apakah Anda ingin mengganti antrean saat ini atau menambahkannya?'; + + @override + String get endless_playback => 'Pemutaran Tanpa Akhir'; + + @override + String get delete_playlist => 'Hapus Daftar Putar'; + + @override + String get delete_playlist_confirmation => + 'Anda yakin ingin menghapus daftar putar ini?'; + + @override + String get local_tracks => 'Trek Lokal'; + + @override + String get local_tab => 'Lokal'; + + @override + String get song_link => 'Tautan Lagu'; + + @override + String get skip_this_nonsense => 'Lewati omong kosong ini'; + + @override + String get freedom_of_music => '“Kebebasan Musik”'; + + @override + String get freedom_of_music_palm => + '“Kebebasan Musik di telapak tangan Anda”'; + + @override + String get get_started => 'Mari kita mulai'; + + @override + String get youtube_source_description => + 'Direkomendasikan dan berfungsi paling baik.'; + + @override + String get piped_source_description => + 'Merasa bebas? Sama seperti YouTube tetapi banyak yang gratis.'; + + @override + String get jiosaavn_source_description => + 'Terbaik untuk wilayah Asia Selatan.'; + + @override + String get invidious_source_description => + 'Similar to Piped but with higher availability.'; + + @override + String highest_quality(Object quality) { + return 'Kualitas Terbaik: $quality'; + } + + @override + String get select_audio_source => 'Pilih Sumber Suara'; + + @override + String get endless_playback_description => + 'Tambahkan lagu baru secara otomatis\nke akhir antrean'; + + @override + String get choose_your_region => 'Pilih wilayah Anda'; + + @override + String get choose_your_region_description => + 'Ini akan membantu Spotube menampilkan konten yang tepat\nuntuk lokasi Anda.'; + + @override + String get choose_your_language => 'Pilih bahasa Anda'; + + @override + String get help_project_grow => 'Bantu proyek ini berkembang'; + + @override + String get help_project_grow_description => + 'Spotube adalah proyek sumber terbuka. Anda dapat membantu proyek ini berkembang dengan berkontribusi pada proyek, melaporkan bug, atau menyarankan fitur baru.'; + + @override + String get contribute_on_github => 'Berkontribusi di GitHub'; + + @override + String get donate_on_open_collective => 'Donasi di Open Collective'; + + @override + String get browse_anonymously => 'Jelajahi Secara Anonim'; + + @override + String get enable_connect => 'Aktifkan Hubungkan'; + + @override + String get enable_connect_description => + 'Kontrol Spotube dari perangkat lain'; + + @override + String get devices => 'Perangkat'; + + @override + String get select => 'Pilih'; + + @override + String connect_client_alert(Object client) { + return 'Anda dikendalikan oleh $client'; + } + + @override + String get this_device => 'Perangkat Ini'; + + @override + String get remote => 'Remot'; + + @override + String get stats => 'Statistik'; + + @override + String and_n_more(Object count) { + return 'dan $count lainnya'; + } + + @override + String get recently_played => 'Baru saja diputar'; + + @override + String get browse_more => 'Telusuri lebih banyak'; + + @override + String get no_title => 'Tanpa judul'; + + @override + String get not_playing => 'Tidak diputar'; + + @override + String get epic_failure => 'Kegagalan epik!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Menambahkan $tracks_length trek ke antrean'; + } + + @override + String get spotube_has_an_update => 'Spotube memiliki pembaruan'; + + @override + String get download_now => 'Unduh sekarang'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum telah dirilis'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version telah dirilis'; + } + + @override + String get read_the_latest => 'Baca yang terbaru '; + + @override + String get release_notes => 'catatan rilis'; + + @override + String get pick_color_scheme => 'Pilih skema warna'; + + @override + String get save => 'Simpan'; + + @override + String get choose_the_device => 'Pilih perangkat:'; + + @override + String get multiple_device_connected => + 'Beberapa perangkat terhubung.\nPilih perangkat tempat Anda ingin melakukan tindakan ini'; + + @override + String get nothing_found => 'Tidak ditemukan apa pun'; + + @override + String get the_box_is_empty => 'Kotak kosong'; + + @override + String get top_artists => 'Artis Teratas'; + + @override + String get top_albums => 'Album Teratas'; + + @override + String get this_week => 'Minggu ini'; + + @override + String get this_month => 'Bulan ini'; + + @override + String get last_6_months => '6 bulan terakhir'; + + @override + String get this_year => 'Tahun ini'; + + @override + String get last_2_years => '2 tahun terakhir'; + + @override + String get all_time => 'Sepanjang waktu'; + + @override + String powered_by_provider(Object providerName) { + return 'Didukung oleh $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Pengikut'; + + @override + String get birthday => 'Ulang Tahun'; + + @override + String get subscription => 'Langganan'; + + @override + String get not_born => 'Belum lahir'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profil'; + + @override + String get no_name => 'Tanpa nama'; + + @override + String get edit => 'Edit'; + + @override + String get user_profile => 'Profil pengguna'; + + @override + String count_plays(Object count) { + return '$count pemutaran'; + } + + @override + String get streaming_fees_hypothetical => 'Biaya streaming (hipotetis)'; + + @override + String get minutes_listened => 'Menit didengarkan'; + + @override + String get streamed_songs => 'Lagu yang disiarkan'; + + @override + String count_streams(Object count) { + return '$count streams'; + } + + @override + String get owned_by_you => 'Dimiliki oleh Anda'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl disalin ke clipboard'; + } + + @override + String get hipotetical_calculation => + '*Ini dihitung berdasarkan pembayaran rata-rata per streaming dari platform streaming musik online sebesar \$0,003 hingga \$0,005. Ini adalah perhitungan hipotetis untuk memberikan wawasan kepada pengguna tentang seberapa banyak yang akan mereka bayarkan kepada artis jika mereka mendengarkan lagu mereka di platform streaming musik yang berbeda.'; + + @override + String count_mins(Object minutes) { + return '$minutes menit'; + } + + @override + String get summary_minutes => 'menit'; + + @override + String get summary_listened_to_music => 'Mendengarkan musik'; + + @override + String get summary_songs => 'lagu'; + + @override + String get summary_streamed_overall => 'Disiarkan secara keseluruhan'; + + @override + String get summary_owed_to_artists => 'Terhutang kepada artis\nBulan ini'; + + @override + String get summary_artists => 'artis'; + + @override + String get summary_music_reached_you => 'Musik mencapai Anda'; + + @override + String get summary_full_albums => 'album lengkap'; + + @override + String get summary_got_your_love => 'Mendapatkan cinta Anda'; + + @override + String get summary_playlists => 'daftar putar'; + + @override + String get summary_were_on_repeat => 'Sedang diulang'; + + @override + String total_money(Object money) { + return 'Total $money'; + } + + @override + String get webview_not_found => 'Webview tidak ditemukan'; + + @override + String get webview_not_found_description => + 'Tidak ada runtime Webview yang diinstal di perangkat Anda.\nJika sudah diinstal, pastikan itu ada di environment PATH\n\nSetelah diinstal, restart aplikasi'; + + @override + String get unsupported_platform => 'Platform tidak didukung'; + + @override + String get cache_music => 'Cache music'; + + @override + String get open => 'Open'; + + @override + String get cache_folder => 'Cache folder'; + + @override + String get export => 'Export'; + + @override + String get clear_cache => 'Clear cache'; + + @override + String get clear_cache_confirmation => 'Do you want to clear the cache?'; + + @override + String get export_cache_files => 'Export Cached Files'; + + @override + String found_n_files(Object count) { + return 'Found $count files'; + } + + @override + String get export_cache_confirmation => + 'Do you want to export these files to'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Exported $filesExported out of $files files'; + } + + @override + String get undo => 'Undo'; + + @override + String get download_all => 'Download all'; + + @override + String get add_all_to_playlist => 'Add all to playlist'; + + @override + String get add_all_to_queue => 'Add all to queue'; + + @override + String get play_all_next => 'Play all next'; + + @override + String get pause => 'Pause'; + + @override + String get view_all => 'View all'; + + @override + String get no_tracks_added_yet => + 'Looks like you haven\'t added any tracks yet'; + + @override + String get no_tracks => 'Looks like there are no tracks here'; + + @override + String get no_tracks_listened_yet => + 'Looks like you haven\'t listened to anything yet'; + + @override + String get not_following_artists => 'You\'re not following any artists'; + + @override + String get no_favorite_albums_yet => + 'Looks like you haven\'t added any albums to your favorites yet'; + + @override + String get no_logs_found => 'No logs found'; + + @override + String get youtube_engine => 'YouTube Engine'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine is not installed'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine is not installed in your system.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Make sure it\'s available in the PATH variable or\nset the absolute path to the $engine executable below'; + } + + @override + String get youtube_engine_unix_issue_message => + 'In macOS/Linux/unix like OS\'s, setting path on .zshrc/.bashrc/.bash_profile etc. won\'t work.\nYou need to set the path in the shell configuration file'; + + @override + String get download => 'Download'; + + @override + String get file_not_found => 'File not found'; + + @override + String get custom => 'Custom'; + + @override + String get add_custom_url => 'Add custom URL'; + + @override + String get edit_port => 'Edit port'; + + @override + String get port_helper_msg => + 'Default adalah -1 yang menunjukkan angka acak. Jika Anda telah mengonfigurasi firewall, disarankan untuk mengatur ini.'; + + @override + String connect_request(Object client) { + return 'Izinkan $client untuk terhubung?'; + } + + @override + String get connection_request_denied => + 'Koneksi ditolak. Pengguna menolak akses.'; + + @override + String get an_error_occurred => 'Terjadi kesalahan'; + + @override + String get copy_to_clipboard => 'Salin ke papan klip'; + + @override + String get view_logs => 'Lihat log'; + + @override + String get retry => 'Coba lagi'; + + @override + String get no_default_metadata_provider_selected => + 'Anda belum mengatur penyedia metadata default'; + + @override + String get manage_metadata_providers => 'Kelola penyedia metadata'; + + @override + String get open_link_in_browser => 'Buka Tautan di Peramban?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Apakah Anda ingin membuka tautan berikut'; + + @override + String get unsafe_url_warning => + 'Tidak aman untuk membuka tautan dari sumber yang tidak tepercaya. Berhati-hatilah!\nAnda juga dapat menyalin tautan ke papan klip Anda.'; + + @override + String get copy_link => 'Salin Tautan'; + + @override + String get building_your_timeline => + 'Membangun garis waktu Anda berdasarkan riwayat mendengarkan Anda...'; + + @override + String get official => 'Resmi'; + + @override + String author_name(Object author) { + return 'Penulis: $author'; + } + + @override + String get third_party => 'Pihak ketiga'; + + @override + String get plugin_requires_authentication => 'Plugin memerlukan otentikasi'; + + @override + String get update_available => 'Pembaruan tersedia'; + + @override + String get supports_scrobbling => 'Mendukung scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Plugin ini scrobble musik Anda untuk menghasilkan riwayat mendengarkan Anda.'; + + @override + String get default_metadata_source => 'Sumber metadata default'; + + @override + String get set_default_metadata_source => 'Atur sumber metadata default'; + + @override + String get default_audio_source => 'Sumber audio default'; + + @override + String get set_default_audio_source => 'Atur sumber audio default'; + + @override + String get set_default => 'Atur sebagai bawaan'; + + @override + String get support => 'Dukungan'; + + @override + String get support_plugin_development => 'Dukung pengembangan plugin'; + + @override + String can_access_name_api(Object name) { + return '- Dapat mengakses API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Apakah Anda ingin menginstal plugin ini?'; + + @override + String get third_party_plugin_warning => + 'Plugin ini berasal dari repositori pihak ketiga. Pastikan Anda memercayai sumbernya sebelum menginstal.'; + + @override + String get author => 'Penulis'; + + @override + String get this_plugin_can_do_following => + 'Plugin ini dapat melakukan hal berikut'; + + @override + String get install => 'Instal'; + + @override + String get install_a_metadata_provider => 'Instal Penyedia Metadata'; + + @override + String get no_tracks_playing => 'Tidak ada Lagu yang sedang diputar saat ini'; + + @override + String get synced_lyrics_not_available => + 'Lirik tersinkronisasi tidak tersedia untuk lagu ini. Silakan gunakan tab'; + + @override + String get plain_lyrics => 'Lirik Polos'; + + @override + String get tab_instead => 'sebagai gantinya.'; + + @override + String get disclaimer => 'Penafian'; + + @override + String get third_party_plugin_dmca_notice => + 'Tim Spotube tidak bertanggung jawab (termasuk hukum) atas plugin \"Pihak ketiga\" mana pun.\nSilakan gunakan dengan risiko Anda sendiri. Untuk bug/masalah apa pun, silakan laporkan ke repositori plugin.\n\nJika ada plugin \"Pihak ketiga\" yang melanggar ToS/DMCA dari layanan/entitas hukum mana pun, silakan minta penulis plugin \"Pihak ketiga\" atau platform hosting, mis. GitHub/Codeberg, untuk mengambil tindakan. Yang tercantum di atas (berlabel \"Pihak ketiga\") adalah semua plugin publik/yang dikelola oleh komunitas. Kami tidak mengkurasi mereka, jadi kami tidak dapat mengambil tindakan apa pun terhadap mereka.\n\n'; + + @override + String get input_does_not_match_format => + 'Masukan tidak cocok dengan format yang diperlukan'; + + @override + String get plugins => 'Plugin'; + + @override + String get paste_plugin_download_url => + 'Tempel url unduhan atau url repo GitHub/Codeberg atau tautan langsung ke file .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Unduh dan instal plugin dari url'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Gagal menambahkan plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'Unggah plugin dari file'; + + @override + String get installed => 'Terinstal'; + + @override + String get available_plugins => 'Plugin yang tersedia'; + + @override + String get configure_plugins => + 'Konfigurasi plugin penyedia metadata dan sumber audio Anda sendiri'; + + @override + String get audio_scrobblers => 'Scrobblers Audio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Sumber: '; + + @override + String get uncompressed => 'Tidak terkompresi'; + + @override + String get dab_music_source_description => + 'Untuk audiophile. Menyediakan aliran audio berkualitas tinggi/tanpa kehilangan. Pencocokkan trek yang akurat berdasarkan ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_it.dart b/lib/l10n/generated/app_localizations_it.dart new file mode 100644 index 00000000..f2dfa5ed --- /dev/null +++ b/lib/l10n/generated/app_localizations_it.dart @@ -0,0 +1,1572 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Italian (`it`). +class AppLocalizationsIt extends AppLocalizations { + AppLocalizationsIt([String locale = 'it']) : super(locale); + + @override + String get guest => 'Ospite'; + + @override + String get browse => 'Sfoglia'; + + @override + String get search => 'Cerca'; + + @override + String get library => 'Libreria'; + + @override + String get lyrics => 'Testi'; + + @override + String get settings => 'Impostazioni'; + + @override + String get genre_categories_filter => 'Filtra categorie e generi...'; + + @override + String get genre => 'Genere'; + + @override + String get personalized => 'Personalizzato'; + + @override + String get featured => 'In evidenza'; + + @override + String get new_releases => 'Novità'; + + @override + String get songs => 'Canzoni'; + + @override + String playing_track(Object track) { + return 'Riproduzione $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Questo cancellerà la coda corrente. $track_length tracce saranno rimosse\nVuoi continuare?'; + } + + @override + String get load_more => 'Carica altro'; + + @override + String get playlists => 'Playlist'; + + @override + String get artists => 'Artisti'; + + @override + String get albums => 'Album'; + + @override + String get tracks => 'Tracce'; + + @override + String get downloads => 'Downloads'; + + @override + String get filter_playlists => 'Filtra le tue playlist...'; + + @override + String get liked_tracks => 'Tracce piaciute'; + + @override + String get liked_tracks_description => 'Tutte le tracce piaciute'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Crea una playlist'; + + @override + String get update_playlist => 'Aggiorna playlist'; + + @override + String get create => 'Crea'; + + @override + String get cancel => 'Annulla'; + + @override + String get update => 'Aggiorna'; + + @override + String get playlist_name => 'Nome Playlist'; + + @override + String get name_of_playlist => 'Nome della playlist'; + + @override + String get description => 'Descrizione'; + + @override + String get public => 'Pubblico'; + + @override + String get collaborative => 'Collaborativo'; + + @override + String get search_local_tracks => 'Cerca tracce locali...'; + + @override + String get play => 'Riproduci'; + + @override + String get delete => 'Cancella'; + + @override + String get none => 'Nessuno'; + + @override + String get sort_a_z => 'Ordina dalla A-Z'; + + @override + String get sort_z_a => 'Ordina dalla Z-A'; + + @override + String get sort_artist => 'Ordina per Artista'; + + @override + String get sort_album => 'Ordina per Album'; + + @override + String get sort_duration => 'Ordina per Durata'; + + @override + String get sort_tracks => 'Ordina tracce'; + + @override + String currently_downloading(Object tracks_length) { + return 'Attualmente in Download ($tracks_length)'; + } + + @override + String get cancel_all => 'Annulla Tutto'; + + @override + String get filter_artist => 'Filtra artisti...'; + + @override + String followers(Object followers) { + return '$followers Seguaci'; + } + + @override + String get add_artist_to_blacklist => 'Aggiungi artista alla lista nera'; + + @override + String get top_tracks => 'Tracce Top'; + + @override + String get fans_also_like => 'Ai fan piace anche'; + + @override + String get loading => 'Caricamento...'; + + @override + String get artist => 'Artista'; + + @override + String get blacklisted => 'In lista nera'; + + @override + String get following => 'Seguendo'; + + @override + String get follow => 'Segui'; + + @override + String get artist_url_copied => 'URL artista copiato negli appunti'; + + @override + String added_to_queue(Object tracks) { + return 'Aggiunto $tracks tracce alla coda'; + } + + @override + String get filter_albums => 'Filtra album...'; + + @override + String get synced => 'Sincronizzato'; + + @override + String get plain => 'Semplice'; + + @override + String get shuffle => 'Casuale'; + + @override + String get search_tracks => 'Cerca tracce...'; + + @override + String get released => 'Rilasciato'; + + @override + String error(Object error) { + return 'Errore $error'; + } + + @override + String get title => 'Titolo'; + + @override + String get time => 'Durata'; + + @override + String get more_actions => 'Più azioni'; + + @override + String download_count(Object count) { + return 'Scaricato ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Aggiungi ($count) alla playlist'; + } + + @override + String add_count_to_queue(Object count) { + return 'Aggiungi ($count) alla Coda'; + } + + @override + String play_count_next(Object count) { + return 'Riproduci ($count) prossime'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return 'Copiato $data negli appunti'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Aggiungi $track nelle seguenti Playlist'; + } + + @override + String get add => 'Aggiungi'; + + @override + String added_track_to_queue(Object track) { + return 'Aggiunto $track alla coda'; + } + + @override + String get add_to_queue => 'Aggiungi alla coda'; + + @override + String track_will_play_next(Object track) { + return 'in seguito sarà riprodotta $track'; + } + + @override + String get play_next => 'Riproduci prossimo'; + + @override + String removed_track_from_queue(Object track) { + return 'Rimosso $track dalla coda'; + } + + @override + String get remove_from_queue => 'Rimuovi dalla coda'; + + @override + String get remove_from_favorites => 'Rimuovi dai preferiti'; + + @override + String get save_as_favorite => 'Salva come preferito'; + + @override + String get add_to_playlist => 'Aggiungi alla playlist'; + + @override + String get remove_from_playlist => 'Rimuovi dalla playlist'; + + @override + String get add_to_blacklist => 'Aggiungi alla blacklist'; + + @override + String get remove_from_blacklist => 'Rimuovi dalla blacklist'; + + @override + String get share => 'Condividi'; + + @override + String get mini_player => 'Mini Riproduttore'; + + @override + String get slide_to_seek => 'Scorri per cercare avanti o indietro'; + + @override + String get shuffle_playlist => 'Playlist casuale'; + + @override + String get unshuffle_playlist => 'Ordina playlist'; + + @override + String get previous_track => 'Traccia precedente'; + + @override + String get next_track => 'Traccia successiva'; + + @override + String get pause_playback => 'Pausa Playback'; + + @override + String get resume_playback => 'Riprendi Playback'; + + @override + String get loop_track => 'Cicla traccia'; + + @override + String get no_loop => 'Nessun ciclo'; + + @override + String get repeat_playlist => 'Ripeti playlist'; + + @override + String get queue => 'Coda'; + + @override + String get alternative_track_sources => 'Sorgenti traccia alternative'; + + @override + String get download_track => 'Scarica traccia'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks tracce in coda'; + } + + @override + String get clear_all => 'Cancella tutto'; + + @override + String get show_hide_ui_on_hover => 'Mostra/Nascondi UI al passaggio'; + + @override + String get always_on_top => 'Sempre in cima'; + + @override + String get exit_mini_player => 'Esci da Mini player'; + + @override + String get download_location => 'Cartella di scarico'; + + @override + String get local_library => 'Biblioteca locale'; + + @override + String get add_library_location => 'Aggiungi alla biblioteca'; + + @override + String get remove_library_location => 'Rimuovi dalla biblioteca'; + + @override + String get account => 'Account'; + + @override + String get logout => 'Esci'; + + @override + String get logout_of_this_account => 'Esci da questo account'; + + @override + String get language_region => 'Lingua & Regione'; + + @override + String get language => 'Lingua'; + + @override + String get system_default => 'Default sistema'; + + @override + String get market_place_region => 'Regione del mercato'; + + @override + String get recommendation_country => 'Paese Raccomandato'; + + @override + String get appearance => 'Aspetto'; + + @override + String get layout_mode => 'Modalità Layout'; + + @override + String get override_layout_settings => + 'Sovrascrivi le impostazioni del layout responsivo'; + + @override + String get adaptive => 'Adattiva'; + + @override + String get compact => 'Compatta'; + + @override + String get extended => 'Estesa'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Scuro'; + + @override + String get light => 'Chiaro'; + + @override + String get system => 'Sistema'; + + @override + String get accent_color => 'Colore accento'; + + @override + String get sync_album_color => 'Syncronizza colore album'; + + @override + String get sync_album_color_description => + 'Usa il colore dominante della copertina dell\'album come colore accento'; + + @override + String get playback => 'Riproduzione'; + + @override + String get audio_quality => 'Qualità Audio'; + + @override + String get high => 'Alta'; + + @override + String get low => 'Bassa'; + + @override + String get pre_download_play => 'Pre-scarica e riproduci'; + + @override + String get pre_download_play_description => + 'Anzi che effettuare lo stream dell\'audio, scarica invece i byte e li riproduce (raccomandato per gli utenti con banda più alta)'; + + @override + String get skip_non_music => 'Salta i segmenti non di musica (SponsorBlock)'; + + @override + String get blacklist_description => 'Tracce e artisti in blacklist'; + + @override + String get wait_for_download_to_finish => + 'Prego attendere che lo scaricamento corrente finisca'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Comportamento Chiusura'; + + @override + String get close => 'Chiudi'; + + @override + String get minimize_to_tray => 'Minimizza in tray'; + + @override + String get show_tray_icon => 'Mostra icona in tray di sistema'; + + @override + String get about => 'A proposito di'; + + @override + String get u_love_spotube => 'Sappiamo che ami Spotube'; + + @override + String get check_for_updates => 'Controlla aggiornamenti'; + + @override + String get about_spotube => 'A proposito di Spotube'; + + @override + String get blacklist => 'Blacklist'; + + @override + String get please_sponsor => 'Per favore sponsorizza/dona'; + + @override + String get spotube_description => + 'Spotube, un client spotify gratis per tutti, multipiattaforma e leggero'; + + @override + String get version => 'Versione'; + + @override + String get build_number => 'Numero Build'; + + @override + String get founder => 'Fondatore'; + + @override + String get repository => 'Repository'; + + @override + String get bug_issues => 'Bug+Problemi'; + + @override + String get made_with => 'Fatto con ❤️ in Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licenza'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Non ti preoccupare, le tue credenziali non saranno inviate o condivise con nessuno'; + + @override + String get know_how_to_login => 'Non sai come farlo?'; + + @override + String get follow_step_by_step_guide => 'Segui la guida passo-passo'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Inserire tutti i campi'; + + @override + String get submit => 'Invia'; + + @override + String get exit => 'Esci'; + + @override + String get previous => 'Precedente'; + + @override + String get next => 'Prossimo'; + + @override + String get done => 'Finito'; + + @override + String get step_1 => 'Passo 1'; + + @override + String get first_go_to => 'Prim, vai a'; + + @override + String get something_went_wrong => 'Qualcosa è andato storto'; + + @override + String get piped_instance => 'Istanza Server Piped'; + + @override + String get piped_description => + 'L\'istanza server Piped da usare per il match della tracccia'; + + @override + String get piped_warning => + 'Alcune di queste non funzioneranno benen. Usa quindi a tuo rischio'; + + @override + String get invidious_instance => 'Istanza del server Invidious'; + + @override + String get invidious_description => + 'L\'istanza del server Invidious da utilizzare per il matching delle tracce'; + + @override + String get invidious_warning => + 'Alcuni potrebbero non funzionare bene. Usali a tuo rischio'; + + @override + String get generate => 'Genera'; + + @override + String track_exists(Object track) { + return 'La traccia $track esiste già'; + } + + @override + String get replace_downloaded_tracks => + 'Sostituisci tutte le tracce scaricate'; + + @override + String get skip_download_tracks => + 'Salta lo scaricamento di tutte le tracce scaricate'; + + @override + String get do_you_want_to_replace => + 'Vuoi sovrascrivere la traccia esistente??'; + + @override + String get replace => 'Sovrascrivi'; + + @override + String get skip => 'Salta'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Seleziona fino a $count $type'; + } + + @override + String get select_genres => 'Seleziona Generi'; + + @override + String get add_genres => 'Aggiungi Generi'; + + @override + String get country => 'Paese'; + + @override + String get number_of_tracks_generate => 'Nnumero di tracce da generare'; + + @override + String get acousticness => 'Acustica'; + + @override + String get danceability => 'Ballabilità'; + + @override + String get energy => 'Energia'; + + @override + String get instrumentalness => 'Strumentalità'; + + @override + String get liveness => 'Vitalità'; + + @override + String get loudness => 'Sonorità'; + + @override + String get speechiness => 'Loquacità'; + + @override + String get valence => 'Valenza'; + + @override + String get popularity => 'Popolarità'; + + @override + String get key => 'Chiave'; + + @override + String get duration => 'Durata (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Modo'; + + @override + String get time_signature => 'Indicazione di tempo'; + + @override + String get short => 'Corta'; + + @override + String get medium => 'Media'; + + @override + String get long => 'Lunga'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Obiettivo'; + + @override + String get moderate => 'Moderato'; + + @override + String get deselect_all => 'Deseleziona Tutto'; + + @override + String get select_all => 'Seleziona Tutto'; + + @override + String get are_you_sure => 'Sei certo?'; + + @override + String get generating_playlist => 'Generazione delle tue playlist custom...'; + + @override + String selected_count_tracks(Object count) { + return '$count tracce selezionate'; + } + + @override + String get download_warning => + 'Se scarichi tutte le Tracce in massa stai chiaramente piratando Musica e causando un danno alla società creativa della Musica. Spero che tu sia cosciente di questo. Cerca di rispettare e supportare sempre il duro lavoro degli Artisti'; + + @override + String get download_ip_ban_warning => + 'A proposito, il tuo IP può essere bloccato da YouTube per il numero di richieste di download eccessive rispetto la norma. Il blocco IP significa che non puoi usare YoutTube (anche hai effettuato l\'accesso) per almeno 2-3 mesi dal dispositivo con questo IP. Spotube non ha responsabilità se questo dovesse accadere'; + + @override + String get by_clicking_accept_terms => + 'Cliccando su \'accetta\' concordi con i seguenti termini:'; + + @override + String get download_agreement_1 => + 'So che sto piratando Musica. Sono cattivo'; + + @override + String get download_agreement_2 => + 'Supporterò l\'Artista come potrò e sto facendo questo solo perchè non ho denaro per acquistare il suo prodotto dell\'ingegno'; + + @override + String get download_agreement_3 => + 'Sono completamente cosciente che il mio IP può essere bloccato da YouTube & non riterrò responsabili Spotube o i suoi autori/contributori per ogni inconveniente causato dalla mia azione corrente'; + + @override + String get decline => 'Declino'; + + @override + String get accept => 'Accetto'; + + @override + String get details => 'Dettagli'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Canale'; + + @override + String get likes => 'Mi Piace'; + + @override + String get dislikes => 'Non Mi Piace'; + + @override + String get views => 'Viste'; + + @override + String get streamUrl => 'URL dello streaming'; + + @override + String get stop => 'Stop'; + + @override + String get sort_newest => 'Ordina per nuovi aggiunti'; + + @override + String get sort_oldest => 'Ordina per aggiunta più vecchia'; + + @override + String get sleep_timer => 'Timer Dormire'; + + @override + String mins(Object minutes) { + return '$minutes Minuti'; + } + + @override + String hours(Object hours) { + return '$hours Ore'; + } + + @override + String hour(Object hours) { + return '$hours Ora'; + } + + @override + String get custom_hours => 'Orari Personalizzati'; + + @override + String get logs => 'Log'; + + @override + String get developers => 'Sviluppatori'; + + @override + String get not_logged_in => 'Non hai effettuato l\'accesso'; + + @override + String get search_mode => 'Modalità Ricerca'; + + @override + String get audio_source => 'Fonte audio'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Criptazione fallita'; + + @override + String get encryption_failed_warning => + 'Spotube usa la criptazione per memorizzare in modo sicuro i dati. Ma ha fallito a farlo. Passerà quindi in ripiego alla memorizzazione non siscura\nSe stai usando Linux assicurati di avere un servizio di segretezza installato (gnome-keyring, kde-wallet, keepassxc etc)'; + + @override + String get querying_info => 'Richiesta informazioni...'; + + @override + String get piped_api_down => 'Le Piped API non funzionano'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'L\'istanza di Piped $pipedInstance è correntemente offline\n\nCambia istanza o cambia \'Tipo API\' alle API ufficiali YouTube\n\nAssicurati di riavviare l\'app dopo il cambio'; + } + + @override + String get you_are_offline => 'Sei correntemente offline'; + + @override + String get connection_restored => 'Connessione ad internet ripristinata'; + + @override + String get use_system_title_bar => 'Usa la barra del titolo di sistema'; + + @override + String get crunching_results => 'Elaborazione risultati...'; + + @override + String get search_to_get_results => 'Cerca per ottenere risultati'; + + @override + String get use_amoled_mode => 'Usa modalità AMOLED'; + + @override + String get pitch_dark_theme => 'Tema nero profondo'; + + @override + String get normalize_audio => 'Normalizza audio'; + + @override + String get change_cover => 'Cambia copertina'; + + @override + String get add_cover => 'Aggiungi copertina'; + + @override + String get restore_defaults => 'Ripristina default'; + + @override + String get download_music_format => 'Formato download musica'; + + @override + String get streaming_music_format => 'Formato streaming musica'; + + @override + String get download_music_quality => 'Qualità download musica'; + + @override + String get streaming_music_quality => 'Qualità streaming musica'; + + @override + String get login_with_lastfm => 'Accesso a Last.fm'; + + @override + String get connect => 'Connetti'; + + @override + String get disconnect_lastfm => 'Disconnetti Last.fm'; + + @override + String get disconnect => 'Disconnetti'; + + @override + String get username => 'Nome utente'; + + @override + String get password => 'Password'; + + @override + String get login => 'Accesso'; + + @override + String get login_with_your_lastfm => 'Accedi con il tuo account Last.fm'; + + @override + String get scrobble_to_lastfm => 'Invia a Last.fm'; + + @override + String get go_to_album => 'Vai all\'album'; + + @override + String get discord_rich_presence => 'Presenza ricca di Discord'; + + @override + String get browse_all => 'Esplora tutto'; + + @override + String get genres => 'Generi'; + + @override + String get explore_genres => 'Esplora generi'; + + @override + String get friends => 'Amici'; + + @override + String get no_lyrics_available => + 'Spiacente, impossibile trovare il testo di questa traccia'; + + @override + String get start_a_radio => 'Avvia una Radio'; + + @override + String get how_to_start_radio => 'Come vuoi avviare la radio?'; + + @override + String get replace_queue_question => + 'Vuoi sostituire la coda attuale o aggiungerla?'; + + @override + String get endless_playback => 'Riproduzione Infinita'; + + @override + String get delete_playlist => 'Elimina Playlist'; + + @override + String get delete_playlist_confirmation => + 'Sei sicuro di voler eliminare questa playlist?'; + + @override + String get local_tracks => 'Tracce Locali'; + + @override + String get local_tab => 'Locale'; + + @override + String get song_link => 'Link della Canzone'; + + @override + String get skip_this_nonsense => 'Salta questa sciocchezza'; + + @override + String get freedom_of_music => '“Libertà della Musica”'; + + @override + String get freedom_of_music_palm => + '“Libertà della Musica nel palmo della tua mano”'; + + @override + String get get_started => 'Cominciamo'; + + @override + String get youtube_source_description => 'Consigliato e funziona meglio.'; + + @override + String get piped_source_description => + 'Ti senti libero? Come YouTube ma molto più gratuito.'; + + @override + String get jiosaavn_source_description => + 'Il migliore per la regione dell\'Asia meridionale.'; + + @override + String get invidious_source_description => + 'Simile a Piped ma con maggiore disponibilità.'; + + @override + String highest_quality(Object quality) { + return 'Massima Qualità: $quality'; + } + + @override + String get select_audio_source => 'Seleziona Sorgente Audio'; + + @override + String get endless_playback_description => + 'Aggiungi automaticamente nuove canzoni alla fine della coda'; + + @override + String get choose_your_region => 'Scegli la tua regione'; + + @override + String get choose_your_region_description => + 'Questo aiuterà Spotube a mostrarti il contenuto giusto per la tua posizione.'; + + @override + String get choose_your_language => 'Scegli la tua lingua'; + + @override + String get help_project_grow => 'Aiuta questo progetto a crescere'; + + @override + String get help_project_grow_description => + 'Spotube è un progetto open-source. Puoi aiutare questo progetto a crescere contribuendo al progetto, segnalando bug o suggerendo nuove funzionalità.'; + + @override + String get contribute_on_github => 'Contribuisci su GitHub'; + + @override + String get donate_on_open_collective => 'Dona su Open Collective'; + + @override + String get browse_anonymously => 'Naviga in modo anonimo'; + + @override + String get enable_connect => 'Abilita connessione'; + + @override + String get enable_connect_description => + 'Controlla Spotube da altri dispositivi'; + + @override + String get devices => 'Dispositivi'; + + @override + String get select => 'Seleziona'; + + @override + String connect_client_alert(Object client) { + return 'Stai venendo controllato da $client'; + } + + @override + String get this_device => 'Questo dispositivo'; + + @override + String get remote => 'Remoto'; + + @override + String get stats => 'Statistiche'; + + @override + String and_n_more(Object count) { + return 'e $count in più'; + } + + @override + String get recently_played => 'Riprodotti di recente'; + + @override + String get browse_more => 'Esplora di più'; + + @override + String get no_title => 'Nessun titolo'; + + @override + String get not_playing => 'Non in riproduzione'; + + @override + String get epic_failure => 'Fallimento epico!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Aggiunti $tracks_length brani alla coda'; + } + + @override + String get spotube_has_an_update => 'Spotube ha un aggiornamento'; + + @override + String get download_now => 'Scarica ora'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum è stato rilasciato'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version è stato rilasciato'; + } + + @override + String get read_the_latest => 'Leggi l\'ultimo '; + + @override + String get release_notes => 'note di rilascio'; + + @override + String get pick_color_scheme => 'Scegli uno schema di colori'; + + @override + String get save => 'Salva'; + + @override + String get choose_the_device => 'Scegli il dispositivo:'; + + @override + String get multiple_device_connected => + 'Sono collegati più dispositivi.\nScegli il dispositivo su cui vuoi che venga eseguita questa azione'; + + @override + String get nothing_found => 'Nessun risultato'; + + @override + String get the_box_is_empty => 'La scatola è vuota'; + + @override + String get top_artists => 'Artisti Top'; + + @override + String get top_albums => 'Album Top'; + + @override + String get this_week => 'Questa settimana'; + + @override + String get this_month => 'Questo mese'; + + @override + String get last_6_months => 'Ultimi 6 mesi'; + + @override + String get this_year => 'Quest\'anno'; + + @override + String get last_2_years => 'Ultimi 2 anni'; + + @override + String get all_time => 'Di tutti i tempi'; + + @override + String powered_by_provider(Object providerName) { + return 'Sostenuto da $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Follower'; + + @override + String get birthday => 'Compleanno'; + + @override + String get subscription => 'Abbonamento'; + + @override + String get not_born => 'Non nato'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profilo'; + + @override + String get no_name => 'Nessun nome'; + + @override + String get edit => 'Modifica'; + + @override + String get user_profile => 'Profilo utente'; + + @override + String count_plays(Object count) { + return '$count riproduzioni'; + } + + @override + String get streaming_fees_hypothetical => 'Spese di streaming (ipotetico)'; + + @override + String get minutes_listened => 'Minuti ascoltati'; + + @override + String get streamed_songs => 'Brani in streaming'; + + @override + String count_streams(Object count) { + return '$count streaming'; + } + + @override + String get owned_by_you => 'Di tua proprietà'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'Copiato $shareUrl negli appunti'; + } + + @override + String get hipotetical_calculation => + '*Questo è calcolato in base al pagamento medio per stream delle piattaforme di streaming musicale online, che va da \$0.003 a \$0.005. Si tratta di un calcolo ipotetico per dare all\'utente un\'idea di quanto avrebbe pagato agli artisti se avesse ascoltato la loro canzone su diverse piattaforme di streaming musicale.'; + + @override + String count_mins(Object minutes) { + return '$minutes min'; + } + + @override + String get summary_minutes => 'minuti'; + + @override + String get summary_listened_to_music => 'Musica ascoltata'; + + @override + String get summary_songs => 'brani'; + + @override + String get summary_streamed_overall => 'Streaming complessivo'; + + @override + String get summary_owed_to_artists => 'Dovuto agli artisti\nquesto mese'; + + @override + String get summary_artists => 'dell\'artista'; + + @override + String get summary_music_reached_you => 'La musica ti ha raggiunto'; + + @override + String get summary_full_albums => 'album completi'; + + @override + String get summary_got_your_love => 'Ha ricevuto il tuo amore'; + + @override + String get summary_playlists => 'playlist'; + + @override + String get summary_were_on_repeat => 'Erano in ripetizione'; + + @override + String total_money(Object money) { + return 'Totale $money'; + } + + @override + String get webview_not_found => 'Webview non trovato'; + + @override + String get webview_not_found_description => + 'Nessun runtime Webview installato nel tuo dispositivo.\nSe è installato, assicurati che sia nel environment PATH\n\nDopo l\'installazione, riavvia l\'app'; + + @override + String get unsupported_platform => 'Piattaforma non supportata'; + + @override + String get cache_music => 'Cache musica'; + + @override + String get open => 'Apri'; + + @override + String get cache_folder => 'Cartella cache'; + + @override + String get export => 'Esporta'; + + @override + String get clear_cache => 'Cancella cache'; + + @override + String get clear_cache_confirmation => 'Vuoi cancellare la cache?'; + + @override + String get export_cache_files => 'Esporta file nella cache'; + + @override + String found_n_files(Object count) { + return 'Trovati $count file'; + } + + @override + String get export_cache_confirmation => 'Vuoi esportare questi file su'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Esportati $filesExported su $files file'; + } + + @override + String get undo => 'Annulla'; + + @override + String get download_all => 'Scarica tutto'; + + @override + String get add_all_to_playlist => 'Aggiungi tutto alla playlist'; + + @override + String get add_all_to_queue => 'Aggiungi tutto alla coda'; + + @override + String get play_all_next => 'Riproduci tutto dopo'; + + @override + String get pause => 'Pausa'; + + @override + String get view_all => 'Vedi tutto'; + + @override + String get no_tracks_added_yet => + 'Sembra che non hai ancora aggiunto nessun brano'; + + @override + String get no_tracks => 'Sembra che non ci siano brani qui'; + + @override + String get no_tracks_listened_yet => + 'Sembra che non hai ascoltato nulla ancora'; + + @override + String get not_following_artists => 'Non stai seguendo alcun artista'; + + @override + String get no_favorite_albums_yet => + 'Sembra che non hai ancora aggiunto album ai tuoi preferiti'; + + @override + String get no_logs_found => 'Nessun registro trovato'; + + @override + String get youtube_engine => 'Motore YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine non è installato'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine non è installato nel tuo sistema.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Assicurati che sia disponibile nella variabile PATH o\nimposta il percorso assoluto all\'eseguibile $engine qui sotto'; + } + + @override + String get youtube_engine_unix_issue_message => + 'In macOS/Linux/os simili a unix, impostare il percorso su .zshrc/.bashrc/.bash_profile ecc. non funzionerà.\nDevi impostare il percorso nel file di configurazione della shell'; + + @override + String get download => 'Scarica'; + + @override + String get file_not_found => 'File non trovato'; + + @override + String get custom => 'Personalizzato'; + + @override + String get add_custom_url => 'Aggiungi URL personalizzato'; + + @override + String get edit_port => 'Modifica porta'; + + @override + String get port_helper_msg => + 'Il valore predefinito è -1, che indica un numero casuale. Se hai configurato un firewall, si consiglia di impostarlo.'; + + @override + String connect_request(Object client) { + return 'Consentire a $client di connettersi?'; + } + + @override + String get connection_request_denied => + 'Connessione negata. L\'utente ha negato l\'accesso.'; + + @override + String get an_error_occurred => 'Si è verificato un errore'; + + @override + String get copy_to_clipboard => 'Copia negli appunti'; + + @override + String get view_logs => 'Visualizza log'; + + @override + String get retry => 'Riprova'; + + @override + String get no_default_metadata_provider_selected => + 'Non hai impostato alcun provider di metadati predefinito'; + + @override + String get manage_metadata_providers => 'Gestisci provider di metadati'; + + @override + String get open_link_in_browser => 'Aprire il link nel browser?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Vuoi aprire il seguente link'; + + @override + String get unsafe_url_warning => + 'Potrebbe essere pericoloso aprire link da fonti non attendibili. Sii cauto!\nPuoi anche copiare il link negli appunti.'; + + @override + String get copy_link => 'Copia link'; + + @override + String get building_your_timeline => + 'Creazione della tua cronologia in base ai tuoi ascolti...'; + + @override + String get official => 'Ufficiale'; + + @override + String author_name(Object author) { + return 'Autore: $author'; + } + + @override + String get third_party => 'Terze parti'; + + @override + String get plugin_requires_authentication => + 'Il plugin richiede l\'autenticazione'; + + @override + String get update_available => 'Aggiornamento disponibile'; + + @override + String get supports_scrobbling => 'Supporta lo scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Questo plugin scrobbla la tua musica per generare la tua cronologia di ascolti.'; + + @override + String get default_metadata_source => 'Fonte metadati predefinita'; + + @override + String get set_default_metadata_source => + 'Imposta fonte metadati predefinita'; + + @override + String get default_audio_source => 'Fonte audio predefinita'; + + @override + String get set_default_audio_source => 'Imposta fonte audio predefinita'; + + @override + String get set_default => 'Imposta come predefinito'; + + @override + String get support => 'Supporto'; + + @override + String get support_plugin_development => 'Sostieni lo sviluppo del plugin'; + + @override + String can_access_name_api(Object name) { + return '- Può accedere all\'API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Vuoi installare questo plugin?'; + + @override + String get third_party_plugin_warning => + 'Questo plugin proviene da un repository di terze parti. Assicurati di fidarti della fonte prima di installarlo.'; + + @override + String get author => 'Autore'; + + @override + String get this_plugin_can_do_following => + 'Questo plugin può fare quanto segue'; + + @override + String get install => 'Installa'; + + @override + String get install_a_metadata_provider => 'Installa un provider di metadati'; + + @override + String get no_tracks_playing => 'Nessun brano in riproduzione al momento'; + + @override + String get synced_lyrics_not_available => + 'Testi sincronizzati non disponibili per questa canzone. Si prega di utilizzare la scheda'; + + @override + String get plain_lyrics => 'Testi semplici'; + + @override + String get tab_instead => 'invece.'; + + @override + String get disclaimer => 'Disclaimer'; + + @override + String get third_party_plugin_dmca_notice => + 'Il team di Spotube non si assume alcuna responsabilità (anche legale) per i plugin di \"terze parti\".\nUsali a tuo rischio e pericolo. Per eventuali bug/problemi, segnalali al repository del plugin.\n\nSe un plugin di \"terze parti\" sta violando i ToS/DMCA di un servizio/entità legale, per favore chiedi all\'autore del plugin \"terzo\" o alla piattaforma di hosting, ad esempio GitHub/Codeberg, di agire. Quelli elencati sopra (etichettati come \"terze parti\") sono tutti plugin pubblici/mantenuti dalla comunità. Non li curiamo, quindi non possiamo intraprendere alcuna azione su di essi.\n\n'; + + @override + String get input_does_not_match_format => + 'L\'input non corrisponde al formato richiesto'; + + @override + String get plugins => 'Plugin'; + + @override + String get paste_plugin_download_url => + 'Incolla l\'URL di download o l\'URL del repository GitHub/Codeberg o il link diretto al file .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Scarica e installa il plugin da URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Impossibile aggiungere il plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'Carica plugin da file'; + + @override + String get installed => 'Installato'; + + @override + String get available_plugins => 'Plugin disponibili'; + + @override + String get configure_plugins => + 'Configura i tuoi plugin per fornitore metadati e fonte audio'; + + @override + String get audio_scrobblers => 'Scrobbler audio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Fonte: '; + + @override + String get uncompressed => 'Non compresso'; + + @override + String get dab_music_source_description => + 'Per audiophile. Fornisce flussi audio di alta qualità/senza perdita. Abbinamento traccia accurato basato su ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_ja.dart b/lib/l10n/generated/app_localizations_ja.dart new file mode 100644 index 00000000..2505f68a --- /dev/null +++ b/lib/l10n/generated/app_localizations_ja.dart @@ -0,0 +1,1534 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Japanese (`ja`). +class AppLocalizationsJa extends AppLocalizations { + AppLocalizationsJa([String locale = 'ja']) : super(locale); + + @override + String get guest => 'ゲスト'; + + @override + String get browse => '閲覧'; + + @override + String get search => '検索'; + + @override + String get library => 'ライブラリ'; + + @override + String get lyrics => '歌詞'; + + @override + String get settings => '設定'; + + @override + String get genre_categories_filter => 'カテゴリーやジャンルを絞り込み...'; + + @override + String get genre => 'ジャンル'; + + @override + String get personalized => 'あなたにおすすめ'; + + @override + String get featured => '注目'; + + @override + String get new_releases => '新着'; + + @override + String get songs => '曲'; + + @override + String playing_track(Object track) { + return '$track を再生'; + } + + @override + String queue_clear_alert(Object track_length) { + return '現在のキューを消去します。$track_length 曲を消去します。\n続行しますか?'; + } + + @override + String get load_more => 'もっと読み込む'; + + @override + String get playlists => '再生リスト'; + + @override + String get artists => 'アーティスト'; + + @override + String get albums => 'アルバム'; + + @override + String get tracks => '曲'; + + @override + String get downloads => 'ダウンロード'; + + @override + String get filter_playlists => 'あなたの再生リストを絞り込み...'; + + @override + String get liked_tracks => 'いいねした曲'; + + @override + String get liked_tracks_description => 'いいねしたすべての曲'; + + @override + String get playlist => '再生リスト'; + + @override + String get create_a_playlist => '再生リストの作成'; + + @override + String get update_playlist => '再生リストを更新'; + + @override + String get create => '作成'; + + @override + String get cancel => 'キャンセル'; + + @override + String get update => '更新'; + + @override + String get playlist_name => '再生リスト名'; + + @override + String get name_of_playlist => '再生リストの名前'; + + @override + String get description => '説明'; + + @override + String get public => '公開'; + + @override + String get collaborative => 'コラボ'; + + @override + String get search_local_tracks => '端末内の曲を検索...'; + + @override + String get play => '再生'; + + @override + String get delete => '削除'; + + @override + String get none => 'なし'; + + @override + String get sort_a_z => 'A-Z 順に並び替え'; + + @override + String get sort_z_a => 'Z-A 順に並び替え'; + + @override + String get sort_artist => 'アーティスト順に並び替え'; + + @override + String get sort_album => 'アルバム順に並び替え'; + + @override + String get sort_duration => '長さ順に並べ替え'; + + @override + String get sort_tracks => '曲の並び替え'; + + @override + String currently_downloading(Object tracks_length) { + return 'ダウンロード中 ($tracks_length) 曲'; + } + + @override + String get cancel_all => 'すべてキャンセル'; + + @override + String get filter_artist => 'アーティストを絞り込み...'; + + @override + String followers(Object followers) { + return '$followers フォロワー'; + } + + @override + String get add_artist_to_blacklist => 'このアーティストをブラックリストに追加'; + + @override + String get top_tracks => '人気の曲'; + + @override + String get fans_also_like => 'ファンの間で人気'; + + @override + String get loading => '読み込み中...'; + + @override + String get artist => 'アーティスト'; + + @override + String get blacklisted => 'ブラックリスト'; + + @override + String get following => 'フォロー中'; + + @override + String get follow => 'フォローする'; + + @override + String get artist_url_copied => 'アーティストの URL をクリップボードにコピーしました'; + + @override + String added_to_queue(Object tracks) { + return '$tracks をキューに追加しました'; + } + + @override + String get filter_albums => 'アルバムを絞り込み...'; + + @override + String get synced => '同期する'; + + @override + String get plain => 'そのまま'; + + @override + String get shuffle => 'シャッフル'; + + @override + String get search_tracks => '曲を検索...'; + + @override + String get released => 'リリース日'; + + @override + String error(Object error) { + return 'エラー $error'; + } + + @override + String get title => 'タイトル'; + + @override + String get time => '長さ'; + + @override + String get more_actions => 'ほかの操作'; + + @override + String download_count(Object count) { + return 'ダウンロード ($count) 曲'; + } + + @override + String add_count_to_playlist(Object count) { + return '再生リストに ($count) 曲を追加'; + } + + @override + String add_count_to_queue(Object count) { + return 'キューに ($count) 曲を追加'; + } + + @override + String play_count_next(Object count) { + return '次に ($count) 曲を再生'; + } + + @override + String get album => 'アルバム'; + + @override + String copied_to_clipboard(Object data) { + return '$data をクリップボードにコピーしました'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track をこの再生リストに追加'; + } + + @override + String get add => '追加'; + + @override + String added_track_to_queue(Object track) { + return 'キューに $track を追加しました'; + } + + @override + String get add_to_queue => 'キューに追加'; + + @override + String track_will_play_next(Object track) { + return '$track を次に再生'; + } + + @override + String get play_next => '次に再生'; + + @override + String removed_track_from_queue(Object track) { + return 'キューから $track を除去しました'; + } + + @override + String get remove_from_queue => 'キューから除去'; + + @override + String get remove_from_favorites => 'お気に入りから除去'; + + @override + String get save_as_favorite => 'お気に入りに保存'; + + @override + String get add_to_playlist => '再生リストに追加'; + + @override + String get remove_from_playlist => '再生リストから除去'; + + @override + String get add_to_blacklist => 'ブラックリストに追加'; + + @override + String get remove_from_blacklist => 'ブラックリストから除去'; + + @override + String get share => '共有'; + + @override + String get mini_player => 'ミニプレイヤー'; + + @override + String get slide_to_seek => '前後にスライドしてシーク'; + + @override + String get shuffle_playlist => '再生リストをシャッフル'; + + @override + String get unshuffle_playlist => '再生リストのシャッフル解除'; + + @override + String get previous_track => '前の曲'; + + @override + String get next_track => '次の曲'; + + @override + String get pause_playback => '再生を停止'; + + @override + String get resume_playback => '再生を再開'; + + @override + String get loop_track => '曲をループ'; + + @override + String get no_loop => 'ループなし'; + + @override + String get repeat_playlist => '再生リストをリピート'; + + @override + String get queue => '再生キュー'; + + @override + String get alternative_track_sources => 'この曲の別の音源を選ぶ'; + + @override + String get download_track => '曲のダウンロード'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks曲の再生キュー'; + } + + @override + String get clear_all => 'すべて消去l'; + + @override + String get show_hide_ui_on_hover => 'マウスを乗せてUIを表示/隠す'; + + @override + String get always_on_top => '常に手前に表示'; + + @override + String get exit_mini_player => 'ミニプレイヤーを終了'; + + @override + String get download_location => 'ダウンロード先'; + + @override + String get local_library => '端末内ライブラリ'; + + @override + String get add_library_location => 'ライブラリに追加'; + + @override + String get remove_library_location => 'ライブラリから削除'; + + @override + String get account => 'アカウント'; + + @override + String get logout => 'ログアウト'; + + @override + String get logout_of_this_account => 'このアカウントからログアウト'; + + @override + String get language_region => '言語 & 地域'; + + @override + String get language => '言語'; + + @override + String get system_default => 'システムの既定値'; + + @override + String get market_place_region => '音楽市場の地域'; + + @override + String get recommendation_country => 'おすすめの国'; + + @override + String get appearance => '外観'; + + @override + String get layout_mode => 'レイアウトの種類'; + + @override + String get override_layout_settings => 'レスポンシブなレイアウトの種類の設定を上書きする'; + + @override + String get adaptive => '適応的'; + + @override + String get compact => 'コンパクト'; + + @override + String get extended => '幅広'; + + @override + String get theme => 'テーマ'; + + @override + String get dark => 'ダーク'; + + @override + String get light => 'ライト'; + + @override + String get system => 'システムに従う'; + + @override + String get accent_color => 'アクセントカラー'; + + @override + String get sync_album_color => 'アルバムの色に合わせる'; + + @override + String get sync_album_color_description => 'アルバムアートの主張色をアクセントカラーとして使用'; + + @override + String get playback => '再生'; + + @override + String get audio_quality => '音声品質'; + + @override + String get high => '高'; + + @override + String get low => '低'; + + @override + String get pre_download_play => '事前ダウンロードと再生'; + + @override + String get pre_download_play_description => + '音声をストリーミングする代わりに、データをバイト単位でダウンロードして再生 (回線速度が早いユーザーにおすすめ)'; + + @override + String get skip_non_music => '音楽でない部分をスキップ (SponsorBlock)'; + + @override + String get blacklist_description => '曲とアーティストのブラックリスト'; + + @override + String get wait_for_download_to_finish => '現在のダウンロードが完了するまでお待ちください'; + + @override + String get desktop => 'デスクトップ'; + + @override + String get close_behavior => '閉じた時の動作'; + + @override + String get close => '閉じる'; + + @override + String get minimize_to_tray => 'トレイに最小化'; + + @override + String get show_tray_icon => 'システムトレイにアイコンを表示'; + + @override + String get about => 'このアプリについて'; + + @override + String get u_love_spotube => 'Spotube が好きだと知っていますよ'; + + @override + String get check_for_updates => 'アップデートの確認'; + + @override + String get about_spotube => 'Spotube について'; + + @override + String get blacklist => 'ブラックリスト'; + + @override + String get please_sponsor => '出資/寄付もお待ちします'; + + @override + String get spotube_description => + 'Spotube は、軽量でクロスプラットフォームな、すべて無料の spotify クライアント'; + + @override + String get version => 'バージョン'; + + @override + String get build_number => 'ビルド番号'; + + @override + String get founder => '創始者'; + + @override + String get repository => 'リポジトリ'; + + @override + String get bug_issues => 'バグや問題'; + + @override + String get made_with => '❤️ を込めてバングラディシュ🇧🇩で開発'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'ライセンス'; + + @override + String get credentials_will_not_be_shared_disclaimer => + '心配ありません。個人情報を収集したり、共有されることはありません'; + + @override + String get know_how_to_login => 'やり方が分からないですか?'; + + @override + String get follow_step_by_step_guide => 'やり方の説明を見る'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookies'; + } + + @override + String get fill_in_all_fields => 'すべての欄に入力してください'; + + @override + String get submit => '送信'; + + @override + String get exit => '終了'; + + @override + String get previous => '前へ'; + + @override + String get next => '次へ'; + + @override + String get done => '完了'; + + @override + String get step_1 => 'ステップ 1'; + + @override + String get first_go_to => '最初にここを開き'; + + @override + String get something_went_wrong => '何か誤りがあります'; + + @override + String get piped_instance => 'Piped サーバーのインスタンス'; + + @override + String get piped_description => '曲の一致に使う Piped サーバーのインスタンス'; + + @override + String get piped_warning => 'それらの一部ではうまく動作しないこともあります。自己責任で使用してください'; + + @override + String get invidious_instance => 'Invidiousサーバーインスタンス'; + + @override + String get invidious_description => '曲の一致に使用するInvidiousサーバーインスタンス'; + + @override + String get invidious_warning => '一部はうまく機能しない可能性があります。自己責任で使用してください'; + + @override + String get generate => '生成'; + + @override + String track_exists(Object track) { + return '曲 $track は既に存在します'; + } + + @override + String get replace_downloaded_tracks => 'すべてのダウンロード済みの曲を置換'; + + @override + String get skip_download_tracks => 'すべてのダウンロード済みの曲をスキップ'; + + @override + String get do_you_want_to_replace => '既存の曲と置換しますか?'; + + @override + String get replace => '置換する'; + + @override + String get skip => 'スキップ'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '$typeを最大$count 個まで選択'; + } + + @override + String get select_genres => 'ジャンルを選択'; + + @override + String get add_genres => 'ジャンルを追加'; + + @override + String get country => '国'; + + @override + String get number_of_tracks_generate => '生成する曲数'; + + @override + String get acousticness => 'アコースティック感'; + + @override + String get danceability => 'ダンス感'; + + @override + String get energy => 'エネルギー'; + + @override + String get instrumentalness => 'インストゥルメンタル'; + + @override + String get liveness => 'ライブ感'; + + @override + String get loudness => 'ラウドネス'; + + @override + String get speechiness => '会話感'; + + @override + String get valence => '多幸性'; + + @override + String get popularity => '人気度'; + + @override + String get key => 'キー'; + + @override + String get duration => '長さ (秒)'; + + @override + String get tempo => 'テンポ (BPM)'; + + @override + String get mode => '長調'; + + @override + String get time_signature => '拍子記号'; + + @override + String get short => '短'; + + @override + String get medium => '中'; + + @override + String get long => '長'; + + @override + String get min => '最小'; + + @override + String get max => '最大'; + + @override + String get target => '目標'; + + @override + String get moderate => '中'; + + @override + String get deselect_all => 'すべて選択解除'; + + @override + String get select_all => 'すべて選択'; + + @override + String get are_you_sure => 'よろしいですか?'; + + @override + String get generating_playlist => 'カスタムの再生リストを生成中...'; + + @override + String selected_count_tracks(Object count) { + return '$count 曲が選ばれました'; + } + + @override + String get download_warning => + '全曲の一括ダウンロードは明らかに音楽への海賊行為であり、音楽を生み出す共同体に損害を与えるでしょう。気づいてほしい。アーティストの多大な努力に敬意を払い、支援するようにしてください'; + + @override + String get download_ip_ban_warning => + 'また、通常よりも過剰なダウンロード要求があれば、YouTubeはあなたのIPをブロックします。つまりそのIPの端末からは、少なくとも2-3か月の間、(ログインしても)YouTubeを利用できなくなりす。そうなっても Spotube は一切の責任を負いません'; + + @override + String get by_clicking_accept_terms => '「同意する」のクリックにより、以下への同意となります:'; + + @override + String get download_agreement_1 => 'ええ、音楽への海賊行為だ。私はよくない'; + + @override + String get download_agreement_2 => '芸術作品を買うお金がないのでそうするしかないが、アーティストをできる限り支援する'; + + @override + String get download_agreement_3 => + '私のIPがYouTubeにブロックされることがあると完全に把握した。私のこの行動により起きたどんな事故も、Spotube やその所有者/貢献者に責任はありません。'; + + @override + String get decline => '同意しない'; + + @override + String get accept => '同意する'; + + @override + String get details => '詳細'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'チャンネル'; + + @override + String get likes => '高評価'; + + @override + String get dislikes => '低評価'; + + @override + String get views => '視聴回数'; + + @override + String get streamUrl => '動画の URL'; + + @override + String get stop => '中止'; + + @override + String get sort_newest => '追加日の新しい順に並び替え'; + + @override + String get sort_oldest => '追加日の古い順に並び替え'; + + @override + String get sleep_timer => 'スリープタイマー'; + + @override + String mins(Object minutes) { + return '$minutes 分'; + } + + @override + String hours(Object hours) { + return '$hours 時間'; + } + + @override + String hour(Object hours) { + return '$hours 時間'; + } + + @override + String get custom_hours => '時間を指定'; + + @override + String get logs => 'ログ'; + + @override + String get developers => '開発'; + + @override + String get not_logged_in => 'ログインしていません'; + + @override + String get search_mode => '検索モード'; + + @override + String get audio_source => '音声の提供元'; + + @override + String get ok => 'OK'; + + @override + String get failed_to_encrypt => '暗号化に失敗しました'; + + @override + String get encryption_failed_warning => + 'SpoTubeはデータを安全に保存するために暗号化を用いますが、暗号化に失敗しました。このため、安全でない保存領域への保存に切り替えます\nOSがLinuxなら、gnome-keyring、kde-wallet、keepassxcなどの管理ツールがインストールされていることを確認してください'; + + @override + String get querying_info => '情報を取得中...'; + + @override + String get piped_api_down => 'Piped APIがダウンしています'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Pipedインスタンス $pipedInstance は現在ダウンしています\n\nインスタンスを変更するか、「APIの種類」を公式のYouTube APIに変更してください\n\n変更後にアプリを再起動してください'; + } + + @override + String get you_are_offline => '現在、オフラインです'; + + @override + String get connection_restored => 'インターネット接続が復旧しました'; + + @override + String get use_system_title_bar => 'システムのタイトルバーを使う'; + + @override + String get crunching_results => '結果を処理中...'; + + @override + String get search_to_get_results => '結果を取得するために検索'; + + @override + String get use_amoled_mode => 'AMOLEDモードを使用'; + + @override + String get pitch_dark_theme => 'ピッチブラック ダークテーマ'; + + @override + String get normalize_audio => '音声を正規化'; + + @override + String get change_cover => 'カバーを変更'; + + @override + String get add_cover => 'カバーを追加'; + + @override + String get restore_defaults => '設定を初期化'; + + @override + String get download_music_format => '音楽ダウンロード形式'; + + @override + String get streaming_music_format => '音楽ストリーミング形式'; + + @override + String get download_music_quality => '音楽ダウンロード品質'; + + @override + String get streaming_music_quality => '音楽ストリーミング品質'; + + @override + String get login_with_lastfm => 'Last.fmでログイン'; + + @override + String get connect => '接続'; + + @override + String get disconnect_lastfm => 'Last.fmから切断'; + + @override + String get disconnect => '切断'; + + @override + String get username => 'ユーザー名'; + + @override + String get password => 'パスワード'; + + @override + String get login => 'ログイン'; + + @override + String get login_with_your_lastfm => 'Last.fmアカウントでログイン'; + + @override + String get scrobble_to_lastfm => 'Last.fmにスクロブルする'; + + @override + String get go_to_album => 'アルバムに移動'; + + @override + String get discord_rich_presence => 'Discord リッチプレゼンス'; + + @override + String get browse_all => 'すべてを閲覧'; + + @override + String get genres => 'ジャンル'; + + @override + String get explore_genres => 'ジャンルを探索'; + + @override + String get friends => '友達'; + + @override + String get no_lyrics_available => 'すみません、この曲の歌詞が見つかりません'; + + @override + String get start_a_radio => 'ラジオを開始'; + + @override + String get how_to_start_radio => 'ラジオをどのように開始しますか?'; + + @override + String get replace_queue_question => '現在のキューを置き換えるか、追加しますか?'; + + @override + String get endless_playback => 'エンドレス再生'; + + @override + String get delete_playlist => '再生リストを削除'; + + @override + String get delete_playlist_confirmation => 'この再生リストを削除しますか?'; + + @override + String get local_tracks => '端末内の曲'; + + @override + String get local_tab => '端末内'; + + @override + String get song_link => '曲のリンク'; + + @override + String get skip_this_nonsense => 'こんなことはスキップ'; + + @override + String get freedom_of_music => '“音楽の自由”'; + + @override + String get freedom_of_music_palm => '“音楽の自由を思いのままに”'; + + @override + String get get_started => 'さあ始めましょう'; + + @override + String get youtube_source_description => '推奨され、最適に機能します。'; + + @override + String get piped_source_description => '自由を感じる?YouTubeと同じだけど、はるかに自由です。'; + + @override + String get jiosaavn_source_description => '南アジア地域では最適です。'; + + @override + String get invidious_source_description => 'Pipedに似ていますが、より利用性があります。'; + + @override + String highest_quality(Object quality) { + return '最高品質:$quality'; + } + + @override + String get select_audio_source => '音声の提供元を選択'; + + @override + String get endless_playback_description => 'キューの最後に新しい曲を自動で追加'; + + @override + String get choose_your_region => '地域を選択'; + + @override + String get choose_your_region_description => 'Spotubeがあなたの地域に適したコンテンツを表示します。'; + + @override + String get choose_your_language => '言語を選択してください'; + + @override + String get help_project_grow => 'プロジェクトの成長を支援する'; + + @override + String get help_project_grow_description => + 'SpoTubeはオープンソースプロジェクトです。貢献したり、バグ報告したり、新機能を提案することで、プロジェクトの成長に貢献できます。'; + + @override + String get contribute_on_github => 'GitHubで貢献'; + + @override + String get donate_on_open_collective => 'Open Collectiveで寄付'; + + @override + String get browse_anonymously => '匿名で閲覧する'; + + @override + String get enable_connect => '接続する'; + + @override + String get enable_connect_description => '他の端末からSpotubeを制御する'; + + @override + String get devices => '機器'; + + @override + String get select => '選択'; + + @override + String connect_client_alert(Object client) { + return '$client から操作されています'; + } + + @override + String get this_device => 'この端末'; + + @override + String get remote => 'リモート'; + + @override + String get stats => '統計'; + + @override + String and_n_more(Object count) { + return 'さらに $count 項目'; + } + + @override + String get recently_played => '最近聴いた曲'; + + @override + String get browse_more => 'もっと表示'; + + @override + String get no_title => 'タイトルなし'; + + @override + String get not_playing => '再生なし'; + + @override + String get epic_failure => '壮大なエラー!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length 曲をキューに追加しました'; + } + + @override + String get spotube_has_an_update => 'Spotube の最新版あり'; + + @override + String get download_now => '今すぐダウンロード'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum がリリースされました'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version がリリースされました'; + } + + @override + String get read_the_latest => '最新の '; + + @override + String get release_notes => '更新情報を読む'; + + @override + String get pick_color_scheme => 'カラーテーマを選択'; + + @override + String get save => '保存'; + + @override + String get choose_the_device => '端末を選択:'; + + @override + String get multiple_device_connected => '複数の端末が接続されています。\nこの操作を実行する端末を選択'; + + @override + String get nothing_found => '何も見つかりませんでした'; + + @override + String get the_box_is_empty => 'ボックスは空です'; + + @override + String get top_artists => 'トップアーティスト'; + + @override + String get top_albums => 'トップアルバム'; + + @override + String get this_week => '今週'; + + @override + String get this_month => '今月'; + + @override + String get last_6_months => '過去6か月'; + + @override + String get this_year => '今年'; + + @override + String get last_2_years => '過去2年間'; + + @override + String get all_time => '全期間'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName 提供'; + } + + @override + String get email => 'メール'; + + @override + String get profile_followers => 'フォロワー'; + + @override + String get birthday => '誕生日'; + + @override + String get subscription => '登録'; + + @override + String get not_born => '未出生'; + + @override + String get hacker => 'ハッカー'; + + @override + String get profile => 'プロフィール'; + + @override + String get no_name => '名前なし'; + + @override + String get edit => '編集'; + + @override + String get user_profile => 'ユーザープロフィール'; + + @override + String count_plays(Object count) { + return '$count 回再生'; + } + + @override + String get streaming_fees_hypothetical => 'ストリーミング料金 (概算)'; + + @override + String get minutes_listened => '視聴時間'; + + @override + String get streamed_songs => 'ストリーミングされた曲'; + + @override + String count_streams(Object count) { + return '$count 回のストリーム'; + } + + @override + String get owned_by_you => 'あなたが所有'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl をクリップボードにコピーしました'; + } + + @override + String get hipotetical_calculation => + '*これは、オンライン音楽ストリーミングプラットフォームの1ストリームあたりの平均支払い額である\$0.003〜\$0.005に基づいて計算されています。これは、ユーザーが異なる音楽ストリーミングプラットフォームで曲を聴いた場合に、アーティストにどれだけ支払ったかを把握するための仮説的な計算です。'; + + @override + String count_mins(Object minutes) { + return '$minutes 分'; + } + + @override + String get summary_minutes => '分'; + + @override + String get summary_listened_to_music => '音楽を聴いた'; + + @override + String get summary_songs => '曲'; + + @override + String get summary_streamed_overall => 'まるごと聴いた'; + + @override + String get summary_owed_to_artists => '今月アーティストに払う\nべき額'; + + @override + String get summary_artists => 'アーティスト'; + + @override + String get summary_music_reached_you => 'の音楽が届いた'; + + @override + String get summary_full_albums => 'フルアルバム'; + + @override + String get summary_got_your_love => 'があなたの愛を受け取った'; + + @override + String get summary_playlists => '再生リスト'; + + @override + String get summary_were_on_repeat => 'をリピートしました'; + + @override + String total_money(Object money) { + return '計 $money'; + } + + @override + String get webview_not_found => 'Webviewが見つかりません'; + + @override + String get webview_not_found_description => + '端末にWebviewランタイムがインストールされていません。\nインストールされている場合は、環境変数のパスにあるか確認してください\n\nインストール後、アプリを再起動してください'; + + @override + String get unsupported_platform => '未対応のプラットフォーム'; + + @override + String get cache_music => '音楽をキャッシュ'; + + @override + String get open => '開く'; + + @override + String get cache_folder => 'キャッシュフォルダー'; + + @override + String get export => 'エクスポート'; + + @override + String get clear_cache => 'キャッシュをクリア'; + + @override + String get clear_cache_confirmation => 'キャッシュをクリアしますか?'; + + @override + String get export_cache_files => 'キャッシュされたファイルをエクスポート'; + + @override + String found_n_files(Object count) { + return '$countファイルが見つかりました'; + } + + @override + String get export_cache_confirmation => 'これらのファイルをエクスポートしますか'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported / $filesファイルがエクスポートされました'; + } + + @override + String get undo => '元に戻す'; + + @override + String get download_all => 'すべてダウンロード'; + + @override + String get add_all_to_playlist => 'すべて再生リストに追加'; + + @override + String get add_all_to_queue => 'すべてキューに追加'; + + @override + String get play_all_next => 'すべてを次に再生'; + + @override + String get pause => '一時停止'; + + @override + String get view_all => 'すべて表示'; + + @override + String get no_tracks_added_yet => 'まだ曲を追加していないようです'; + + @override + String get no_tracks => 'ここには曲がないようです'; + + @override + String get no_tracks_listened_yet => 'まだ何も聞いていないようです'; + + @override + String get not_following_artists => 'アーティストをフォローしていません'; + + @override + String get no_favorite_albums_yet => 'まだお気に入りのアルバムを追加していないようです'; + + @override + String get no_logs_found => 'ログなし'; + + @override + String get youtube_engine => 'YouTubeエンジン'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engineはインストールされていません'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engineはシステムにインストールされていません。'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'PATH変数に設定されていることを確認するか\n$engine実行ファイルの絶対パスを下記に設定してください'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/Unix系OSでは、.zshrc/.bashrc/.bash_profileなどでパスを設定しても動作しません。\nシェルの設定ファイルにパスを設定する必要があります'; + + @override + String get download => 'ダウンロード'; + + @override + String get file_not_found => 'ファイルが見つかりません'; + + @override + String get custom => '独自'; + + @override + String get add_custom_url => '独自にURLを追加'; + + @override + String get edit_port => 'ポートを編集'; + + @override + String get port_helper_msg => + '初期設定は-1で、ランダムな番号を示します。ファイアウォールを設定している場合に設定することを推奨します。'; + + @override + String connect_request(Object client) { + return '$clientの接続を許可しますか?'; + } + + @override + String get connection_request_denied => '接続が拒否されました。ユーザーがアクセスを拒否しました。'; + + @override + String get an_error_occurred => 'エラーが発生しました'; + + @override + String get copy_to_clipboard => 'クリップボードにコピー'; + + @override + String get view_logs => 'ログを表示'; + + @override + String get retry => '再試行'; + + @override + String get no_default_metadata_provider_selected => + 'デフォルトのメタデータプロバイダーが設定されていません'; + + @override + String get manage_metadata_providers => 'メタデータプロバイダーを管理'; + + @override + String get open_link_in_browser => 'リンクをブラウザで開きますか?'; + + @override + String get do_you_want_to_open_the_following_link => '次のリンクを開きますか'; + + @override + String get unsafe_url_warning => + '信頼できないソースからのリンクを開くのは安全ではない場合があります。注意してください!\nリンクをクリップボードにコピーすることもできます。'; + + @override + String get copy_link => 'リンクをコピー'; + + @override + String get building_your_timeline => 'あなたの視聴履歴に基づいてタイムラインを作成しています...'; + + @override + String get official => '公式'; + + @override + String author_name(Object author) { + return '作者: $author'; + } + + @override + String get third_party => 'サードパーティ'; + + @override + String get plugin_requires_authentication => 'プラグインには認証が必要です'; + + @override + String get update_available => 'アップデートが利用可能です'; + + @override + String get supports_scrobbling => 'scrobblingに対応'; + + @override + String get plugin_scrobbling_info => 'このプラグインは、あなたの音楽をscrobbleして視聴履歴を生成します。'; + + @override + String get default_metadata_source => 'デフォルトメタデータソース'; + + @override + String get set_default_metadata_source => 'デフォルトメタデータソースを設定'; + + @override + String get default_audio_source => 'デフォルトオーディオソース'; + + @override + String get set_default_audio_source => 'デフォルトオーディオソースを設定'; + + @override + String get set_default => 'デフォルトに設定'; + + @override + String get support => 'サポート'; + + @override + String get support_plugin_development => 'プラグイン開発をサポート'; + + @override + String can_access_name_api(Object name) { + return '- **$name** APIにアクセスできます'; + } + + @override + String get do_you_want_to_install_this_plugin => 'このプラグインをインストールしますか?'; + + @override + String get third_party_plugin_warning => + 'このプラグインはサードパーティのリポジトリからのものです。インストールする前にソースを信頼できるか確認してください。'; + + @override + String get author => '作者'; + + @override + String get this_plugin_can_do_following => 'このプラグインは以下のことができます'; + + @override + String get install => 'インストール'; + + @override + String get install_a_metadata_provider => 'メタデータプロバイダーをインストール'; + + @override + String get no_tracks_playing => '現在再生中のトラックはありません'; + + @override + String get synced_lyrics_not_available => 'この曲の同期歌詞は利用できません。代わりに'; + + @override + String get plain_lyrics => 'シンプルな歌詞'; + + @override + String get tab_instead => 'タブを使用してください。'; + + @override + String get disclaimer => '免責事項'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotubeチームは、いかなる「サードパーティ」プラグインについても責任(法的責任を含む)を負いません。\nご自身の責任でご使用ください。バグや問題については、プラグインリポジトリに報告してください。\n\n「サードパーティ」プラグインが何らかのサービス/法人のToS/DMCAを侵害している場合、その「サードパーティ」プラグインの作者またはホスティングプラットフォーム(例:GitHub/Codeberg)に措置を講じるよう依頼してください。上記に記載されている(「サードパーティ」とラベル付けされた)ものはすべて、パブリック/コミュニティによって維持されているプラグインです。私たちはそれらをキュレーションしていないため、それらに対して措置を講じることはできません。\n\n'; + + @override + String get input_does_not_match_format => '入力が必須フォーマットと一致しません'; + + @override + String get plugins => 'プラグイン'; + + @override + String get paste_plugin_download_url => + 'ダウンロードURL、GitHub/CodebergリポジトリURL、または.smplugファイルへの直接リンクを貼り付けます'; + + @override + String get download_and_install_plugin_from_url => + 'URLからプラグインをダウンロードしてインストール'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'プラグインの追加に失敗しました: $error'; + } + + @override + String get upload_plugin_from_file => 'ファイルからプラグインをアップロード'; + + @override + String get installed => 'インストール済み'; + + @override + String get available_plugins => '利用可能なプラグイン'; + + @override + String get configure_plugins => '独自のメタデータプロバイダーとオーディオソースプラグインを設定'; + + @override + String get audio_scrobblers => 'オーディオスクロッブラー'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'ソース: '; + + @override + String get uncompressed => '非圧縮'; + + @override + String get dab_music_source_description => + 'オーディオファイル向け。高品質/ロスレスオーディオストリームを提供。正確なISRCベースのトラックマッチング。'; +} diff --git a/lib/l10n/generated/app_localizations_ka.dart b/lib/l10n/generated/app_localizations_ka.dart new file mode 100644 index 00000000..c8557037 --- /dev/null +++ b/lib/l10n/generated/app_localizations_ka.dart @@ -0,0 +1,1573 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Georgian (`ka`). +class AppLocalizationsKa extends AppLocalizations { + AppLocalizationsKa([String locale = 'ka']) : super(locale); + + @override + String get guest => 'სტუმარი'; + + @override + String get browse => 'ნახვა'; + + @override + String get search => 'ძებნა'; + + @override + String get library => 'ბიბლიოთეკა'; + + @override + String get lyrics => 'ტექსტები'; + + @override + String get settings => 'კონფიგურაციები'; + + @override + String get genre_categories_filter => 'კატეგორიების ან ჟანრების ფილტრი...'; + + @override + String get genre => 'ჟანრი'; + + @override + String get personalized => 'პეერსონალიზებული'; + + @override + String get featured => 'გამორჩეული'; + + @override + String get new_releases => 'ახალი გამოცემები'; + + @override + String get songs => 'სიმღერები'; + + @override + String playing_track(Object track) { + return 'უკრავს $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'ეს გაასუფთავებს მიმდინარე რიგს. $track_length ტრეკი წაიშლება\nᲒინდა გააგრძელო?'; + } + + @override + String get load_more => 'მეტის ჩატვირთვა'; + + @override + String get playlists => 'ფლეილისტები'; + + @override + String get artists => 'არტისტები'; + + @override + String get albums => 'ალბომები'; + + @override + String get tracks => 'ტრეკები'; + + @override + String get downloads => 'ჩამოტვირთვები'; + + @override + String get filter_playlists => 'ფლეილისტების გაფილტვრა...'; + + @override + String get liked_tracks => 'მოწონებული ტრეკები'; + + @override + String get liked_tracks_description => 'ყველა შენი მოწონებული ტრეკი'; + + @override + String get playlist => 'პლეისთი'; + + @override + String get create_a_playlist => 'ფლეილისტის შექმნა'; + + @override + String get update_playlist => 'ფლეილისტის განახლება'; + + @override + String get create => 'შექმნა'; + + @override + String get cancel => 'გაუქმება'; + + @override + String get update => 'განახლება'; + + @override + String get playlist_name => 'ფლეილისტის სახელი'; + + @override + String get name_of_playlist => 'ფლეილისტის სახელი'; + + @override + String get description => 'აღწერა'; + + @override + String get public => 'საჯარო'; + + @override + String get collaborative => 'კოლაბორაციული'; + + @override + String get search_local_tracks => 'ლოცალური ტრეკების ძებნა...'; + + @override + String get play => 'დაკვრა'; + + @override + String get delete => 'წაშლა'; + + @override + String get none => 'არცერთი'; + + @override + String get sort_a_z => 'დალაგება A-Z-ს მიხედვით'; + + @override + String get sort_z_a => 'დალაგება Z-A-ს მიხედვით'; + + @override + String get sort_artist => 'დალაგება არტისტის მიხედვით'; + + @override + String get sort_album => 'დალაგება ალბომის მიხედვით'; + + @override + String get sort_duration => 'დალაგება ხანგრძლივობის მიხედვით'; + + @override + String get sort_tracks => 'ტრეკების დალაგება'; + + @override + String currently_downloading(Object tracks_length) { + return 'მიმდინარეობს ჩამოტვირთვა ($tracks_length)'; + } + + @override + String get cancel_all => 'ყველას გაუქმება'; + + @override + String get filter_artist => 'არტისტების ფილტრი...'; + + @override + String followers(Object followers) { + return '$followers ფოლოვერები'; + } + + @override + String get add_artist_to_blacklist => 'არტისტის შავ სიაში დამატება'; + + @override + String get top_tracks => 'ტოპ ტრეკები'; + + @override + String get fans_also_like => 'ფანებს ასევე მოსწონთ'; + + @override + String get loading => 'იტვირთება...'; + + @override + String get artist => 'არტისტი'; + + @override + String get blacklisted => 'შავ სიაში მყოფი'; + + @override + String get following => 'ფოლოვინგი'; + + @override + String get follow => 'დაფოლოვება'; + + @override + String get artist_url_copied => 'არტისტის ლინკი დაკოპირებულია'; + + @override + String added_to_queue(Object tracks) { + return '$tracks ტრეკი დაემატა რიგში'; + } + + @override + String get filter_albums => 'ალბომების გაფილტვრა...'; + + @override + String get synced => 'სინქრონიზებული'; + + @override + String get plain => 'Plain'; + + @override + String get shuffle => 'რიგის არევა'; + + @override + String get search_tracks => 'ტრეკების ძებნა...'; + + @override + String get released => 'გამოშვებული'; + + @override + String error(Object error) { + return 'შეცდომა $error'; + } + + @override + String get title => 'სათაური'; + + @override + String get time => 'დრო'; + + @override + String get more_actions => 'მეტი მოქმედებები'; + + @override + String download_count(Object count) { + return 'გადმოწერა ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'ფლეილისტში ($count)-ის დამატება'; + } + + @override + String add_count_to_queue(Object count) { + return 'რიგში ($count)-ის დამატება'; + } + + @override + String play_count_next(Object count) { + return 'შემდეგი ($count)-ის დაკვრა'; + } + + @override + String get album => 'ალბომი'; + + @override + String copied_to_clipboard(Object data) { + return '$data დაკოპირებულია'; + } + + @override + String add_to_following_playlists(Object track) { + return 'დაამატე $track ამ ფლეილისტებში'; + } + + @override + String get add => 'დამატება'; + + @override + String added_track_to_queue(Object track) { + return 'რიგში დაემატა $track'; + } + + @override + String get add_to_queue => 'რიგში დამატება'; + + @override + String track_will_play_next(Object track) { + return '$track დაუკრავს შემდეგს'; + } + + @override + String get play_next => 'შემდეგის დაკვრა'; + + @override + String removed_track_from_queue(Object track) { + return 'რიგიდან წაიშალა $track'; + } + + @override + String get remove_from_queue => 'რიგიდან წაშლა'; + + @override + String get remove_from_favorites => 'ფავორიტებიდან წაშლა'; + + @override + String get save_as_favorite => 'ფავორიტებში დამატება'; + + @override + String get add_to_playlist => 'ფლეილისტში დამატება'; + + @override + String get remove_from_playlist => 'ფლეილისტიდან წაშლა'; + + @override + String get add_to_blacklist => 'შავ სიაში დამატება'; + + @override + String get remove_from_blacklist => 'შავი სიიდან წაშლა'; + + @override + String get share => 'გაზიარება'; + + @override + String get mini_player => 'მინი დამკვრელი'; + + @override + String get slide_to_seek => 'გადახვევისთვის გაასრიალეთ წინ ან უკან'; + + @override + String get shuffle_playlist => 'ფლეილისტის არევა'; + + @override + String get unshuffle_playlist => 'ფლეილისტის დალაგება'; + + @override + String get previous_track => 'წინა ტრეკი'; + + @override + String get next_track => 'შემდეგი ტრეკი'; + + @override + String get pause_playback => 'დაკვრის გაჩერება'; + + @override + String get resume_playback => 'დაკვრის გაგრძელება'; + + @override + String get loop_track => 'ტრეკის ლუპზე დაკვრა'; + + @override + String get no_loop => 'არ არის ციკლი'; + + @override + String get repeat_playlist => 'ფლეილისტის გამეორება'; + + @override + String get queue => 'რიგი'; + + @override + String get alternative_track_sources => 'ალტერნატიული ტრეკების წყაროები'; + + @override + String get download_track => 'გადმოწერე ტრეკი'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks ტრეკი რიგში'; + } + + @override + String get clear_all => 'ყველას წაშლა'; + + @override + String get show_hide_ui_on_hover => 'UI-ის ჩვენება/დამალვა ჰოვერზე'; + + @override + String get always_on_top => 'ტოველთვის ზემოდან'; + + @override + String get exit_mini_player => 'მინი დამკვრელიდან გამოსვლა'; + + @override + String get download_location => 'ჩამოტვირთვის მდებარეობა'; + + @override + String get local_library => 'ადგილობრივი ბიბლიოთეკა'; + + @override + String get add_library_location => 'ბიბლიოთეკაში დამატება'; + + @override + String get remove_library_location => 'ბიბლიოთეკიდან წაშლა'; + + @override + String get account => 'ანგარიში'; + + @override + String get logout => 'გასვლა'; + + @override + String get logout_of_this_account => 'ანგარიშიდან გასვლა'; + + @override + String get language_region => 'ენა და რეგიონი'; + + @override + String get language => 'ენა'; + + @override + String get system_default => 'სისტემის ნაგულისხმევი'; + + @override + String get market_place_region => 'მარკეტფლეისის რეგიონი'; + + @override + String get recommendation_country => 'რეკომენდირებული ქვეყანა'; + + @override + String get appearance => 'გარეგნობა'; + + @override + String get layout_mode => 'განლაგების რეჟიმი'; + + @override + String get override_layout_settings => + 'რესფონსივ განლაგების რეჟიმის კონფიგურაციაზე გადაწერა'; + + @override + String get adaptive => 'ადაპტირებული'; + + @override + String get compact => 'კომპაქტური'; + + @override + String get extended => 'გაფართოებული'; + + @override + String get theme => 'თემა'; + + @override + String get dark => 'ბნელი'; + + @override + String get light => 'ღია'; + + @override + String get system => 'სისტემის'; + + @override + String get accent_color => 'აქცენტის ფერი'; + + @override + String get sync_album_color => 'ალბომის ფერის სინქრონიზაცია'; + + @override + String get sync_album_color_description => + 'დომინანტური ალბომის ფერის აქცენტის ფერად გამოყენება'; + + @override + String get playback => 'დაკვრა'; + + @override + String get audio_quality => 'აუდიოს ხარისხი'; + + @override + String get high => 'მაღალი'; + + @override + String get low => 'დაბალი'; + + @override + String get pre_download_play => 'წინასწარ ჩამოტვირთვა და დაკვრა'; + + @override + String get pre_download_play_description => + 'აუდიოს სტრიმინგის ნაცვლად, ბაიტების ჩამოტვირთვა და დაკვრა (რეკომენდებულია უფრო მაღალი გამტარუნარიანობის მომხმარებლებისთვის)'; + + @override + String get skip_non_music => + 'არა მუსიკალური ნაწილის გამოტოვება (სპონსორის ბლოკი)'; + + @override + String get blacklist_description => 'შავ სიაში მყოფი არტისტები და ტრეკები'; + + @override + String get wait_for_download_to_finish => + 'გთხოვთ, დაელოდოთ მიმდინარე ჩამოტვირთვის დასრულებას'; + + @override + String get desktop => 'დესკტოპი'; + + @override + String get close_behavior => 'დახურვის ქცევა'; + + @override + String get close => 'დახურვა'; + + @override + String get minimize_to_tray => 'მინიმიზაცია'; + + @override + String get show_tray_icon => 'სისტემის აიკონის ჩვენება'; + + @override + String get about => 'ჩვენს შესახებ'; + + @override + String get u_love_spotube => 'We know you love Spotube'; + + @override + String get check_for_updates => 'განახლებების შემოწმება'; + + @override + String get about_spotube => 'Spotube-ს შესახებ'; + + @override + String get blacklist => 'შავი სია'; + + @override + String get please_sponsor => 'გთხოვთ დაგვასპონსოროთ'; + + @override + String get spotube_description => + 'Spotube, a lightweight, cross-platform, free-for-all spotify client'; + + @override + String get version => 'ვერსია'; + + @override + String get build_number => 'Build Number'; + + @override + String get founder => 'დამფუძნებელი'; + + @override + String get repository => 'რეპოზიტორია'; + + @override + String get bug_issues => 'Bug+Issues'; + + @override + String get made_with => 'Made with ❤️ in Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'ლიცენზია'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'არ ინერვიულოთ, თქვენი მონაცემები არ იქნება შეგროვებული ან გაზიარებული ვინმესთან'; + + @override + String get know_how_to_login => 'არ იცით როგორ გააკეთოთ ეს?'; + + @override + String get follow_step_by_step_guide => + 'მიჰყევით ნაბიჯ-ნაბიჯ სახელმძღვანელოს'; + + @override + String cookie_name_cookie(Object name) { + return '$name ქუქი'; + } + + @override + String get fill_in_all_fields => 'გთხოვთ შეავსოთ ყველა ველი'; + + @override + String get submit => 'გაგზავნა'; + + @override + String get exit => 'გამოსვლა'; + + @override + String get previous => 'წინა'; + + @override + String get next => 'შემდეგი'; + + @override + String get done => 'მზადაა'; + + @override + String get step_1 => 'ნაბიჯი 1'; + + @override + String get first_go_to => 'პირველი, გადადით'; + + @override + String get something_went_wrong => 'Რაღაც არასწორად წავიდა'; + + @override + String get piped_instance => 'Piped Server Instance'; + + @override + String get piped_description => + 'The Piped server instance to use for track matching'; + + @override + String get piped_warning => 'ზოგიერთი მათგანმა შეიძლება კარგად არ იმუშაოს. '; + + @override + String get invidious_instance => 'Invidious სერვერის ინსტანცია'; + + @override + String get invidious_description => + 'Invidious სერვერის ინსტანცია, რომელიც გამოიყენება ტრეკის შესატყვისად'; + + @override + String get invidious_warning => + 'ზოგიერთი შეიძლება კარგად არ მუშაობდეს. გამოიყენეთ თქვენს პასუხისმგებლობაზე'; + + @override + String get generate => 'გააგენერირეთ'; + + @override + String track_exists(Object track) { + return 'ტრეკი $track უკვე არსებობს'; + } + + @override + String get replace_downloaded_tracks => 'ყველა ჩამოტვირთული ტრეკის შეცვლა'; + + @override + String get skip_download_tracks => 'ყველა ჩამოტვირთული ტრეკის გამოტოვება'; + + @override + String get do_you_want_to_replace => 'გსურთ შეცვალოთ არსებული ტრეკი??'; + + @override + String get replace => 'შეცვლა'; + + @override + String get skip => 'გამოტოვება'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'აირჩიე $count-მდე $type'; + } + + @override + String get select_genres => 'ჟანრების არჩევა'; + + @override + String get add_genres => 'ჟანრების დამატება'; + + @override + String get country => 'ქვეყანა'; + + @override + String get number_of_tracks_generate => 'დასაგენერირებელი ტრეკების რაოდენობა'; + + @override + String get acousticness => 'Acousticness'; + + @override + String get danceability => 'Danceability'; + + @override + String get energy => 'Energy'; + + @override + String get instrumentalness => 'Instrumentalness'; + + @override + String get liveness => 'Liveness'; + + @override + String get loudness => 'Loudness'; + + @override + String get speechiness => 'Speechiness'; + + @override + String get valence => 'Valence'; + + @override + String get popularity => 'Popularity'; + + @override + String get key => 'Key'; + + @override + String get duration => 'Duration (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'Time Signature'; + + @override + String get short => 'Short'; + + @override + String get medium => 'საშუალო'; + + @override + String get long => 'გრძელი'; + + @override + String get min => 'მინიმალური'; + + @override + String get max => 'მაქსიმალური'; + + @override + String get target => 'სამიზნე'; + + @override + String get moderate => 'საშუალო'; + + @override + String get deselect_all => 'ყველა მონიშვნის გაუქმება'; + + @override + String get select_all => 'ყველას მონიშვნა'; + + @override + String get are_you_sure => 'Დარწმუნებული ხართ?'; + + @override + String get generating_playlist => + 'მიმდინარეობს თქვენი მორგებული ფლეილისტის გენერირება...'; + + @override + String selected_count_tracks(Object count) { + return 'არჩეულია $count ტრეკი'; + } + + @override + String get download_warning => + 'If you download all Tracks at bulk you\'re clearly pirating Music & causing damage to the creative society of Music. I hope you are aware of this. Always, try respecting & supporting Artist\'s hard work'; + + @override + String get download_ip_ban_warning => + 'BTW, your IP can get blocked on YouTube due excessive download requests than usual. IP block means you can\'t use YouTube (even if you\'re logged in) for at least 2-3 months from that IP device. And Spotube doesn\'t hold any responsibility if this ever happens'; + + @override + String get by_clicking_accept_terms => + 'By clicking \'accept\' you agree to following terms:'; + + @override + String get download_agreement_1 => 'I know I\'m pirating Music. I\'m bad'; + + @override + String get download_agreement_2 => + 'I\'ll support the Artist wherever I can and I\'m only doing this because I don\'t have money to buy their art'; + + @override + String get download_agreement_3 => + 'I\'m completely aware that my IP can get blocked on YouTube & I don\'t hold Spotube or his owners/contributors responsible for any accidents caused by my current action'; + + @override + String get decline => 'უარყოფა'; + + @override + String get accept => 'დათანხმება'; + + @override + String get details => 'დეტალები'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Channel'; + + @override + String get likes => 'მოწონებები'; + + @override + String get dislikes => 'არ მოწონებები'; + + @override + String get views => 'ნახვები'; + + @override + String get streamUrl => 'სტრიმის ლინკი'; + + @override + String get stop => 'გაჩერება'; + + @override + String get sort_newest => 'ფალაგება სიახლის მიხედიტ'; + + @override + String get sort_oldest => 'დალაგება სიძველის მიხედვით'; + + @override + String get sleep_timer => 'ძილის ტაიმერი'; + + @override + String mins(Object minutes) { + return '$minutes წუთი'; + } + + @override + String hours(Object hours) { + return '$hours საათი'; + } + + @override + String hour(Object hours) { + return '$hours საათი'; + } + + @override + String get custom_hours => 'მორგებული საათები'; + + @override + String get logs => 'ლოგები'; + + @override + String get developers => 'დეველოპერები'; + + @override + String get not_logged_in => 'არ ხარ დალოგინებული'; + + @override + String get search_mode => 'ძებნის რეჟიმი'; + + @override + String get audio_source => 'აუდიოს წყარო'; + + @override + String get ok => 'ოკ'; + + @override + String get failed_to_encrypt => 'დაშიფვრა ვერ მოხერხდა'; + + @override + String get encryption_failed_warning => + 'Spotube uses encryption to securely store your data. But failed to do so. So it\'ll fallback to insecure storage\nIf you\'re using linux, please make sure you\'ve any secret-service (gnome-keyring, kde-wallet, keepassxc etc) installed'; + + @override + String get querying_info => 'Querying info...'; + + @override + String get piped_api_down => 'Piped API is down'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'The Piped instance $pipedInstance is currently down\n\nEither change the instance or change the \'API type\' to official YouTube API\n\nMake sure to restart the app after change'; + } + + @override + String get you_are_offline => 'ამჟამად ხაზგარეშე ხართ'; + + @override + String get connection_restored => 'თქვენი ინტერნეტ კავშირი აღდგა'; + + @override + String get use_system_title_bar => 'სისტემის სათაურის ზოლის გამოყენება'; + + @override + String get crunching_results => 'იტვირთება შედეგები...'; + + @override + String get search_to_get_results => 'მოძებნეთ შედეგების მისაღებად'; + + @override + String get use_amoled_mode => 'Pitch black dark theme'; + + @override + String get pitch_dark_theme => 'AMOLED Mode'; + + @override + String get normalize_audio => 'აუდიოს ნორმალიზება'; + + @override + String get change_cover => 'Ქავერის შეცვლა'; + + @override + String get add_cover => 'Ქავერის ფოტოს დამატება'; + + @override + String get restore_defaults => 'ნაგულისხმევი პარამეტრების აღდგენა'; + + @override + String get download_music_format => 'მუსიკის ჩამოტვირთვის ფორმატი'; + + @override + String get streaming_music_format => 'სტრიმინგის მუსიკის ფორმატი'; + + @override + String get download_music_quality => 'ჩამოტვირთვის ხარისხი'; + + @override + String get streaming_music_quality => 'სტრიმინგის ხარისხი'; + + @override + String get login_with_lastfm => 'Last.fm-ით შესვლა'; + + @override + String get connect => 'დაკავშირება'; + + @override + String get disconnect_lastfm => 'Last.fm-იდან გამოსვლა'; + + @override + String get disconnect => 'გამოსვლა'; + + @override + String get username => 'მომხმარებელი'; + + @override + String get password => 'პაროლი'; + + @override + String get login => 'შესვლა'; + + @override + String get login_with_your_lastfm => 'Last.fm ანგარიშით შესვლა'; + + @override + String get scrobble_to_lastfm => 'Scrobble to Last.fm'; + + @override + String get go_to_album => 'ალბომზე გადასვლა'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'ყველას ნახვა'; + + @override + String get genres => 'ჟანრები'; + + @override + String get explore_genres => 'შეისწავლეთ ჟანრები'; + + @override + String get friends => 'მეგობრები'; + + @override + String get no_lyrics_available => + 'უკაცრავად, ამ ტრეკისთვის ტექსტის პოვნა შეუძლებელია'; + + @override + String get start_a_radio => 'რადიოს ჩართვა'; + + @override + String get how_to_start_radio => 'როგორ გნებავთ რადიოს ჩართვა?'; + + @override + String get replace_queue_question => + 'გნებავთ ჩაანაცვლოთ არსებული რიგი თუ დაამატოთ მასზე?'; + + @override + String get endless_playback => 'დაუსრულებელი დაკვრა'; + + @override + String get delete_playlist => 'ფლეილისტის წაშლა'; + + @override + String get delete_playlist_confirmation => + 'დარწმუნებული ხართ რომ გნებავთ ფლეილისტის წაშლა?'; + + @override + String get local_tracks => 'ლოკალური ტრეკები'; + + @override + String get local_tab => 'ადგილობრივი'; + + @override + String get song_link => 'ტრეკის ლინკი'; + + @override + String get skip_this_nonsense => 'ამ სისულელის გამოტოვება'; + + @override + String get freedom_of_music => '“მუსიკის თავისუფლება”'; + + @override + String get freedom_of_music_palm => '“მუსიკის თავისუფლება შენს ხელის გულზე”'; + + @override + String get get_started => 'დავიწყოთ'; + + @override + String get youtube_source_description => + 'რეკომენდებულია და მუშაობს საუკეთესოდ.'; + + @override + String get piped_source_description => + 'თავისუფლად გრძნობთ თავს? იგივეა, რაც YouTube, მაგრამ ბევრი თავისუფალი.'; + + @override + String get jiosaavn_source_description => + 'საუკეთესოა სამხრეთ აზიის რეგიონისთვის.'; + + @override + String get invidious_source_description => + 'მსგავსია Piped-ის, მაგრამ მაღალი ხელმისაწვდომობით.'; + + @override + String highest_quality(Object quality) { + return 'საუკეთესო ხარისხი: $quality'; + } + + @override + String get select_audio_source => 'აუდიოს წყაროს არჩევა'; + + @override + String get endless_playback_description => + 'ახალი სიმთერების ავტომატურად რიგის ბოლოში დამატება'; + + @override + String get choose_your_region => 'აირჩიე შენი რეგიონი'; + + @override + String get choose_your_region_description => + 'This will help Spotube show you the right content\nfor your location.'; + + @override + String get choose_your_language => 'აირჩიე ენა'; + + @override + String get help_project_grow => 'დაეხმარეთ ამ პროექტს განვითარებაში'; + + @override + String get help_project_grow_description => + 'Spotube is an open-source project. You can help this project grow by contributing to the project, reporting bugs, or suggesting new features.'; + + @override + String get contribute_on_github => 'GitHub-ზე კონტრიბუცია'; + + @override + String get donate_on_open_collective => 'Open Collective-ზე დონაცია'; + + @override + String get browse_anonymously => 'ანონიმურად ნახვა'; + + @override + String get enable_connect => 'დაკავშირების ჩართვა'; + + @override + String get enable_connect_description => + 'აკონტროლე Spotube სხვა მოწყობილობებიდან'; + + @override + String get devices => 'მოწყობილობები'; + + @override + String get select => 'არჩევა'; + + @override + String connect_client_alert(Object client) { + return 'თქვენ კონტროლირებული ხართ $client მოწყობილობით'; + } + + @override + String get this_device => 'ეს მოწყობილობა'; + + @override + String get remote => 'დისტანციური'; + + @override + String get stats => 'სტატისტიკა'; + + @override + String and_n_more(Object count) { + return 'და $count მეტი'; + } + + @override + String get recently_played => 'მიუწვდელი'; + + @override + String get browse_more => 'დაიცალეთ მეტი'; + + @override + String get no_title => 'არ აქვს სათაური'; + + @override + String get not_playing => 'არ ერთვის'; + + @override + String get epic_failure => 'ეპიკური მარცხი!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'დამატებული $tracks_length ტრეკი რიგში'; + } + + @override + String get spotube_has_an_update => 'Spotube-ს აქვს განახლება'; + + @override + String get download_now => 'ჩამოტვირთეთ ახლავე'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum გამოშვებულია'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version გამოშვებულია'; + } + + @override + String get read_the_latest => 'წაიკითხეთ უახლესი '; + + @override + String get release_notes => 'გამოშვების შენიშვნები'; + + @override + String get pick_color_scheme => 'აირჩიეთ ფერის სქემა'; + + @override + String get save => 'შეინახეთ'; + + @override + String get choose_the_device => 'აირჩიეთ მოწყობილობა:'; + + @override + String get multiple_device_connected => + 'დაკავშირებულია რამდენიმე მოწყობილობა.\nაირჩიეთ მოწყობილობა, რომელზეც უნდა განხორციელდეს ეს მოქმედება'; + + @override + String get nothing_found => 'არაფერი მოიძებნა'; + + @override + String get the_box_is_empty => 'კვადრატია ცარიელი'; + + @override + String get top_artists => 'ტოპ არტისტები'; + + @override + String get top_albums => 'ტოპ ალბომები'; + + @override + String get this_week => 'ამ კვირას'; + + @override + String get this_month => 'ამ თვეში'; + + @override + String get last_6_months => 'ბოლო 6 თვე'; + + @override + String get this_year => 'ამ წელს'; + + @override + String get last_2_years => 'ბოლო 2 წელი'; + + @override + String get all_time => 'ყველა დრო'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName-ით გაწვდილი'; + } + + @override + String get email => 'ელ. ფოსტა'; + + @override + String get profile_followers => 'გამყვანები'; + + @override + String get birthday => 'დაბადების დღე'; + + @override + String get subscription => 'გამოწერა'; + + @override + String get not_born => 'არ დაბადებულა'; + + @override + String get hacker => 'ჰაკერი'; + + @override + String get profile => 'პროფილი'; + + @override + String get no_name => 'არ არის სახელი'; + + @override + String get edit => 'რედაქტირება'; + + @override + String get user_profile => 'მომხმარებლის პროფილი'; + + @override + String count_plays(Object count) { + return '$count გაწვდვა'; + } + + @override + String get streaming_fees_hypothetical => + '*ეს рассчитывается на основе выплат за поток от Spotify\nот \$0.003 до \$0.005. ეს ჰიპოთეტური გამოთვლა იძლევა მომხმარებელს წარმოდგენას იმაზე, რამდენად\nგადახდილი იქნებოდა არტისტებისთვის, თუ მათ მოუსმინოს Spotify-ს ტრეკებს.'; + + @override + String get minutes_listened => 'წუთები მოუსმინეს'; + + @override + String get streamed_songs => 'სტრიმირებული სიმღერები'; + + @override + String count_streams(Object count) { + return '$count სტრიმი'; + } + + @override + String get owned_by_you => 'შენ მიერ საკუთრებული'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl აიღო კლიპბორდზე'; + } + + @override + String get hipotetical_calculation => + '*ეს გამოითვლება ონლაინ მუსიკალური სტრიმინგის პლატფორმების საშუალო ანაზღაურების საფუძველზე, რომელიც შეადგენს \$0.003-დან \$0.005-მდე. ეს არის ჰიპოთეტური გაანგარიშება, რომელიც მომხმარებელს აძლევს წარმოდგენას, თუ რამდენს გადაუხდიდნენ ისინი არტისტებს, თუ მათ სიმღერებს მოუსმენდნენ სხვადასხვა მუსიკალურ სტრიმინგ პლატფორმაზე.'; + + @override + String count_mins(Object minutes) { + return '$minutes წუთი'; + } + + @override + String get summary_minutes => 'წუთები'; + + @override + String get summary_listened_to_music => 'მუსიკა გაწვდილი'; + + @override + String get summary_songs => 'მელოდია'; + + @override + String get summary_streamed_overall => 'გაწვდილი საერთო'; + + @override + String get summary_owed_to_artists => 'გადასახადი არტისტებს\nამ თვეში'; + + @override + String get summary_artists => 'არტისტების'; + + @override + String get summary_music_reached_you => 'მუსიკა ჩაგივარდა'; + + @override + String get summary_full_albums => 'სრული ალბომები'; + + @override + String get summary_got_your_love => 'მოსულა თქვენი სიყვარული'; + + @override + String get summary_playlists => 'პლეილისტები'; + + @override + String get summary_were_on_repeat => 'გადაწვდილი იყო'; + + @override + String total_money(Object money) { + return 'მთლიანი $money'; + } + + @override + String get webview_not_found => 'ვებვიუ ვერ მოიძებნა'; + + @override + String get webview_not_found_description => + 'თქვენს მოწყობილობაზე ვებვიუის შესრულების დრო არ არის დაყენებული.\nთუ დაყენებულია, დარწმუნდით, რომ ის environment PATH-შია\n\nდაყენების შემდეგ, გადატვირთეთ აპი'; + + @override + String get unsupported_platform => 'მოუხერხებელი პლატფორმა'; + + @override + String get cache_music => 'მუსიკის ქეში'; + + @override + String get open => 'გახსენით'; + + @override + String get cache_folder => 'ქეშის საქაღალდე'; + + @override + String get export => 'ექსპორტი'; + + @override + String get clear_cache => 'ქეშის გასუფთავება'; + + @override + String get clear_cache_confirmation => 'გსურთ ქეშის გასუფთავება?'; + + @override + String get export_cache_files => 'ქეშირებული ფაილების ექსპორტი'; + + @override + String found_n_files(Object count) { + return 'ნაპოვნია $count ფაილი'; + } + + @override + String get export_cache_confirmation => 'გსურთ ამ ფაილების ექსპორტი'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported ფაილი $files-დან ექსპორტირებულია'; + } + + @override + String get undo => 'დაბრუნება'; + + @override + String get download_all => 'ყველას ჩამოტვირთვა'; + + @override + String get add_all_to_playlist => 'ყველა დაამატეთ პლეისთში'; + + @override + String get add_all_to_queue => 'ყველა დაამატეთ რიგში'; + + @override + String get play_all_next => 'ყველა შემდეგ ითამაშე'; + + @override + String get pause => 'შეჩერება'; + + @override + String get view_all => 'ყველა ნახვა'; + + @override + String get no_tracks_added_yet => + 'გაჩნდება რომ ჯერ არ გაქვთ დამატებული ტრეკები'; + + @override + String get no_tracks => 'გავლებული არ ჩანს არ არსებობს ტრეკები'; + + @override + String get no_tracks_listened_yet => + 'გქონდეთ გრძნობა, რომ ჯერ არაფერი უსმენია'; + + @override + String get not_following_artists => 'არ მიჰყვებით რომელიმე არტისტს'; + + @override + String get no_favorite_albums_yet => + 'გაჩნდება რომ ჯერ არ გაქვთ დამატებული ალბომები თქვენს ფავორიტებში'; + + @override + String get no_logs_found => 'ჩაწერები ვერ მოიძებნა'; + + @override + String get youtube_engine => 'YouTube ძრავა'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine არ არის ინსტალირებული'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine არ არის ინსტალირებული თქვენს სისტემაში.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'დარწმუნდით, რომ ის ხელმისაწვდომია PATH ცვლადში ან\nდაუყავით $engine პროგრამის ფაილის სრული გზა'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/Unix მსგავსი ოპერაციული სისტემებში, .zshrc/.bashrc/.bash_profile-ით პათის დაყენება ვერ იმუშავებს.\nთქვენ უნდა დააყენოთ პათი შელ ფაილში'; + + @override + String get download => 'ჩამოტვირთვა'; + + @override + String get file_not_found => 'ფაილი ვერ მოიძებნა'; + + @override + String get custom => 'პერსონალიზირებული'; + + @override + String get add_custom_url => 'დამატება პერსონალური URL'; + + @override + String get edit_port => 'პორტის რედაქტირება'; + + @override + String get port_helper_msg => + 'ნაგულისხმევი არის -1, რაც შემთხვევითი ნომრის მითითებას ნიშნავს. თუ لديك firewall настроен, рекомендуется установить это.'; + + @override + String connect_request(Object client) { + return '$client-ის დაკავშირების ნებართვა?'; + } + + @override + String get connection_request_denied => + 'კავშირი უარყოფილია. მომხმარებელმა უარყო წვდომა.'; + + @override + String get an_error_occurred => 'მოხდა შეცდომა'; + + @override + String get copy_to_clipboard => 'კოპირება ბუფერში'; + + @override + String get view_logs => 'იხილეთ ჟურნალები'; + + @override + String get retry => 'ხელახლა ცდა'; + + @override + String get no_default_metadata_provider_selected => + 'თქვენ არ გაქვთ დაყენებული ნაგულისხმევი მეტამონაცემების პროვაიდერი'; + + @override + String get manage_metadata_providers => + 'მეტამონაცემების პროვაიდერების მართვა'; + + @override + String get open_link_in_browser => 'ბმულის გახსნა ბრაუზერში?'; + + @override + String get do_you_want_to_open_the_following_link => + 'გსურთ გახსნათ შემდეგი ბმული'; + + @override + String get unsafe_url_warning => + 'შეიძლება სახიფათო იყოს ბმულების გახსნა უნდობელი წყაროებიდან. იყავით ფრთხილად!\nასევე შეგიძლიათ დააკოპიროთ ბმული თქვენს ბუფერში.'; + + @override + String get copy_link => 'ბმულის კოპირება'; + + @override + String get building_your_timeline => + 'თქვენი დროის ხაზის აგება თქვენი მოსმენების საფუძველზე...'; + + @override + String get official => 'ოფიციალური'; + + @override + String author_name(Object author) { + return 'ავტორი: $author'; + } + + @override + String get third_party => 'მესამე მხარის'; + + @override + String get plugin_requires_authentication => + 'პლაგინი საჭიროებს ავთენტიფიკაციას'; + + @override + String get update_available => 'განახლება ხელმისაწვდომია'; + + @override + String get supports_scrobbling => 'მხარს უჭერს სქრობლინგს'; + + @override + String get plugin_scrobbling_info => + 'ეს პლაგინი აწარმოებს თქვენი მუსიკის სქრობლინგს, რათა შექმნას თქვენი მოსმენის ისტორია.'; + + @override + String get default_metadata_source => 'ნაგულისხმევი მეტამონაცემების წყარო'; + + @override + String get set_default_metadata_source => + 'ნაგულისხმევი მეტამონაცემების წყაროს დაყენება'; + + @override + String get default_audio_source => 'ნაგულისხმევი აუდიო წყარო'; + + @override + String get set_default_audio_source => 'ნაგულისხმევი აუდიო წყაროს დაყენება'; + + @override + String get set_default => 'ნაგულისხმევად დაყენება'; + + @override + String get support => 'მხარდაჭერა'; + + @override + String get support_plugin_development => 'პლაგინის განვითარების მხარდაჭერა'; + + @override + String can_access_name_api(Object name) { + return '- შეუძლია წვდომა **$name** API-ზე'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'გსურთ ამ პლაგინის დაყენება?'; + + @override + String get third_party_plugin_warning => + 'ეს პლაგინი არის მესამე მხარის საცავიდან. გთხოვთ, დარწმუნდეთ, რომ ენდობით წყაროს დაყენებამდე.'; + + @override + String get author => 'ავტორი'; + + @override + String get this_plugin_can_do_following => + 'ამ პლაგინს შეუძლია შემდეგის გაკეთება'; + + @override + String get install => 'დაყენება'; + + @override + String get install_a_metadata_provider => + 'დააყენეთ მეტამონაცემების პროვაიდერი'; + + @override + String get no_tracks_playing => 'ამჟამად არ უკრავს არცერთი ტრეკი'; + + @override + String get synced_lyrics_not_available => + 'ამ სიმღერისთვის სინქრონიზებული ტექსტები არ არის ხელმისაწვდომი. გთხოვთ, გამოიყენოთ'; + + @override + String get plain_lyrics => 'მარტივი ტექსტები'; + + @override + String get tab_instead => 'ჩანართი, სანაცვლოდ.'; + + @override + String get disclaimer => 'პასუხისმგებლობის უარყოფა'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube-ის გუნდი არ იღებს პასუხისმგებლობას (მათ შორის, იურიდიულს) არცერთ \"მესამე მხარის\" პლაგინზე.\nგთხოვთ, გამოიყენოთ ისინი თქვენი რისკის ქვეშ. ნებისმიერი ხარვეზის/პრობლემის შესახებ შეატყობინეთ პლაგინის საცავს.\n\nთუ რომელიმე \"მესამე მხარის\" პლაგინი არღვევს რაიმე სერვისის/იურიდიული პირის ToS/DMCA-ს, გთხოვთ, სთხოვეთ \"მესამე მხარის\" პლაგინის ავტორს ან ჰოსტინგის პლატფორმას, მაგალითად GitHub/Codeberg, მიიღოს ზომები. ზემოთ ჩამოთვლილი (\"მესამე მხარის\" ეტიკეტის მქონე) ყველა არის საჯარო/საზოგადოების მიერ შენარჩუნებული პლაგინები. ჩვენ მათ არ ვაკონტროლებთ, ამიტომ არ შეგვიძლია მათზე რაიმე ზომების მიღება.\n\n'; + + @override + String get input_does_not_match_format => + 'შეყვანა არ ემთხვევა საჭირო ფორმატს'; + + @override + String get plugins => 'პლაგინები'; + + @override + String get paste_plugin_download_url => + 'ჩასვით ჩამოტვირთვის url ან GitHub/Codeberg-ის რეპოს url ან პირდაპირი ბმული .smplug ფაილზე'; + + @override + String get download_and_install_plugin_from_url => + 'პლაგინის ჩამოტვირთვა და დაყენება url-დან'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'პლაგინის დამატება ვერ მოხერხდა: $error'; + } + + @override + String get upload_plugin_from_file => 'პლაგინის ატვირთვა ფაილიდან'; + + @override + String get installed => 'დაინსტალირებული'; + + @override + String get available_plugins => 'ხელმისაწვდომი პლაგინები'; + + @override + String get configure_plugins => + 'თქვენი საკუთარი მეტამონაცემებისა და აუდიო წყაროს პლაგინების კონფიგურაცია'; + + @override + String get audio_scrobblers => 'აუდიო სქრობლერები'; + + @override + String get scrobbling => 'სქრობლინგი'; + + @override + String get source => 'წყარო: '; + + @override + String get uncompressed => 'შეუკუმშავი'; + + @override + String get dab_music_source_description => + 'აუდიოფილებისთვის. უზრუნველყოფს მაღალი ხარისხის/უკომპრესო აუდიო სტრიმებს. ზუსტი შესაბამისობა ISRC-ის მიხედვით.'; +} diff --git a/lib/l10n/generated/app_localizations_ko.dart b/lib/l10n/generated/app_localizations_ko.dart new file mode 100644 index 00000000..42ea337a --- /dev/null +++ b/lib/l10n/generated/app_localizations_ko.dart @@ -0,0 +1,1538 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Korean (`ko`). +class AppLocalizationsKo extends AppLocalizations { + AppLocalizationsKo([String locale = 'ko']) : super(locale); + + @override + String get guest => '게스트'; + + @override + String get browse => '찾아보기'; + + @override + String get search => '검색'; + + @override + String get library => '라이브러리'; + + @override + String get lyrics => '가사'; + + @override + String get settings => '설정'; + + @override + String get genre_categories_filter => '카테고리 혹은 장르별로 불러오기'; + + @override + String get genre => '장르'; + + @override + String get personalized => '맞춤 추천'; + + @override + String get featured => '인기'; + + @override + String get new_releases => '신곡'; + + @override + String get songs => '노래'; + + @override + String playing_track(Object track) { + return '$track 을 재생'; + } + + @override + String queue_clear_alert(Object track_length) { + return '현재 재생 대기열을 없앱니다。$track_length 곡이 제거됩니다。\n계속 진행할까요?'; + } + + @override + String get load_more => '더 불러오기'; + + @override + String get playlists => '플레이리스트'; + + @override + String get artists => '아티스트'; + + @override + String get albums => '앨범'; + + @override + String get tracks => '곡'; + + @override + String get downloads => '다운로드한 곡'; + + @override + String get filter_playlists => '플레이리스트를 필터링'; + + @override + String get liked_tracks => '좋아하는 곡'; + + @override + String get liked_tracks_description => '좋아요를 남긴 곡들'; + + @override + String get playlist => '재생 목록'; + + @override + String get create_a_playlist => '플레이리스트를 생성'; + + @override + String get update_playlist => '플레이리스트를 업데이트'; + + @override + String get create => '생성'; + + @override + String get cancel => '취소'; + + @override + String get update => '업데이트'; + + @override + String get playlist_name => '플레이리스트명'; + + @override + String get name_of_playlist => '플레이리스트의 이름'; + + @override + String get description => '설명'; + + @override + String get public => '공개'; + + @override + String get collaborative => '공유 플레이리스트'; + + @override + String get search_local_tracks => '기기에 저장된 곡을 검색하기'; + + @override + String get play => '재생'; + + @override + String get delete => '삭제'; + + @override + String get none => '없음'; + + @override + String get sort_a_z => 'A-Z 순으로 정렬'; + + @override + String get sort_z_a => 'Z-A 순으로 정렬'; + + @override + String get sort_artist => '아티스트 순으로 정렬'; + + @override + String get sort_album => '앨범 순으로 정렬'; + + @override + String get sort_duration => '시간순 정렬'; + + @override + String get sort_tracks => '곡명 순으로 정렬'; + + @override + String currently_downloading(Object tracks_length) { + return '현재 ($tracks_length) 곡 다운로드 중'; + } + + @override + String get cancel_all => '모두 취소'; + + @override + String get filter_artist => '아티스트 필터링'; + + @override + String followers(Object followers) { + return '$followers 팔로워'; + } + + @override + String get add_artist_to_blacklist => '이 아티스트를 블랙리스트에 추가'; + + @override + String get top_tracks => '인기곡'; + + @override + String get fans_also_like => '애청자들이 좋아하는 곡'; + + @override + String get loading => '불러오는 중...'; + + @override + String get artist => '아티스트'; + + @override + String get blacklisted => '블랙리스트'; + + @override + String get following => '팔로우 중'; + + @override + String get follow => '팔로우하기'; + + @override + String get artist_url_copied => '아티스트의 URL 주소를 클립보드에 복사함'; + + @override + String added_to_queue(Object tracks) { + return '$tracks 곡을 대기열에 추가함'; + } + + @override + String get filter_albums => '앨범 필터링'; + + @override + String get synced => '동기화됨'; + + @override + String get plain => '그대로'; + + @override + String get shuffle => '셔플'; + + @override + String get search_tracks => '곡 검색하기'; + + @override + String get released => '공개일'; + + @override + String error(Object error) { + return '에러'; + } + + @override + String get title => '타이틀'; + + @override + String get time => '길이'; + + @override + String get more_actions => '다른 작업'; + + @override + String download_count(Object count) { + return '($count) 곡 다운로드'; + } + + @override + String add_count_to_playlist(Object count) { + return '플레이리스트에 ($count) 곡을 추가'; + } + + @override + String add_count_to_queue(Object count) { + return '대기열에 ($count) 곡을 추가'; + } + + @override + String play_count_next(Object count) { + return '이 다음에 ($count) 곡을 재생'; + } + + @override + String get album => '앨범'; + + @override + String copied_to_clipboard(Object data) { + return '$data 를 클립보드에 복사함'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track 을 이 플레이리스트에 추가'; + } + + @override + String get add => '추가'; + + @override + String added_track_to_queue(Object track) { + return '대기열에 $track 을 추가함'; + } + + @override + String get add_to_queue => '대기열에 추가'; + + @override + String track_will_play_next(Object track) { + return '$track 을 이 다음에 재생'; + } + + @override + String get play_next => '이 다음에 재생'; + + @override + String removed_track_from_queue(Object track) { + return '대기열에서 $track 를 제거함'; + } + + @override + String get remove_from_queue => '대기열에서 제거'; + + @override + String get remove_from_favorites => '즐겨찾기에서 제거'; + + @override + String get save_as_favorite => '즐겨찾기에 추가'; + + @override + String get add_to_playlist => '플레이리스트에 추가'; + + @override + String get remove_from_playlist => '플레이리스트에서 제거'; + + @override + String get add_to_blacklist => '블랙리스트에 추가'; + + @override + String get remove_from_blacklist => '블랙리스트에서 제거'; + + @override + String get share => '공유'; + + @override + String get mini_player => '미니 플레이어'; + + @override + String get slide_to_seek => '앞뒤로 슬라이드하여 탐색'; + + @override + String get shuffle_playlist => '플레이리스트를 섞기'; + + @override + String get unshuffle_playlist => '플레이리스트를 섞지 않기'; + + @override + String get previous_track => '이전 곡'; + + @override + String get next_track => '다음 곡'; + + @override + String get pause_playback => '일시정지'; + + @override + String get resume_playback => '재개'; + + @override + String get loop_track => '반복 재생'; + + @override + String get no_loop => '반복 없음'; + + @override + String get repeat_playlist => '플레이리스트 반복'; + + @override + String get queue => '재생 대기열'; + + @override + String get alternative_track_sources => '대체가능한 음악 서버'; + + @override + String get download_track => '곡 다운로드'; + + @override + String tracks_in_queue(Object tracks) { + return '대기열에 $tracks 곡이 있음'; + } + + @override + String get clear_all => '모두 제거'; + + @override + String get show_hide_ui_on_hover => '마우스를 올리면 UI를 표시/숨김'; + + @override + String get always_on_top => '항상 위에 표시'; + + @override + String get exit_mini_player => '미니 플레이어 닫기'; + + @override + String get download_location => '다운로드 경로'; + + @override + String get local_library => '로컬 도서관'; + + @override + String get add_library_location => '도서관에 추가'; + + @override + String get remove_library_location => '도서관에서 제거'; + + @override + String get account => '계정'; + + @override + String get logout => '로그아웃'; + + @override + String get logout_of_this_account => '이 계정에서 로그아웃'; + + @override + String get language_region => '언어 & 지역'; + + @override + String get language => '언어'; + + @override + String get system_default => '시스템 기본설정'; + + @override + String get market_place_region => '마켓플레이스 지역'; + + @override + String get recommendation_country => '추천 국가'; + + @override + String get appearance => '디자인'; + + @override + String get layout_mode => '레이아웃 모드'; + + @override + String get override_layout_settings => '반응형 레이아웃 모드 설정 덮어씌우기'; + + @override + String get adaptive => '적응형'; + + @override + String get compact => '컴팩트'; + + @override + String get extended => '확장'; + + @override + String get theme => '테마'; + + @override + String get dark => '다크'; + + @override + String get light => '라이트'; + + @override + String get system => '시스템과 동일'; + + @override + String get accent_color => '보조색'; + + @override + String get sync_album_color => '앨범 색상'; + + @override + String get sync_album_color_description => '앨범아트의 주요 색상을 보조색으로 사용'; + + @override + String get playback => '재생'; + + @override + String get audio_quality => '음질'; + + @override + String get high => '높음'; + + @override + String get low => '낮음'; + + @override + String get pre_download_play => '재생할 곡을 미리 다운로드'; + + @override + String get pre_download_play_description => + '스트리밍 방식을 쓰는 대신 파일 단위로 다운로드 받고 재생 (인터넷 대역폭이 높은 환경에서 추천)'; + + @override + String get skip_non_music => '음악이 아닌 부분을 스킵 (SponsorBlock)'; + + @override + String get blacklist_description => '블랙리스트에 추가된 곡과 아티스트'; + + @override + String get wait_for_download_to_finish => '현재 진행중인 다운로드가 끝날 때까지 기다려주세요'; + + @override + String get desktop => '데스크톱'; + + @override + String get close_behavior => '닫을 때의 동작'; + + @override + String get close => '닫기'; + + @override + String get minimize_to_tray => '트레이로 최소화'; + + @override + String get show_tray_icon => '시스템 트레이 아이콘 표시'; + + @override + String get about => '앱 정보'; + + @override + String get u_love_spotube => 'Spotube... 사랑하시죠?'; + + @override + String get check_for_updates => '업데이트 확인'; + + @override + String get about_spotube => 'Spotube에 관해'; + + @override + String get blacklist => '블랙리스트'; + + @override + String get please_sponsor => '후원해주시면 감사하겠습니다.'; + + @override + String get spotube_description => + 'Spotube는, 경량에 크로스플랫폼인데다 무료이기까지한 스포티파이 클라이언트입니다'; + + @override + String get version => '버전'; + + @override + String get build_number => '빌드 번호'; + + @override + String get founder => '창시자'; + + @override + String get repository => '리포지토리'; + + @override + String get bug_issues => '버그 및 이슈'; + + @override + String get made_with => '❤️을 담아 방글라데시에서 만듦'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => '라이선스'; + + @override + String get credentials_will_not_be_shared_disclaimer => + '걱정마세요. 개인정보를 수집하거나 공유하지 않습니다.'; + + @override + String get know_how_to_login => '어떻게 하는건지 모르겠나요?'; + + @override + String get follow_step_by_step_guide => '사용법 확인하기'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookies'; + } + + @override + String get fill_in_all_fields => '모든 필드에 정보를 입력해주세요'; + + @override + String get submit => '제출'; + + @override + String get exit => '종료'; + + @override + String get previous => '이전으로'; + + @override + String get next => '다음으로'; + + @override + String get done => '완료'; + + @override + String get step_1 => '1단계'; + + @override + String get first_go_to => '가장 먼저 먼저 들어갈 곳은 '; + + @override + String get something_went_wrong => '알 수 없는 이유로 동작에 실패했습니다.'; + + @override + String get piped_instance => 'Piped 서버의 인스턴스'; + + @override + String get piped_description => '곡 탐색에 사용할 Piped 서버 인스턴스'; + + @override + String get piped_warning => '몇몇 서버는 제대로 동작하지 않을 수 있습니다. 본인 책임 하에 이용해주세요.'; + + @override + String get invidious_instance => 'Invidious 서버 인스턴스'; + + @override + String get invidious_description => '트랙 매칭에 사용할 Invidious 서버 인스턴스'; + + @override + String get invidious_warning => '일부는 제대로 작동하지 않을 수 있습니다. 자신의 책임 하에 사용하세요'; + + @override + String get generate => '생성'; + + @override + String track_exists(Object track) { + return '곡 $track 은 이미 리스트에 있습니다'; + } + + @override + String get replace_downloaded_tracks => '다운로드한 모든 곡을 교체'; + + @override + String get skip_download_tracks => '다운로드가 끝난 곡을 모두 건너뛰기'; + + @override + String get do_you_want_to_replace => '현재 곡을 교체하시겠습니까?'; + + @override + String get replace => '교체'; + + @override + String get skip => '건너뛰기'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '$type을 $count개까지 선택'; + } + + @override + String get select_genres => '장르 선택'; + + @override + String get add_genres => '장르 추가'; + + @override + String get country => '국가'; + + @override + String get number_of_tracks_generate => '생성할 곡 수'; + + @override + String get acousticness => '반주 구간 (Acousticness)'; + + @override + String get danceability => '흥겨운 정도 (Danceability)'; + + @override + String get energy => '에너지 (Energy)'; + + @override + String get instrumentalness => '기악성 (Instrumentalness)'; + + @override + String get liveness => '생동감 (Liveness)'; + + @override + String get loudness => '라우드니스 (Loudness)'; + + @override + String get speechiness => '회화성 (Speechniss)'; + + @override + String get valence => '감정가 (Valence)'; + + @override + String get popularity => '인기도 (Popularity)'; + + @override + String get key => '조성 (키)'; + + @override + String get duration => '길이 (초)'; + + @override + String get tempo => '템포 (BPM)'; + + @override + String get mode => '장조'; + + @override + String get time_signature => '박자'; + + @override + String get short => '짧음'; + + @override + String get medium => '중간'; + + @override + String get long => '긺'; + + @override + String get min => '최소'; + + @override + String get max => '최대'; + + @override + String get target => '목표'; + + @override + String get moderate => '보통'; + + @override + String get deselect_all => '모두 선택해제'; + + @override + String get select_all => '모두 선택'; + + @override + String get are_you_sure => '괜찮겠습니까?'; + + @override + String get generating_playlist => '커스텀 플레이리스트를 생성하는 중...'; + + @override + String selected_count_tracks(Object count) { + return '$count 곡이 선택되었습니다.'; + } + + @override + String get download_warning => + '모든 트랙을 대량으로 다운로드하는 것은 명백한 불법 복제이며 음악 창작 사회에 피해를 입히는 행위입니다. 이 점을 알아주셨으면 합니다. 항상 아티스트의 노력을 존중하고 응원해 주세요.'; + + @override + String get download_ip_ban_warning => + '참고로, 평소보다 과도한 다운로드 요청으로 인해 YouTube에서 IP가 차단될 수 있습니다. IP 차단은 해당 IP 기기에서 최소 2~3개월 동안 (로그인한 상태에서도) YouTube를 사용할 수 없음을 의미합니다. 그리고 이런 일이 발생하더라도 스포튜브는 어떠한 책임도 지지 않습니다.'; + + @override + String get by_clicking_accept_terms => '\'동의\'를 클릭하면 다음 약관에 동의하는 것입니다:'; + + @override + String get download_agreement_1 => '알고 있습니다. 전 나쁜 사람입니다.'; + + @override + String get download_agreement_2 => + '제가 할 수 있는 모든 곳에서 아티스트를 지원할 것이며, 저는 그들의 작품을 살 돈이 없기 때문에 이렇게 하는 것뿐입니다.'; + + @override + String get download_agreement_3 => + '본인은 YouTube에서 내 IP가 차단될 수 있음을 완전히 알고 있으며, 현재 내 행동으로 인해 발생하는 사고에 대해 Spotube 또는 그 소유자/기여자에게 책임을 묻지 않습니다.'; + + @override + String get decline => '거절'; + + @override + String get accept => '동의'; + + @override + String get details => '상세'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => '채널'; + + @override + String get likes => '좋아요'; + + @override + String get dislikes => '싫어요'; + + @override + String get views => '조회수'; + + @override + String get streamUrl => '스트림 URL'; + + @override + String get stop => '중지'; + + @override + String get sort_newest => '최근에 추가된 순으로 정렬'; + + @override + String get sort_oldest => '예전에 추가된 순으로 정렬'; + + @override + String get sleep_timer => '취침 타이머'; + + @override + String mins(Object minutes) { + return '$minutes 분'; + } + + @override + String hours(Object hours) { + return '$hours 시간'; + } + + @override + String hour(Object hours) { + return '$hours 시간'; + } + + @override + String get custom_hours => '시간 설정'; + + @override + String get logs => '로그'; + + @override + String get developers => '개발'; + + @override + String get not_logged_in => '로그인하지 않았습니다'; + + @override + String get search_mode => '검색 모드'; + + @override + String get audio_source => '오디오 출처'; + + @override + String get ok => '알겠습니다'; + + @override + String get failed_to_encrypt => '암호화에 실패했습니다'; + + @override + String get encryption_failed_warning => + 'Spotube는 암호화를 사용하여 데이터를 안전하게 저장합니다. 하지만 그렇게 하지 못했습니다. 따라서 안전하지 않은 저장소로 대체됩니다.\n리눅스를 사용하는 경우, 비밀 서비스(gnome-keyring, kde-wallet, keepassxc 등)가 설치되어 있는지 확인하세요.'; + + @override + String get querying_info => '정보를 얻는 중...'; + + @override + String get piped_api_down => 'Piped API가 응답하지 않습니다'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Piped 인스턴스 $pipedInstance가 현재 다운되었습니다.\n\n인스턴스를 변경하거나 \'API 유형\'을 공식 YouTube API로 변경하세요.\n\n변경 후 앱을 다시 시작해야 합니다.'; + } + + @override + String get you_are_offline => '현재 오프라인입니다'; + + @override + String get connection_restored => '인터넷에 다시 연결되었습니다'; + + @override + String get use_system_title_bar => '시스템 타이틀바를 사용'; + + @override + String get crunching_results => '결과를 처리하는 중...'; + + @override + String get search_to_get_results => '결과를 얻으려면 검색해주세요'; + + @override + String get use_amoled_mode => 'AMOLED모드를 사용'; + + @override + String get pitch_dark_theme => '검정색 기반의 어두운 테마'; + + @override + String get normalize_audio => '오디오 노멀라이즈'; + + @override + String get change_cover => '커버 변경'; + + @override + String get add_cover => '커버 추가'; + + @override + String get restore_defaults => '기본값으로 복원'; + + @override + String get download_music_format => '다운로드 음악 포맷'; + + @override + String get streaming_music_format => '스트리밍 음악 포맷'; + + @override + String get download_music_quality => '다운로드 음질'; + + @override + String get streaming_music_quality => '스트리밍 음질'; + + @override + String get login_with_lastfm => 'Last.fm에 로그인'; + + @override + String get connect => '연결'; + + @override + String get disconnect_lastfm => 'Last.fm에서 연결 해제'; + + @override + String get disconnect => '연결 해제'; + + @override + String get username => '사용자명'; + + @override + String get password => '비밀번호'; + + @override + String get login => '로그인'; + + @override + String get login_with_your_lastfm => '내 Last.fm 계정으로로그인'; + + @override + String get scrobble_to_lastfm => 'Scrobble to Last.fm'; + + @override + String get go_to_album => '앨범으로 이동'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => '모두 탐색'; + + @override + String get genres => '장르'; + + @override + String get explore_genres => '장르 탐색'; + + @override + String get friends => '친구'; + + @override + String get no_lyrics_available => '죄송하지만 이 곡의 가사를 찾지 못했습니다'; + + @override + String get start_a_radio => '라디오 시작'; + + @override + String get how_to_start_radio => '라디오를 어떻게 시작하시겠습니까?'; + + @override + String get replace_queue_question => '현재 큐를 대체하시겠습니까 아니면 추가하시겠습니까?'; + + @override + String get endless_playback => '끝없는 재생'; + + @override + String get delete_playlist => '재생 목록 삭제'; + + @override + String get delete_playlist_confirmation => '이 재생 목록을 삭제하시겠습니까?'; + + @override + String get local_tracks => '로컬 트랙'; + + @override + String get local_tab => '로컬'; + + @override + String get song_link => '곡 링크'; + + @override + String get skip_this_nonsense => '이 허튼소리 건너뛰기'; + + @override + String get freedom_of_music => '“음악의 자유”'; + + @override + String get freedom_of_music_palm => '“손바닥 안의 음악의 자유”'; + + @override + String get get_started => '시작합시다'; + + @override + String get youtube_source_description => '추천되며 가장 잘 작동합니다.'; + + @override + String get piped_source_description => + '자유로운 기분이 듭니까? YouTube와 같지만 훨씬 더 무료합니다.'; + + @override + String get jiosaavn_source_description => '남아시아 지역에 최적입니다.'; + + @override + String get invidious_source_description => 'Piped와 비슷하지만 가용성이 높습니다.'; + + @override + String highest_quality(Object quality) { + return '최고 품질: $quality'; + } + + @override + String get select_audio_source => '오디오 소스 선택'; + + @override + String get endless_playback_description => '자동으로 새로운 노래를 대기열의 끝에 추가'; + + @override + String get choose_your_region => '지역 선택'; + + @override + String get choose_your_region_description => + '이것은 Spotube가 위치에 맞는 콘텐츠를 표시하는 데 도움이 됩니다.'; + + @override + String get choose_your_language => '언어 선택'; + + @override + String get help_project_grow => '이 프로젝트 성장에 도움을 주세요'; + + @override + String get help_project_grow_description => + 'Spotube는 오픈 소스 프로젝트입니다. 프로젝트에 기여하거나 버그를 보고하거나 새로운 기능을 제안하여이 프로젝트의 성장에 도움을 줄 수 있습니다.'; + + @override + String get contribute_on_github => 'GitHub에서 기여하기'; + + @override + String get donate_on_open_collective => 'Open Collective에 기부하기'; + + @override + String get browse_anonymously => '익명으로 둘러보기'; + + @override + String get enable_connect => '연결 활성화'; + + @override + String get enable_connect_description => '다른 장치에서 Spotube 제어'; + + @override + String get devices => '장치'; + + @override + String get select => '선택'; + + @override + String connect_client_alert(Object client) { + return '$client님에 의해 제어되고 있습니다'; + } + + @override + String get this_device => '이 장치'; + + @override + String get remote => '원격'; + + @override + String get stats => '통계'; + + @override + String and_n_more(Object count) { + return '그리고 $count개 더'; + } + + @override + String get recently_played => '최근 재생'; + + @override + String get browse_more => '더 보기'; + + @override + String get no_title => '제목 없음'; + + @override + String get not_playing => '재생 중이 아님'; + + @override + String get epic_failure => '서사적 실패!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length 곡을 대기열에 추가했습니다'; + } + + @override + String get spotube_has_an_update => 'Spotube에 업데이트가 있습니다'; + + @override + String get download_now => '지금 다운로드'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum이 출시되었습니다'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version이 출시되었습니다'; + } + + @override + String get read_the_latest => '최신 '; + + @override + String get release_notes => '릴리스 노트'; + + @override + String get pick_color_scheme => '색상 테마 선택'; + + @override + String get save => '저장'; + + @override + String get choose_the_device => '디바이스 선택:'; + + @override + String get multiple_device_connected => + '여러 디바이스가 연결되어 있습니다.\n이 작업을 실행할 디바이스를 선택하세요'; + + @override + String get nothing_found => '찾을 수 없음'; + + @override + String get the_box_is_empty => '상자가 비어 있습니다'; + + @override + String get top_artists => '톱 아티스트'; + + @override + String get top_albums => '톱 앨범'; + + @override + String get this_week => '이번 주'; + + @override + String get this_month => '이번 달'; + + @override + String get last_6_months => '지난 6개월'; + + @override + String get this_year => '올해'; + + @override + String get last_2_years => '지난 2년'; + + @override + String get all_time => '모든 시간'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName 제공'; + } + + @override + String get email => '이메일'; + + @override + String get profile_followers => '팔로워'; + + @override + String get birthday => '생일'; + + @override + String get subscription => '구독'; + + @override + String get not_born => '태어나지 않음'; + + @override + String get hacker => '해커'; + + @override + String get profile => '프로필'; + + @override + String get no_name => '이름 없음'; + + @override + String get edit => '편집'; + + @override + String get user_profile => '사용자 프로필'; + + @override + String count_plays(Object count) { + return '$count 재생'; + } + + @override + String get streaming_fees_hypothetical => + '*이것은 Spotify의 스트림당 지급액\n\$0.003에서 \$0.005를 기준으로 계산된 것입니다.\n이것은 사용자가 Spotify에서 곡을 들었을 때\n아티스트에게 지불했을 금액에 대한 통찰을 제공하기 위한\n가상의 계산입니다.'; + + @override + String get minutes_listened => '청취한 시간'; + + @override + String get streamed_songs => '스트리밍된 곡'; + + @override + String count_streams(Object count) { + return '$count 스트림'; + } + + @override + String get owned_by_you => '당신이 소유'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl를 클립보드에 복사했습니다'; + } + + @override + String get hipotetical_calculation => + '*이것은 온라인 음악 스트리밍 플랫폼의 스트림당 평균 지불액인 \$0.003에서 \$0.005를 기준으로 계산됩니다. 이것은 사용자가 다른 음악 스트리밍 플랫폼에서 노래를 들었다면 아티스트에게 얼마를 지불했을지에 대한 통찰력을 제공하기 위한 가상 계산입니다.'; + + @override + String count_mins(Object minutes) { + return '$minutes 분'; + } + + @override + String get summary_minutes => '분'; + + @override + String get summary_listened_to_music => '듣는 음악'; + + @override + String get summary_songs => '곡'; + + @override + String get summary_streamed_overall => '전체 스트리밍'; + + @override + String get summary_owed_to_artists => '이번 달 아티스트에게 지급해야 할 금액'; + + @override + String get summary_artists => '아티스트의'; + + @override + String get summary_music_reached_you => '음악이 도달함'; + + @override + String get summary_full_albums => '전체 앨범'; + + @override + String get summary_got_your_love => '당신의 사랑을 받음'; + + @override + String get summary_playlists => '플레이리스트'; + + @override + String get summary_were_on_repeat => '반복 재생됨'; + + @override + String total_money(Object money) { + return '총 $money'; + } + + @override + String get webview_not_found => '웹뷰를 찾을 수 없음'; + + @override + String get webview_not_found_description => + '기기에 웹뷰 런타임이 설치되지 않았습니다.\n설치되어 있으면 environment PATH에 있는지 확인하십시오\n\n설치 후 앱을 다시 시작하세요'; + + @override + String get unsupported_platform => '지원되지 않는 플랫폼'; + + @override + String get cache_music => '음악 캐시'; + + @override + String get open => '열기'; + + @override + String get cache_folder => '캐시 폴더'; + + @override + String get export => '내보내기'; + + @override + String get clear_cache => '캐시 지우기'; + + @override + String get clear_cache_confirmation => '캐시를 지우시겠습니까?'; + + @override + String get export_cache_files => '캐시된 파일 내보내기'; + + @override + String found_n_files(Object count) { + return '$count개의 파일을 찾았습니다'; + } + + @override + String get export_cache_confirmation => '이 파일들을 내보내시겠습니까'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$files개 중 $filesExported개 파일을 내보냈습니다'; + } + + @override + String get undo => '실행 취소'; + + @override + String get download_all => '모두 다운로드'; + + @override + String get add_all_to_playlist => '모두 재생 목록에 추가'; + + @override + String get add_all_to_queue => '모두 큐에 추가'; + + @override + String get play_all_next => '모두 다음에 재생'; + + @override + String get pause => '일시 정지'; + + @override + String get view_all => '모두 보기'; + + @override + String get no_tracks_added_yet => '아직 트랙을 추가하지 않은 것 같습니다'; + + @override + String get no_tracks => '여기에 트랙이 없는 것 같습니다'; + + @override + String get no_tracks_listened_yet => '아직 아무 것도 듣지 않은 것 같습니다'; + + @override + String get not_following_artists => '아티스트를 팔로우하지 않고 있습니다'; + + @override + String get no_favorite_albums_yet => '아직 즐겨찾기 앨범을 추가하지 않은 것 같습니다'; + + @override + String get no_logs_found => '로그를 찾을 수 없습니다'; + + @override + String get youtube_engine => 'YouTube 엔진'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine가 설치되지 않았습니다'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine가 시스템에 설치되지 않았습니다.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'PATH 변수에서 사용할 수 있는지 확인하거나\n아래에 $engine 실행 파일의 절대 경로를 설정하세요'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/unix와 같은 운영 체제에서는 .zshrc/.bashrc/.bash_profile 등에 경로 설정이 작동하지 않습니다.\n셸 구성 파일에 경로를 설정해야 합니다'; + + @override + String get download => '다운로드'; + + @override + String get file_not_found => '파일을 찾을 수 없습니다'; + + @override + String get custom => '사용자 정의'; + + @override + String get add_custom_url => '사용자 정의 URL 추가'; + + @override + String get edit_port => '포트 편집'; + + @override + String get port_helper_msg => + '기본값은 -1로 무작위 숫자를 나타냅니다. 방화벽이 구성된 경우 이를 설정하는 것이 좋습니다.'; + + @override + String connect_request(Object client) { + return '$client의 연결을 허용하시겠습니까?'; + } + + @override + String get connection_request_denied => '연결이 거부되었습니다. 사용자가 액세스를 거부했습니다.'; + + @override + String get an_error_occurred => '오류가 발생했습니다'; + + @override + String get copy_to_clipboard => '클립보드에 복사'; + + @override + String get view_logs => '로그 보기'; + + @override + String get retry => '다시 시도'; + + @override + String get no_default_metadata_provider_selected => + '기본 메타데이터 제공자가 설정되지 않았습니다'; + + @override + String get manage_metadata_providers => '메타데이터 제공자 관리'; + + @override + String get open_link_in_browser => '브라우저에서 링크를 여시겠습니까?'; + + @override + String get do_you_want_to_open_the_following_link => '다음 링크를 여시겠습니까'; + + @override + String get unsafe_url_warning => + '신뢰할 수 없는 출처의 링크를 여는 것은 안전하지 않을 수 있습니다. 주의하세요!\n링크를 클립보드에 복사할 수도 있습니다.'; + + @override + String get copy_link => '링크 복사'; + + @override + String get building_your_timeline => '청취 기록을 기반으로 타임라인을 구축하고 있습니다...'; + + @override + String get official => '공식'; + + @override + String author_name(Object author) { + return '저자: $author'; + } + + @override + String get third_party => '타사'; + + @override + String get plugin_requires_authentication => '플러그인에 인증이 필요합니다'; + + @override + String get update_available => '업데이트 사용 가능'; + + @override + String get supports_scrobbling => '스크로블링 지원'; + + @override + String get plugin_scrobbling_info => '이 플러그인은 음악을 스크로블하여 청취 기록을 생성합니다.'; + + @override + String get default_metadata_source => '기본 메타데이터 소스'; + + @override + String get set_default_metadata_source => '기본 메타데이터 소스 설정'; + + @override + String get default_audio_source => '기본 오디오 소스'; + + @override + String get set_default_audio_source => '기본 오디오 소스 설정'; + + @override + String get set_default => '기본값으로 설정'; + + @override + String get support => '지원'; + + @override + String get support_plugin_development => '플러그인 개발 지원'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API에 액세스할 수 있습니다'; + } + + @override + String get do_you_want_to_install_this_plugin => '이 플러그인을 설치하시겠습니까?'; + + @override + String get third_party_plugin_warning => + '이 플러그인은 타사 리포지토리에서 제공됩니다. 설치하기 전에 출처를 신뢰하는지 확인하세요.'; + + @override + String get author => '저자'; + + @override + String get this_plugin_can_do_following => '이 플러그인은 다음을 수행할 수 있습니다'; + + @override + String get install => '설치'; + + @override + String get install_a_metadata_provider => '메타데이터 제공자 설치'; + + @override + String get no_tracks_playing => '현재 재생 중인 트랙이 없습니다'; + + @override + String get synced_lyrics_not_available => '이 노래에 대한 동기화된 가사를 사용할 수 없습니다. 대신'; + + @override + String get plain_lyrics => '일반 가사'; + + @override + String get tab_instead => '탭을 사용하세요.'; + + @override + String get disclaimer => '면책 조항'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube 팀은 어떠한 \"타사\" 플러그인에 대해서도 (법적 포함) 어떠한 책임도 지지 않습니다.\n사용자 자신의 책임하에 사용하시기 바랍니다. 버그/문제에 대해서는 플러그인 리포지토리에 보고해 주세요.\n\n만약 \"타사\" 플러그인이 서비스/법인의 ToS/DMCA를 위반하는 경우, \"타사\" 플러그인 저자 또는 호스팅 플랫폼(예: GitHub/Codeberg)에 조치를 취하도록 요청해 주세요. 위에 나열된 (\"타사\"로 표시된) 플러그인은 모두 공개/커뮤니티에서 유지 관리하는 플러그인입니다. 저희는 이를 큐레이션하지 않으므로 어떠한 조치도 취할 수 없습니다.\n\n'; + + @override + String get input_does_not_match_format => '입력이 필요한 형식과 일치하지 않습니다'; + + @override + String get plugins => '플러그인'; + + @override + String get paste_plugin_download_url => + '다운로드 URL, GitHub/Codeberg 리포지토리 URL 또는 .smplug 파일에 대한 직접 링크를 붙여넣으세요'; + + @override + String get download_and_install_plugin_from_url => 'URL에서 플러그인 다운로드 및 설치'; + + @override + String failed_to_add_plugin_error(Object error) { + return '플러그인 추가 실패: $error'; + } + + @override + String get upload_plugin_from_file => '파일에서 플러그인 업로드'; + + @override + String get installed => '설치됨'; + + @override + String get available_plugins => '사용 가능한 플러그인'; + + @override + String get configure_plugins => '직접 메타데이터 제공자와 오디오 소스 플러그인을 구성하세요'; + + @override + String get audio_scrobblers => '오디오 스크로블러'; + + @override + String get scrobbling => '스크로블링'; + + @override + String get source => '출처: '; + + @override + String get uncompressed => '비압축'; + + @override + String get dab_music_source_description => + '오디오파일을 위한 소스입니다. 고음질/무손실 오디오 스트림을 제공하며 ISRC 기반으로 정확한 트랙 매칭을 지원합니다.'; +} diff --git a/lib/l10n/generated/app_localizations_ne.dart b/lib/l10n/generated/app_localizations_ne.dart new file mode 100644 index 00000000..8f881b51 --- /dev/null +++ b/lib/l10n/generated/app_localizations_ne.dart @@ -0,0 +1,1578 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Nepali (`ne`). +class AppLocalizationsNe extends AppLocalizations { + AppLocalizationsNe([String locale = 'ne']) : super(locale); + + @override + String get guest => 'अतिथि'; + + @override + String get browse => 'ब्राउज़ गर्नुहोस्'; + + @override + String get search => 'खोजी गर्नुहोस्'; + + @override + String get library => 'पुस्तकालय'; + + @override + String get lyrics => 'गीतको शब्द'; + + @override + String get settings => 'सेटिङ'; + + @override + String get genre_categories_filter => 'शैली वा शैलीहरू फिल्टर गर्नुहोस्...'; + + @override + String get genre => 'शैली'; + + @override + String get personalized => 'व्यक्तिगत'; + + @override + String get featured => 'विशेष'; + + @override + String get new_releases => 'नयाँ रिलिज'; + + @override + String get songs => 'गीतहरू'; + + @override + String playing_track(Object track) { + return '$track बज्यो'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'यो हालको कतारलाई हटाउँछ। $track_length ट्र्याकहरू हटाईन्छ\nके तपाईं जारी राख्न चाहनुहुन्छ?'; + } + + @override + String get load_more => 'थप लोड गर्नुहोस्'; + + @override + String get playlists => 'प्लेलिस्टहरू'; + + @override + String get artists => 'कलाकारहरू'; + + @override + String get albums => 'आल्बमहरू'; + + @override + String get tracks => 'ट्र्याकहरू'; + + @override + String get downloads => 'डाउनलोडहरू'; + + @override + String get filter_playlists => 'तपाईंको प्लेलिस्टहरू फिल्टर गर्नुहोस्...'; + + @override + String get liked_tracks => 'मन परेका ट्र्याकहरू'; + + @override + String get liked_tracks_description => 'तपाईंको मन परेका सबै ट्र्याकहरू'; + + @override + String get playlist => 'प्लेलिस्ट'; + + @override + String get create_a_playlist => 'प्लेलिस्ट बनाउनुहोस्'; + + @override + String get update_playlist => 'प्लेलिस्ट अपडेट गर्नुहोस्'; + + @override + String get create => 'बनाउनुहोस्'; + + @override + String get cancel => 'रद्द गर्नुहोस्'; + + @override + String get update => 'अपडेट गर्नुहोस्'; + + @override + String get playlist_name => 'प्लेलिस्टको नाम'; + + @override + String get name_of_playlist => 'प्लेलिस्टको नाम'; + + @override + String get description => 'विवरण'; + + @override + String get public => 'सार्वजनिक'; + + @override + String get collaborative => 'सहकारी'; + + @override + String get search_local_tracks => 'स्थानीय ट्र्याकहरू खोजी गर्नुहोस्...'; + + @override + String get play => 'बजाउनुहोस्'; + + @override + String get delete => 'मेटाउनुहोस्'; + + @override + String get none => 'कुनै पनि होइन'; + + @override + String get sort_a_z => 'A-Zमा क्रमबद्ध गर्नुहोस्'; + + @override + String get sort_z_a => 'Z-Aमा क्रमबद्ध गर्नुहोस्'; + + @override + String get sort_artist => 'कलाकारबाट क्रमबद्ध गर्नुहोस्'; + + @override + String get sort_album => 'आल्बमबाट क्रमबद्ध गर्नुहोस्'; + + @override + String get sort_duration => 'अवधिको अनुसार क्रमबद्ध गर्नुहोस्'; + + @override + String get sort_tracks => 'ट्र्याकहरूलाई क्रमबद्ध गर्नुहोस्'; + + @override + String currently_downloading(Object tracks_length) { + return 'हाल डाउनलोड गर्दैछ ($tracks_length)'; + } + + @override + String get cancel_all => 'सब रद्द गर्नुहोस्'; + + @override + String get filter_artist => 'कलाकारहरूलाई फिल्टर गर्नुहोस्...'; + + @override + String followers(Object followers) { + return '$followers अनुयायीहरू'; + } + + @override + String get add_artist_to_blacklist => 'कलाकारलाई कालोसूचीमा थप्नुहोस्'; + + @override + String get top_tracks => 'शीर्ष ट्र्याकहरू'; + + @override + String get fans_also_like => 'अनुयायीहरू पनि लाइक गर्छन्'; + + @override + String get loading => 'लोड हुँदैछ...'; + + @override + String get artist => 'कलाकार'; + + @override + String get blacklisted => 'कालोसूचीमा'; + + @override + String get following => 'फल्लो गर्दै'; + + @override + String get follow => 'फल्लो गर्नुहोस्'; + + @override + String get artist_url_copied => 'कलाकार URL क्लिपबोर्डमा प्रतिलिपि गरिएको छ'; + + @override + String added_to_queue(Object tracks) { + return '$tracks ट्र्याकहरूलाई कतारमा थपिएको छ'; + } + + @override + String get filter_albums => 'आल्बमहरूलाई फिल्टर गर्नुहोस्...'; + + @override + String get synced => 'सिङ्क गरिएको'; + + @override + String get plain => 'साधा'; + + @override + String get shuffle => 'शफल'; + + @override + String get search_tracks => 'ट्र्याकहरू खोजी गर्नुहोस्...'; + + @override + String get released => 'रिलिज गरिएको'; + + @override + String error(Object error) { + return 'त्रुटि $error'; + } + + @override + String get title => 'शीर्षक'; + + @override + String get time => 'समय'; + + @override + String get more_actions => 'थप कार्यहरू'; + + @override + String download_count(Object count) { + return 'डाउनलोड ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'प्लेलिस्टमा थप्नुहोस् ($count)'; + } + + @override + String add_count_to_queue(Object count) { + return 'कतारमा थप्नुहोस् ($count)'; + } + + @override + String play_count_next(Object count) { + return 'प्लेगरी गर्नुहोस् ($count)'; + } + + @override + String get album => 'आल्बम'; + + @override + String copied_to_clipboard(Object data) { + return '$data क्लिपबोर्डमा प्रतिलिपि गरिएको छ'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track लाई तलका प्लेलिस्टमा थप्नुहोस्'; + } + + @override + String get add => 'थप्नुहोस्'; + + @override + String added_track_to_queue(Object track) { + return '$track लाई कतारमा थपिएको छ'; + } + + @override + String get add_to_queue => 'कतारमा थप्नुहोस्'; + + @override + String track_will_play_next(Object track) { + return '$track अरूलाई पहिलोमा बज्नेछ'; + } + + @override + String get play_next => 'पछिबजाउनुहोस्'; + + @override + String removed_track_from_queue(Object track) { + return '$track लाई कतारबाट हटाइएको छ'; + } + + @override + String get remove_from_queue => 'कतारबाट हटाउनुहोस्'; + + @override + String get remove_from_favorites => 'पसन्दीदामा बाट हटाउनुहोस्'; + + @override + String get save_as_favorite => 'पसन्दीदा बनाउनुहोस्'; + + @override + String get add_to_playlist => 'प्लेलिस्टमा थप्नुहोस्'; + + @override + String get remove_from_playlist => 'प्लेलिस्टबाट हटाउनुहोस्'; + + @override + String get add_to_blacklist => 'कालोसूचीमा थप्नुहोस्'; + + @override + String get remove_from_blacklist => 'कालोसूचीबाट हटाउनुहोस्'; + + @override + String get share => 'साझा गर्नुहोस्'; + + @override + String get mini_player => 'मिनि प्लेयर'; + + @override + String get slide_to_seek => + 'अगाडि वा पछाडि खोजी गर्नका लागि स्लाइड गर्नुहोस्'; + + @override + String get shuffle_playlist => 'प्लेलिस्ट शफल गर्नुहोस्'; + + @override + String get unshuffle_playlist => 'प्लेलिस्ट शफल नगर्नुहोस्'; + + @override + String get previous_track => 'पूर्व ट्र्याक'; + + @override + String get next_track => 'अरू ट्र्याक'; + + @override + String get pause_playback => 'प्लेब्याक रोक्नुहोस्'; + + @override + String get resume_playback => 'प्लेब्याक पुनः सुरु गर्नुहोस्'; + + @override + String get loop_track => 'ट्र्याकलाई दोहोरोपट्टी बजाउनुहोस्'; + + @override + String get no_loop => 'कोई लूप नहीं'; + + @override + String get repeat_playlist => 'प्लेलिस्ट पुनः बजाउनुहोस्'; + + @override + String get queue => 'कतार'; + + @override + String get alternative_track_sources => 'वैकल्पिक ट्र्याक स्रोतहरू'; + + @override + String get download_track => 'ट्र्याक डाउनलोड गर्नुहोस्'; + + @override + String tracks_in_queue(Object tracks) { + return 'कतारमा $tracks ट्र्याकहरू'; + } + + @override + String get clear_all => 'सब मेटाउनुहोस्'; + + @override + String get show_hide_ui_on_hover => 'हवर गरेपछि UI देखाउनुहोस्/लुकाउनुहोस्'; + + @override + String get always_on_top => 'सधैं टपमा राख्नुहोस्'; + + @override + String get exit_mini_player => 'मिनि प्लेयर बाट बाहिर निस्कनुहोस्'; + + @override + String get download_location => 'डाउनलोड स्थान'; + + @override + String get local_library => 'स्थानिय पुस्तकालय'; + + @override + String get add_library_location => 'पुस्तकालयमा थप्नुहोस्'; + + @override + String get remove_library_location => 'पुस्तकालयबाट हटाउनुहोस्'; + + @override + String get account => 'खाता'; + + @override + String get logout => 'बाहिर निस्कनुहोस्'; + + @override + String get logout_of_this_account => 'यो खाताबाट बाहिर निस्कनुहोस्'; + + @override + String get language_region => 'भाषा र क्षेत्र'; + + @override + String get language => 'भाषा'; + + @override + String get system_default => 'सिस्टम पूर्वनिर्धारित'; + + @override + String get market_place_region => 'बजार स्थान'; + + @override + String get recommendation_country => 'सिफारिस गरिएको देश'; + + @override + String get appearance => 'दृष्टिकोण'; + + @override + String get layout_mode => 'लेआउट मोड'; + + @override + String get override_layout_settings => + 'अनुकूलित प्रतिकृयात्मक लेआउट मोड सेटिङ्गहरू'; + + @override + String get adaptive => 'अनुकूलित'; + + @override + String get compact => 'संकुचित'; + + @override + String get extended => 'बढाइएको'; + + @override + String get theme => 'थिम'; + + @override + String get dark => 'गाढा'; + + @override + String get light => 'प्रकाश'; + + @override + String get system => 'सिस्टम'; + + @override + String get accent_color => 'एक्सेन्ट रङ्ग'; + + @override + String get sync_album_color => 'एल्बम रङ्ग सिङ्क गर्नुहोस्'; + + @override + String get sync_album_color_description => + 'एल्बम कला को प्रमुख रङ्गलाई एक्सेन्ट रङ्गको रूपमा प्रयोग गर्दछ'; + + @override + String get playback => 'प्लेब्याक'; + + @override + String get audio_quality => 'आडियो गुणस्तर'; + + @override + String get high => 'उच्च'; + + @override + String get low => 'न्यून'; + + @override + String get pre_download_play => 'पूर्व-डाउनलोड र प्ले गर्नुहोस्'; + + @override + String get pre_download_play_description => + 'आडियो स्ट्रिम गर्नु नगरी बाइटहरू डाउनलोड गरी बजाउँछ (उच्च ब्यान्डविथ उपयोगकर्ताहरूको लागि सिफारिस गरिएको)'; + + @override + String get skip_non_music => + 'गीतहरू बाहेक कुनै अनुष्ठान छोड्नुहोस् (स्पन्सरब्लक)'; + + @override + String get blacklist_description => 'कालोसूची गीत र कलाकारहरू'; + + @override + String get wait_for_download_to_finish => + 'कृपया हालको डाउनलोड समाप्त हुन लागि पर्खनुहोस्'; + + @override + String get desktop => 'डेस्कटप'; + + @override + String get close_behavior => 'बन्द व्यवहार'; + + @override + String get close => 'बन्द गर्नुहोस्'; + + @override + String get minimize_to_tray => 'ट्रेमा कम गर्नुहोस्'; + + @override + String get show_tray_icon => 'सिस्टम ट्रे आइकन देखाउनुहोस्'; + + @override + String get about => 'बारेमा'; + + @override + String get u_love_spotube => + 'हामीले थाहा पारेका छौं तपाईंलाई Spotube मन पर्छ'; + + @override + String get check_for_updates => 'अपडेटहरूको लागि जाँच गर्नुहोस्'; + + @override + String get about_spotube => 'Spotube को बारेमा'; + + @override + String get blacklist => 'कालोसूची'; + + @override + String get please_sponsor => 'कृपया स्पन्सर/डोनेट गर्नुहोस्'; + + @override + String get spotube_description => + 'Spotube, एक हल्का, समृद्ध, स्वतन्त्र Spotify क्लाइयन'; + + @override + String get version => 'संस्करण'; + + @override + String get build_number => 'निर्माण नम्बर'; + + @override + String get founder => 'संस्थापक'; + + @override + String get repository => 'पुनरावलोकन स्थल'; + + @override + String get bug_issues => 'त्रुटि + समस्याहरू'; + + @override + String get made_with => '❤️ 2021-2024 बाट बनाइएको'; + + @override + String get kingkor_roy_tirtho => 'किङ्कोर राय तिर्थो'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year किङ्कोर राय तिर्थो'; + } + + @override + String get license => 'लाइसेन्स'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'चिन्ता नगर्नुहोस्, तपाईंको कुनै पनि क्रेडेन्शियलहरूले कसैले संग्रह वा साझा गर्नेछैन'; + + @override + String get know_how_to_login => 'कसरी लगिन गर्ने भन्ने थाहा छैन?'; + + @override + String get follow_step_by_step_guide => + 'चरणबद्ध मार्गदर्शनमा साथी बनाउनुहोस्'; + + @override + String cookie_name_cookie(Object name) { + return '$name कुकी'; + } + + @override + String get fill_in_all_fields => 'कृपया सबै क्षेत्रहरू भर्नुहोस्'; + + @override + String get submit => 'पेश गर्नुहोस्'; + + @override + String get exit => 'बाहिर निस्कनुहोस्'; + + @override + String get previous => 'पूर्ववत'; + + @override + String get next => 'अरू'; + + @override + String get done => 'गरिएको'; + + @override + String get step_1 => 'कदम 1'; + + @override + String get first_go_to => 'पहिलो, जानुहोस्'; + + @override + String get something_went_wrong => 'केहि गल्ति भएको छ'; + + @override + String get piped_instance => 'पाइपड सर्भर इन्स्ट्यान्स'; + + @override + String get piped_description => + 'गीत मिलाउको लागि प्रयोग गर्ने पाइपड सर्भर इन्स्ट्यान्स'; + + @override + String get piped_warning => + 'तिनीहरूमध्ये केहि ठिक गर्न सक्छ। यसलाई आफ्नो जोखिममा प्रयोग गर्नुहोस्'; + + @override + String get invidious_instance => 'Invidious सर्भर इन्स्टेन्स'; + + @override + String get invidious_description => + 'ट्र्याक मिलाउनका लागि प्रयोग हुने Invidious सर्भर इन्स्टेन्स'; + + @override + String get invidious_warning => + 'केहीले राम्रोसँग काम नगर्न सक्छ। आफ्नो जोखिममा प्रयोग गर्नुहोस्'; + + @override + String get generate => 'जनरेट'; + + @override + String track_exists(Object track) { + return 'ट्र्याक $track पहिले नै छ'; + } + + @override + String get replace_downloaded_tracks => + 'सबै डाउनलोड गरिएका ट्र्याकहरूलाई परिवर्तन गर्नुहोस्'; + + @override + String get skip_download_tracks => + 'सबै डाउनलोड गरिएका ट्र्याकहरूलाई छोड्नुहोस्'; + + @override + String get do_you_want_to_replace => + 'के तपाईंले वर्तमान ट्र्याकलाई परिवर्तन गर्न चाहनुहुन्छ?'; + + @override + String get replace => 'परिवर्तन गर्नुहोस्'; + + @override + String get skip => 'छोड्नुहोस्'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '$count $type सम्म चयन गर्नुहोस्'; + } + + @override + String get select_genres => 'जनरहरू चयन गर्नुहोस्'; + + @override + String get add_genres => 'जनरहरू थप्नुहोस्'; + + @override + String get country => 'देश'; + + @override + String get number_of_tracks_generate => 'बनाउनका लागि ट्र्याकहरूको संख्या'; + + @override + String get acousticness => 'एकोस्टिकनेस'; + + @override + String get danceability => 'नृत्यक्षमता'; + + @override + String get energy => 'ऊर्जा'; + + @override + String get instrumentalness => 'साजा रहेकोता'; + + @override + String get liveness => 'प्राणिकता'; + + @override + String get loudness => 'शोर'; + + @override + String get speechiness => 'भाषण'; + + @override + String get valence => 'मानसिक स्वभाव'; + + @override + String get popularity => 'लोकप्रियता'; + + @override + String get key => 'कुञ्जी'; + + @override + String get duration => 'अवधि (सेकेण्ड)'; + + @override + String get tempo => 'गति (बीपीएम)'; + + @override + String get mode => 'मोड'; + + @override + String get time_signature => 'समय हस्ताक्षर'; + + @override + String get short => 'सानो'; + + @override + String get medium => 'मध्यम'; + + @override + String get long => 'लामो'; + + @override + String get min => 'न्यून'; + + @override + String get max => 'अधिक'; + + @override + String get target => 'लक्ष्य'; + + @override + String get moderate => 'मध्यस्थ'; + + @override + String get deselect_all => 'सबै छान्नुहोस्'; + + @override + String get select_all => 'सबै चयन गर्नुहोस्'; + + @override + String get are_you_sure => 'के तपाईं सुनिश्चित हुनुहुन्छ?'; + + @override + String get generating_playlist => 'तपाईंको विशेष प्लेलिस्ट बनाइएको छ...'; + + @override + String selected_count_tracks(Object count) { + return '$count ट्र्याकहरू छन् चयन गरिएका'; + } + + @override + String get download_warning => + 'यदि तपाईं सबै ट्र्याकहरूलाई बल्कमा डाउनलोड गर्छनु हो भने तपाईं स्पष्ट रूपमा साङ्गीत चोरी गरिरहेका छन् र यो साङ्गीतको रचनात्मक समाजलाई क्षति पनि पुर्याउँछ। उमेराइएको छ कि तपाईं यसको बारेमा जागरूक छिनुहुन्छ। सधैं, कला गर्दै र कलाकारको कडा परम्परा समर्थन गर्दै आइन्छ।'; + + @override + String get download_ip_ban_warning => + 'बितिएका डाउनलोड अनुरोधहरूका कारण तपाईंको आइपीले YouTube मा ब्लक हुन सक्छ। आइपी ब्लक भनेको कम्तीमा 2-3 महिनासम्म तपाईं त्यस आइपी यन्त्रबाट YouTube प्रयोग गर्न सक्नुहुन्छ। र यदि यो हुँदैछ भने स्पट्यूबले यसलाई कसैले गरेको बारेमा कुनै दायित्व लिन्छैन।'; + + @override + String get by_clicking_accept_terms => + '\'स्वीकृत\' गरेर तपाईं निम्नलिखित निर्वाचन गर्दैछिन्:'; + + @override + String get download_agreement_1 => + 'म मन्ने छु कि म साङ्गीत चोरी गरिरहेको छु। म बुरो हुँ'; + + @override + String get download_agreement_2 => + 'म कहिल्यै कहिल्यै तिनीहरूलाई समर्थन गर्नेछु र म यो तिनीहरूको कला किन्ने पैसा छैन भने मा मात्र यो गरेको छु'; + + @override + String get download_agreement_3 => + 'म पूरा रूपमा जान्छु कि मेरो आइपी YouTube मा ब्लक हुन सक्छ र म मन्छेहरूले मेरो चासोबाट भएको कुनै दुर्घटनामा स्पट्यूब वा तिनीहरूको मालिकहरू/सहयोगीहरूलाई दायित्वी ठान्छुँभन्ने पूर्ण जानकारी छैन'; + + @override + String get decline => 'अस्वीकृत'; + + @override + String get accept => 'स्वीकृत'; + + @override + String get details => 'विवरण'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'च्यानल'; + + @override + String get likes => 'लाइकहरू'; + + @override + String get dislikes => 'असुनुहरू'; + + @override + String get views => 'हेरिएको'; + + @override + String get streamUrl => 'स्ट्रिम यूआरएल'; + + @override + String get stop => 'रोक्नुहोस्'; + + @override + String get sort_newest => 'नयाँ थपिएकोमा क्रमबद्ध गर्नुहोस्'; + + @override + String get sort_oldest => 'पुरानो थपिएकोमा क्रमबद्ध गर्नुहोस्'; + + @override + String get sleep_timer => 'सुत्ने टाइमर'; + + @override + String mins(Object minutes) { + return '$minutes मिनेटहरू'; + } + + @override + String hours(Object hours) { + return '$hours घण्टाहरू'; + } + + @override + String hour(Object hours) { + return '$hours घण्टा'; + } + + @override + String get custom_hours => 'कस्टम घण्टाहरू'; + + @override + String get logs => 'लगहरू'; + + @override + String get developers => 'डेभेलपर्स'; + + @override + String get not_logged_in => 'तपाईंले लगइन गरेका छैनौं'; + + @override + String get search_mode => 'खोज मोड'; + + @override + String get audio_source => 'अडियो स्रोत'; + + @override + String get ok => 'ठिक छ'; + + @override + String get failed_to_encrypt => 'एन्क्रिप्ट गर्न सकिएन'; + + @override + String get encryption_failed_warning => + 'स्पट्यूबले तपाईंको डेटा सुरक्षित रूपमा स्टोर गर्नका लागि एन्क्रिप्ट गर्न खोजेको छ। तर यसले गरेको छैन। यसले असुरक्षित स्टोरेजमा फल्लब्याक गर्दछ\nयदि तपाईंले लिनक्स प्रयोग गरिरहेका छन् भने कृपया सुनिश्चित गर्नुहोस् कि तपाईंले कुनै सीक्रेट-सर्भिस (गोनोम-किरिङ, केडीइ-वालेट, किपासेक्ससि इत्यादि) इन्स्टल गरेका छौं'; + + @override + String get querying_info => 'जानकारी हेर्दै...'; + + @override + String get piped_api_down => 'पाइपड एपीआई डाउन छ'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'पाइपड इन्स्ट्यान्स $pipedInstance हाल डाउन छ\n\nजीसनै इन्स्ट्यान्स परिवर्तन गर्नुहोस् वा \'एपीआई प्रकार\' लाइ YouTube आफिसियल एपीआईमा परिवर्तन गर्नुहोस्\n\nपरिवर्तनपछि एप्लिकेसन पुन: सुरु गर्नुहोस्'; + } + + @override + String get you_are_offline => 'तपाईं वर्तमान अफलाइन हुनुहुन्छ'; + + @override + String get connection_restored => + 'तपाईंको इन्टरनेट कनेक्सन पुन: स्थापित भएको छ'; + + @override + String get use_system_title_bar => 'सिस्टम शीर्षक पट्टी प्रयोग गर्नुहोस्'; + + @override + String get crunching_results => 'परिणामहरू कपालबाट पीस्दै...'; + + @override + String get search_to_get_results => + 'परिणामहरू प्राप्त गर्नका लागि खोज्नुहोस्'; + + @override + String get use_amoled_mode => 'कृष्ण ब्ल्याक गाढा थिम प्रयोग गर्नुहोस्'; + + @override + String get pitch_dark_theme => 'एमोलेड मोड'; + + @override + String get normalize_audio => 'अडियो सामान्य गर्नुहोस्'; + + @override + String get change_cover => 'कवर परिवर्तन गर्नुहोस्'; + + @override + String get add_cover => 'कवर थप्नुहोस्'; + + @override + String get restore_defaults => 'पूर्वनिर्धारितहरू पुनः स्थापित गर्नुहोस्'; + + @override + String get download_music_format => 'सङ्गीत डाउनलोड ढाँचा'; + + @override + String get streaming_music_format => 'स्ट्रिमिङ सङ्गीत ढाँचा'; + + @override + String get download_music_quality => 'डाउनलोड गुणस्तर'; + + @override + String get streaming_music_quality => 'स्ट्रिमिङ गुणस्तर'; + + @override + String get login_with_lastfm => 'लास्ट.एफ.एम सँग लगइन गर्नुहोस्'; + + @override + String get connect => 'जडान गर्नुहोस्'; + + @override + String get disconnect_lastfm => 'लास्ट.एफ.एम डिसकनेक्ट गर्नुहोस्'; + + @override + String get disconnect => 'डिसकनेक्ट'; + + @override + String get username => 'प्रयोगकर्ता नाम'; + + @override + String get password => 'पासवर्ड'; + + @override + String get login => 'लगइन'; + + @override + String get login_with_your_lastfm => + 'तपाईंको लास्ट.एफ.एम खातामा लगइन गर्नुहोस्'; + + @override + String get scrobble_to_lastfm => 'लास्ट.एफ.एम मा स्क्रबल गर्नुहोस्'; + + @override + String get go_to_album => 'आल्बममा जानुहोस्'; + + @override + String get discord_rich_presence => 'डिस्कर्ड धनी उपस्थिति'; + + @override + String get browse_all => 'सबै हेर्नुहोस्'; + + @override + String get genres => 'शैलीहरू'; + + @override + String get explore_genres => 'शैलीहरू अन्वेषण गर्नुहोस्'; + + @override + String get friends => 'साथीहरू'; + + @override + String get no_lyrics_available => + 'क्षमा गर्दैछौं, यस ट्र्याकका लागि गीतका शब्दहरू फेला परेन'; + + @override + String get start_a_radio => 'रेडियो सुरु गर्नुहोस्'; + + @override + String get how_to_start_radio => 'तपाईं रेडियो कसरी सुरु गर्न चाहानुहुन्छ?'; + + @override + String get replace_queue_question => + 'के तपाईं वर्तमान कताक्ष कोट बदल्न चाहानुहुन्छ वा यसलाई थप्नुहुन्छ?'; + + @override + String get endless_playback => 'अनन्त प्लेब्याक'; + + @override + String get delete_playlist => 'प्लेलिस्ट मेटाउनुहोस्'; + + @override + String get delete_playlist_confirmation => + 'के तपाईं यो प्लेलिस्ट मेटाउन निश्चित हुनुहुन्छ?'; + + @override + String get local_tracks => 'स्थानिय ट्र्याकहरू'; + + @override + String get local_tab => 'स्थानिय'; + + @override + String get song_link => 'गीत लिंक'; + + @override + String get skip_this_nonsense => 'यस अबश्यकता छोड्नुहोस्'; + + @override + String get freedom_of_music => '“संगीतको स्वतन्त्रता”'; + + @override + String get freedom_of_music_palm => '“तपाईंको हातमा संगीतको स्वतन्त्रता”'; + + @override + String get get_started => 'आइयाँ प्रारम्भ गरौं'; + + @override + String get youtube_source_description => 'सिफारिस गरिएको र बेस्ट काम गर्दछ।'; + + @override + String get piped_source_description => + 'मुक्त सुस्त? YouTube जस्तै तर धेरै मुक्त।'; + + @override + String get jiosaavn_source_description => + 'दक्षिण एशियाली क्षेत्रको लागि सर्वोत्तम।'; + + @override + String get invidious_source_description => 'Piped जस्तै तर उच्च उपलब्धतासँग।'; + + @override + String highest_quality(Object quality) { + return 'उच्चतम गुणस्तर: $quality'; + } + + @override + String get select_audio_source => 'आडियो स्रोत चयन गर्नुहोस्'; + + @override + String get endless_playback_description => + 'नयाँ गीतहरूलाई स्वचालित रूपमा कताक्षको अन्तमा जोड्नुहोस्'; + + @override + String get choose_your_region => 'तपाईंको क्षेत्र छनौट गर्नुहोस्'; + + @override + String get choose_your_region_description => + 'यो Spotubeलाई तपाईंको स्थानका लागि सहि सामग्री देखाउने मद्दत गर्नेछ।'; + + @override + String get choose_your_language => 'तपाईंको भाषा छनौट गर्नुहोस्'; + + @override + String get help_project_grow => 'यस परियोजनामा वृद्धि गराउनुहोस्'; + + @override + String get help_project_grow_description => + 'Spotube एक खुला स्रोतको परियोजना हो। तपाईं परियोजनामा योगदान गरेर, त्रुटिहरू सूचिकै, वा नयाँ सुविधाहरू सुझाव दिएर यस परियोजनामा वृद्धि गर्न सक्नुहुन्छ।'; + + @override + String get contribute_on_github => 'GitHubमा योगदान गर्नुहोस्'; + + @override + String get donate_on_open_collective => 'खुला संगठनमा दान गर्नुहोस्'; + + @override + String get browse_anonymously => 'अनामित रूपमा ब्राउज़ गर्नुहोस्'; + + @override + String get enable_connect => 'कनेक्ट सक्रिय गर्नुहोस्'; + + @override + String get enable_connect_description => + 'अन्य उपकरणहरूबाट Spotube कन्ट्रोल गर्नुहोस्'; + + @override + String get devices => 'उपकरणहरू'; + + @override + String get select => 'चयन गर्नुहोस्'; + + @override + String connect_client_alert(Object client) { + return 'तपाईंलाई $client द्वारा नियन्त्रित गरिएको छ'; + } + + @override + String get this_device => 'यो उपकरण'; + + @override + String get remote => 'दूरसंचार'; + + @override + String get stats => 'तथ्याङ्क'; + + @override + String and_n_more(Object count) { + return 'राम्रो $count थप'; + } + + @override + String get recently_played => 'हालै खेलेको'; + + @override + String get browse_more => 'थप हेर्नुहोस्'; + + @override + String get no_title => 'शीर्षक छैन'; + + @override + String get not_playing => 'खेलिरहेको छैन'; + + @override + String get epic_failure => 'महाकवि असफलता!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length ट्र्याकहरू तालिकामा थपिएका छन्'; + } + + @override + String get spotube_has_an_update => 'Spotube मा अपडेट छ'; + + @override + String get download_now => 'अहिले डाउनलोड गर्नुहोस्'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum रिलिज गरिएको छ'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version रिलिज गरिएको छ'; + } + + @override + String get read_the_latest => 'अर्को '; + + @override + String get release_notes => 'रिलिज नोटहरू'; + + @override + String get pick_color_scheme => 'रंग योजना चयन गर्नुहोस्'; + + @override + String get save => 'सुरक्षित गर्नुहोस्'; + + @override + String get choose_the_device => 'उपकरण चयन गर्नुहोस्:'; + + @override + String get multiple_device_connected => + 'धेरै उपकरण जडान गरिएको छ।\nयो क्रियाकलाप गर्ने उपकरण चयन गर्नुहोस्'; + + @override + String get nothing_found => 'केही फेला परेन'; + + @override + String get the_box_is_empty => 'बक्स खाली छ'; + + @override + String get top_artists => 'शीर्ष कलाकारहरू'; + + @override + String get top_albums => 'शीर्ष एल्बमहरू'; + + @override + String get this_week => 'यो हप्ता'; + + @override + String get this_month => 'यो महिना'; + + @override + String get last_6_months => 'पछिल्लो ६ महिना'; + + @override + String get this_year => 'यो वर्ष'; + + @override + String get last_2_years => 'पछिल्लो २ वर्ष'; + + @override + String get all_time => 'सबै समय'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName द्वारा शक्ति प्राप्त'; + } + + @override + String get email => 'ईमेल'; + + @override + String get profile_followers => 'अनुयायीहरू'; + + @override + String get birthday => 'जन्मदिन'; + + @override + String get subscription => 'सदस्यता'; + + @override + String get not_born => 'जन्मिएको छैन'; + + @override + String get hacker => 'ह्याकर'; + + @override + String get profile => 'प्रोफाइल'; + + @override + String get no_name => 'नाम छैन'; + + @override + String get edit => 'सम्पादन गर्नुहोस्'; + + @override + String get user_profile => 'प्रयोगकर्ता प्रोफाइल'; + + @override + String count_plays(Object count) { + return '$count खेलाइन्छ'; + } + + @override + String get streaming_fees_hypothetical => + '*यो Spotify को प्रति स्ट्रिमको आधारमा गणना गरिएको छ\n\$0.003 देखि \$0.005 बीचको भुक्तानी। यो एक काल्पनिक गणना हो\nउपयोगकर्तालाई यो थाहा दिनको लागि कि उनीहरूले अर्टिस्टहरूलाई\nSpotify मा गीत सुनेको भए कति भुक्तानी गर्ने थिए।'; + + @override + String get minutes_listened => 'सुनिएका मिनेटहरू'; + + @override + String get streamed_songs => 'स्ट्रीम गरिएका गीतहरू'; + + @override + String count_streams(Object count) { + return '$count स्ट्रिम'; + } + + @override + String get owned_by_you => 'तपाईंले स्वामित्व गरेको'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl क्लिपबोर्डमा कपी गरियो'; + } + + @override + String get hipotetical_calculation => + '*यो अनलाइन संगीत स्ट्रिमिङ प्लेटफर्मको प्रति स्ट्रिम भुक्तानी \$0.003 देखि \$0.005 को औसतमा आधारित छ। यो एक काल्पनिक गणना हो जुन प्रयोगकर्तालाई उनीहरूले विभिन्न संगीत स्ट्रिमिङ प्लेटफर्ममा आफ्ना गीतहरू सुनेमा कलाकारहरूलाई कति भुक्तानी गर्ने थिए भन्ने बारेमा अन्तरदृष्टि दिनको लागि हो।'; + + @override + String count_mins(Object minutes) { + return '$minutes मिनेट'; + } + + @override + String get summary_minutes => 'मिनेट'; + + @override + String get summary_listened_to_music => 'सङ्गीत सुन्नु'; + + @override + String get summary_songs => 'गीतहरू'; + + @override + String get summary_streamed_overall => 'सामान्य रूपले स्ट्रीम गरिएको'; + + @override + String get summary_owed_to_artists => 'यस महिना कलाकारहरूलाई देन'; + + @override + String get summary_artists => 'कलाकारको'; + + @override + String get summary_music_reached_you => 'सङ्गीत तपाईंलाई पुग्यो'; + + @override + String get summary_full_albums => 'पूर्ण एल्बमहरू'; + + @override + String get summary_got_your_love => 'तपाईंको माया प्राप्त गरियो'; + + @override + String get summary_playlists => 'प्लेइस्ट'; + + @override + String get summary_were_on_repeat => 'पुनरावृत्ति गरियो'; + + @override + String total_money(Object money) { + return 'कुल $money'; + } + + @override + String get webview_not_found => 'वेबभ्यू फेला परेन'; + + @override + String get webview_not_found_description => + 'तपाईंको उपकरणमा कुनै वेबभ्यू रनटाइम स्थापना गरिएको छैन।\nयदि स्थापना गरिएको छ भने, environment PATH मा छ कि छैन भनेर सुनिश्चित गर्नुहोस्\n\nस्थापना पछि, अनुप्रयोग पुनः सुरु गर्नुहोस्'; + + @override + String get unsupported_platform => 'असमर्थित प्लेटफार्म'; + + @override + String get cache_music => 'सङ्गीत क्यास गर्नुहोस्'; + + @override + String get open => 'खोल्नुहोस्'; + + @override + String get cache_folder => 'क्यास फोल्डर'; + + @override + String get export => 'निर्यात गर्नुहोस्'; + + @override + String get clear_cache => 'क्यास खाली गर्नुहोस्'; + + @override + String get clear_cache_confirmation => 'के तपाई क्यास खाली गर्न चाहनुहुन्छ?'; + + @override + String get export_cache_files => 'क्यास फाइलहरू निर्यात गर्नुहोस्'; + + @override + String found_n_files(Object count) { + return '$count फाइलहरू फेला परे'; + } + + @override + String get export_cache_confirmation => 'यी फाइलहरू निर्यात गर्न चाहनुहुन्छ'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported मध्ये $files फाइलहरू निर्यात गरियो'; + } + + @override + String get undo => 'पूर्ववत'; + + @override + String get download_all => 'सभी डाउनलोड करें'; + + @override + String get add_all_to_playlist => 'सभी को प्लेलिस्ट में जोड़ें'; + + @override + String get add_all_to_queue => 'सभी को कतार में जोड़ें'; + + @override + String get play_all_next => 'सभी को अगला प्ले करें'; + + @override + String get pause => 'विराम'; + + @override + String get view_all => 'सभी देखें'; + + @override + String get no_tracks_added_yet => + 'लगता है आपने अभी तक कोई ट्रैक नहीं जोड़ा है'; + + @override + String get no_tracks => 'यहाँ कोई ट्रैक नहीं दिख रहे हैं'; + + @override + String get no_tracks_listened_yet => + 'आपने अभी तक कुछ नहीं सुना है ऐसा लगता है'; + + @override + String get not_following_artists => 'आप किसी कलाकार को फॉलो नहीं कर रहे हैं'; + + @override + String get no_favorite_albums_yet => + 'लगता है आपने अभी तक कोई एल्बम पसंदीदा में नहीं जोड़ा है'; + + @override + String get no_logs_found => 'कोई लॉग नहीं मिला'; + + @override + String get youtube_engine => 'YouTube इंजन'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine इंस्टॉल नहीं है'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine आपके सिस्टम में इंस्टॉल नहीं है।'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'सुनिश्चित करें कि यह PATH वेरिएबल में उपलब्ध है या\nनीचे $engine एक्जीक्यूटेबल का पूर्ण पथ सेट करें'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/unix जैसे ऑपरेटिंग सिस्टम में, .zshrc/.bashrc/.bash_profile आदि में पथ सेट करना काम नहीं करेगा।\nआपको शेल कॉन्फ़िगरेशन फ़ाइल में पथ सेट करना होगा'; + + @override + String get download => 'डाउनलोड'; + + @override + String get file_not_found => 'फ़ाइल नहीं मिली'; + + @override + String get custom => 'कस्टम'; + + @override + String get add_custom_url => 'कस्टम URL जोड़ें'; + + @override + String get edit_port => 'पोर्ट सम्पादन गर्नुहोस्'; + + @override + String get port_helper_msg => + 'डिफ़ॉल्ट -1 हो जुन यादृच्छिक संख्या जनाउँछ। यदि तपाईंले फायरवाल कन्फिगर गर्नुभएको छ भने, यसलाई सेट गर्न सिफारिस गरिन्छ।'; + + @override + String connect_request(Object client) { + return '$client लाई जडान गर्न अनुमति दिनुहोस्?'; + } + + @override + String get connection_request_denied => + 'जडान अस्वीकृत। प्रयोगकर्ताले पहुँच अस्वीकृत गर्यो।'; + + @override + String get an_error_occurred => 'त्रुटि भयो'; + + @override + String get copy_to_clipboard => 'क्लिपबोर्डमा प्रतिलिपि गर्नुहोस्'; + + @override + String get view_logs => 'लगहरू हेर्नुहोस्'; + + @override + String get retry => 'पुनः प्रयास गर्नुहोस्'; + + @override + String get no_default_metadata_provider_selected => + 'तपाईंले कुनै पूर्वनिर्धारित मेटाडेटा प्रदायक सेट गर्नुभएको छैन'; + + @override + String get manage_metadata_providers => + 'मेटाडेटा प्रदायकहरू प्रबन्ध गर्नुहोस्'; + + @override + String get open_link_in_browser => 'ब्राउजरमा लिङ्क खोल्ने?'; + + @override + String get do_you_want_to_open_the_following_link => + 'के तपाईं निम्न लिङ्क खोल्न चाहनुहुन्छ'; + + @override + String get unsafe_url_warning => + 'अविश्वसनीय स्रोतहरूबाट लिङ्कहरू खोल्नु असुरक्षित हुन सक्छ। सावधान रहनुहोस्!\nतपाईं लिङ्कलाई आफ्नो क्लिपबोर्डमा पनि प्रतिलिपि गर्न सक्नुहुन्छ।'; + + @override + String get copy_link => 'लिङ्क प्रतिलिपि गर्नुहोस्'; + + @override + String get building_your_timeline => + 'तपाईंको सुन्ने आधारमा तपाईंको समयरेखा निर्माण गर्दै...'; + + @override + String get official => 'आधिकारिक'; + + @override + String author_name(Object author) { + return 'लेखक: $author'; + } + + @override + String get third_party => 'तेस्रो-पक्ष'; + + @override + String get plugin_requires_authentication => 'प्लगइनलाई प्रमाणीकरण चाहिन्छ'; + + @override + String get update_available => 'अपडेट उपलब्ध छ'; + + @override + String get supports_scrobbling => 'स्क्रब्बलिंगलाई समर्थन गर्दछ'; + + @override + String get plugin_scrobbling_info => + 'यो प्लगइनले तपाईंको सुन्ने इतिहास उत्पन्न गर्न तपाईंको संगीतलाई स्क्रब्बल गर्दछ।'; + + @override + String get default_metadata_source => 'पूर्वनिर्धारित मेटाडाटा स्रोत'; + + @override + String get set_default_metadata_source => + 'पूर्वनिर्धारित मेटाडाटा स्रोत सेट गर्नुहोस्'; + + @override + String get default_audio_source => 'पूर्वनिर्धारित अडियो स्रोत'; + + @override + String get set_default_audio_source => + 'पूर्वनिर्धारित अडियो स्रोत सेट गर्नुहोस्'; + + @override + String get set_default => 'पूर्वनिर्धारित सेट गर्नुहोस्'; + + @override + String get support => 'समर्थन'; + + @override + String get support_plugin_development => 'प्लगइन विकासलाई समर्थन गर्नुहोस्'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API मा पहुँच गर्न सक्छ'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'के तपाईं यो प्लगइन स्थापना गर्न चाहनुहुन्छ?'; + + @override + String get third_party_plugin_warning => + 'यो प्लगइन तेस्रो-पक्ष रिपोसिटरीबाट हो। कृपया स्थापना गर्नु अघि तपाईंले स्रोतमा विश्वास गर्नुहुन्छ भनी सुनिश्चित गर्नुहोस्।'; + + @override + String get author => 'लेखक'; + + @override + String get this_plugin_can_do_following => 'यो प्लगइनले निम्न गर्न सक्छ'; + + @override + String get install => 'स्थापना गर्नुहोस्'; + + @override + String get install_a_metadata_provider => + 'मेटाडेटा प्रदायक स्थापना गर्नुहोस्'; + + @override + String get no_tracks_playing => 'हाल कुनै ट्र्याक बजिरहेको छैन'; + + @override + String get synced_lyrics_not_available => + 'यो गीतको लागि सिङ्क गरिएका बोलहरू उपलब्ध छैनन्। कृपया यसको सट्टा'; + + @override + String get plain_lyrics => 'सादा बोलहरू'; + + @override + String get tab_instead => 'ट्याब प्रयोग गर्नुहोस्।'; + + @override + String get disclaimer => 'अस्वीकरण'; + + @override + String get third_party_plugin_dmca_notice => + 'स्पोट्यूब टोलीले कुनै पनि \"तेस्रो-पक्ष\" प्लगइनहरूको लागि कुनै जिम्मेवारी (कानुनी सहित) लिँदैन।\nकृपया तिनीहरूलाई आफ्नो जोखिममा प्रयोग गर्नुहोस्। कुनै पनि बग/समस्याहरूको लागि, कृपया तिनीहरूलाई प्लगइन रिपोसिटरीमा रिपोर्ट गर्नुहोस्।\n\nयदि कुनै \"तेस्रो-पक्ष\" प्लगइनले कुनै सेवा/कानुनी संस्थाको ToS/DMCA तोडिरहेको छ भने, कृपया \"तेस्रो-पक्ष\" प्लगइन लेखक वा होस्टिङ प्लेटफर्म e.g. GitHub/Codeberg लाई कारबाही गर्न अनुरोध गर्नुहोस्। माथि सूचीबद्ध (\"तेस्रो-पक्ष\" लेबल गरिएका) सबै सार्वजनिक/सामुदायिक रूपमा राखिएका प्लगइनहरू हुन्। हामी तिनीहरूलाई क्युरेट गरिरहेका छैनौं, त्यसैले हामी तिनीहरूमा कुनै कारबाही गर्न सक्दैनौं।\n\n'; + + @override + String get input_does_not_match_format => 'इनपुट आवश्यक ढाँचासँग मेल खाँदैन'; + + @override + String get plugins => 'प्लगइनहरू'; + + @override + String get paste_plugin_download_url => + 'डाउनलोड url वा GitHub/Codeberg repo url वा .smplug फाइलमा सिधा लिङ्क टाँस्नुहोस्'; + + @override + String get download_and_install_plugin_from_url => + 'url बाट प्लगइन डाउनलोड र स्थापना गर्नुहोस्'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'प्लगइन थप्न असफल: $error'; + } + + @override + String get upload_plugin_from_file => 'फाइलबाट प्लगइन अपलोड गर्नुहोस्'; + + @override + String get installed => 'स्थापित'; + + @override + String get available_plugins => 'उपलब्ध प्लगइनहरू'; + + @override + String get configure_plugins => + 'आफ्नै मेटाडाटा प्रदायक र अडियो स्रोत प्लगइनहरू कन्फिगर गर्नुहोस्'; + + @override + String get audio_scrobblers => 'अडियो स्क्रब्बलरहरू'; + + @override + String get scrobbling => 'स्क्रब्बलिंग'; + + @override + String get source => 'स्रोत: '; + + @override + String get uncompressed => 'असंक्षिप्त'; + + @override + String get dab_music_source_description => + 'अडियोप्रेमीहरूका लागि। उच्च गुणस्तर/लसलेस अडियो स्ट्रिमहरू उपलब्ध गराउँछ। ISRC-मा आधारित सटीक ट्र्याक मिलान।'; +} diff --git a/lib/l10n/generated/app_localizations_nl.dart b/lib/l10n/generated/app_localizations_nl.dart new file mode 100644 index 00000000..0a73c640 --- /dev/null +++ b/lib/l10n/generated/app_localizations_nl.dart @@ -0,0 +1,1570 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Dutch Flemish (`nl`). +class AppLocalizationsNl extends AppLocalizations { + AppLocalizationsNl([String locale = 'nl']) : super(locale); + + @override + String get guest => 'Gast'; + + @override + String get browse => 'Bladeren'; + + @override + String get search => 'Zoeken'; + + @override + String get library => 'Bibliotheek'; + + @override + String get lyrics => 'Teksten'; + + @override + String get settings => 'Instellingen'; + + @override + String get genre_categories_filter => 'Categorieën of genres filteren…'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Gepersonaliseerd'; + + @override + String get featured => 'Aanbevolen'; + + @override + String get new_releases => 'Nieuwe uitgaven'; + + @override + String get songs => 'Liedjes'; + + @override + String playing_track(Object track) { + return '$track afspelen'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Dit zal de huidige wachtrij wissen. $track_length nummers worden verwijderd\nWil je doorgaan?'; + } + + @override + String get load_more => 'Meer laden'; + + @override + String get playlists => 'Afspeellijsten'; + + @override + String get artists => 'Artiesten'; + + @override + String get albums => 'Albums'; + + @override + String get tracks => 'Nummers'; + + @override + String get downloads => 'Downloads'; + + @override + String get filter_playlists => 'Afspeellijsten filteren…'; + + @override + String get liked_tracks => 'Geliefde tracks'; + + @override + String get liked_tracks_description => 'Al je favoriete nummers'; + + @override + String get playlist => 'Afspeellijst'; + + @override + String get create_a_playlist => 'Een afspeellijst aanmaken'; + + @override + String get update_playlist => 'Afspeellijst bijwerken'; + + @override + String get create => 'Aanmaken'; + + @override + String get cancel => 'Annuleren'; + + @override + String get update => 'Bijwerken'; + + @override + String get playlist_name => 'Naam afspeellijst'; + + @override + String get name_of_playlist => 'Naam van de afspeellijst'; + + @override + String get description => 'Beschrijving'; + + @override + String get public => 'Openbaar'; + + @override + String get collaborative => 'Samenwerkend'; + + @override + String get search_local_tracks => 'Lokale nummers zoeken…'; + + @override + String get play => 'Afspelen'; + + @override + String get delete => 'Wissen'; + + @override + String get none => 'Geen'; + + @override + String get sort_a_z => 'Sorteren op A-Z'; + + @override + String get sort_z_a => 'Sorteren op Z-A'; + + @override + String get sort_artist => 'Sorteren op artiest'; + + @override + String get sort_album => 'Sorteren op album'; + + @override + String get sort_duration => 'Sorteren op lengte'; + + @override + String get sort_tracks => 'Nummers sorteren'; + + @override + String currently_downloading(Object tracks_length) { + return 'Momenteel aan het downloaden ($tracks_length)'; + } + + @override + String get cancel_all => 'Alles annuleren'; + + @override + String get filter_artist => 'Artiesten filteren…'; + + @override + String followers(Object followers) { + return '$followers volgers'; + } + + @override + String get add_artist_to_blacklist => 'Artiest toevoegen aan zwarte lijst'; + + @override + String get top_tracks => 'Topnummers'; + + @override + String get fans_also_like => 'Fans luisteren ook'; + + @override + String get loading => 'Laden…'; + + @override + String get artist => 'Artiest'; + + @override + String get blacklisted => 'Zwarte lijst'; + + @override + String get following => 'Volgen'; + + @override + String get follow => 'Volgen'; + + @override + String get artist_url_copied => 'URL artiest gekopieerd naar klembord'; + + @override + String added_to_queue(Object tracks) { + return '$tracks nummers toegevoegd aan wachtrij'; + } + + @override + String get filter_albums => 'Albums filteren…'; + + @override + String get synced => 'Gesynchroniseerd'; + + @override + String get plain => 'Eenvoudig'; + + @override + String get shuffle => 'Willekeurig'; + + @override + String get search_tracks => 'Nummers zoeken…'; + + @override + String get released => 'Uitgegeven'; + + @override + String error(Object error) { + return 'Fout $error'; + } + + @override + String get title => 'Titel'; + + @override + String get time => 'Tijd'; + + @override + String get more_actions => 'Meer acties'; + + @override + String download_count(Object count) { + return '($count) downloads'; + } + + @override + String add_count_to_playlist(Object count) { + return '($count) aan afspeellijst toevoegen'; + } + + @override + String add_count_to_queue(Object count) { + return '($count) aan wachtrij toevoegen'; + } + + @override + String play_count_next(Object count) { + return 'Volgende ($count) afspelen'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return '$data naar klembord gekopieerd'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track aan volgende afspeellijsten toevoegen'; + } + + @override + String get add => 'Toevoegen'; + + @override + String added_track_to_queue(Object track) { + return '$track aan wachtrij toegevoegd'; + } + + @override + String get add_to_queue => 'Toevoegen aan wachtrij'; + + @override + String track_will_play_next(Object track) { + return '$track wordt hierna afgespeeld'; + } + + @override + String get play_next => 'Volgende afspelen'; + + @override + String removed_track_from_queue(Object track) { + return '$track van wachtrij verwijderd'; + } + + @override + String get remove_from_queue => 'Van wachtrij verwijderen'; + + @override + String get remove_from_favorites => 'Van favorieten verwijderen'; + + @override + String get save_as_favorite => 'Opslaan als favoriet'; + + @override + String get add_to_playlist => 'Aan afspeellijst toevoegen'; + + @override + String get remove_from_playlist => 'Van afspeellijst verwijderen'; + + @override + String get add_to_blacklist => 'Aan zwarte lijst toevoegen'; + + @override + String get remove_from_blacklist => 'Van zwarte lijst verwijderen'; + + @override + String get share => 'Delen'; + + @override + String get mini_player => 'Minispeler'; + + @override + String get slide_to_seek => 'Schuiven om vooruit of achteruit te zoeken'; + + @override + String get shuffle_playlist => 'Afspeellijst willekeurig'; + + @override + String get unshuffle_playlist => 'Afspeellijst op volgorde'; + + @override + String get previous_track => 'Vorige nummer'; + + @override + String get next_track => 'Volgende nummer'; + + @override + String get pause_playback => 'Afspelen pauzeren'; + + @override + String get resume_playback => 'Afspelen hervatten'; + + @override + String get loop_track => 'Nummer herhalen'; + + @override + String get no_loop => 'Geen herhaling'; + + @override + String get repeat_playlist => 'Afspeellijst herhalen'; + + @override + String get queue => 'Wachtrij'; + + @override + String get alternative_track_sources => 'Alternatieve bronnen voor nummers'; + + @override + String get download_track => 'Nummer downloaden'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks nummers in wachtrij'; + } + + @override + String get clear_all => 'Alles wissen'; + + @override + String get show_hide_ui_on_hover => 'UI tonen/verbergen bij zweven'; + + @override + String get always_on_top => 'Altijd bovenaan'; + + @override + String get exit_mini_player => 'Minispeler afsluiten'; + + @override + String get download_location => 'Downloadlocatie'; + + @override + String get local_library => 'Lokale bibliotheek'; + + @override + String get add_library_location => 'Toevoegen aan bibliotheek'; + + @override + String get remove_library_location => 'Verwijderen uit bibliotheek'; + + @override + String get account => 'Account'; + + @override + String get logout => 'Afmelden'; + + @override + String get logout_of_this_account => 'Afmelden van dit account'; + + @override + String get language_region => 'Taal & regio'; + + @override + String get language => 'Taal'; + + @override + String get system_default => 'Systeemstandaard'; + + @override + String get market_place_region => 'Marktplaats-regio'; + + @override + String get recommendation_country => 'Aanbeveling Land'; + + @override + String get appearance => 'Uiterlijk'; + + @override + String get layout_mode => 'Opmaakmodus'; + + @override + String get override_layout_settings => + 'Instellingen voor responsieve opmaakmodus opheffen'; + + @override + String get adaptive => 'Adaptief'; + + @override + String get compact => 'Compact'; + + @override + String get extended => 'Uitgebreid'; + + @override + String get theme => 'Thema'; + + @override + String get dark => 'Donker'; + + @override + String get light => 'Licht'; + + @override + String get system => 'Systeem'; + + @override + String get accent_color => 'Accentkleur'; + + @override + String get sync_album_color => 'Albumkleur synchroniseren'; + + @override + String get sync_album_color_description => + 'Gebruikt de overheersende kleur van het album als accentkleur'; + + @override + String get playback => 'Weergave'; + + @override + String get audio_quality => 'Audiokwaliteit'; + + @override + String get high => 'Hoog'; + + @override + String get low => 'Laag'; + + @override + String get pre_download_play => 'Vooraf downloaden en afspelen'; + + @override + String get pre_download_play_description => + 'In plaats van audio te streamen, kun je bytes downloaden en afspelen (aanbevolen voor gebruikers met een hogere bandbreedte)'; + + @override + String get skip_non_music => 'Niet-muzieksegmenten overslaan (SponsorBlock)'; + + @override + String get blacklist_description => 'Nummers en artiesten op de zwarte lijst'; + + @override + String get wait_for_download_to_finish => + 'Wacht tot de huidige download is voltooid'; + + @override + String get desktop => 'Bureaublad'; + + @override + String get close_behavior => 'Sluitgedrag'; + + @override + String get close => 'Afsluiten'; + + @override + String get minimize_to_tray => 'Minimaliseren naar systeemvak'; + + @override + String get show_tray_icon => 'Systeemvakpictogram tonen'; + + @override + String get about => 'Over'; + + @override + String get u_love_spotube => 'We weten dat je van Spotube houd'; + + @override + String get check_for_updates => 'Controleren op updates'; + + @override + String get about_spotube => 'Over Spotube'; + + @override + String get blacklist => 'Zwarte lijst'; + + @override + String get please_sponsor => 'Sponsor/Doneer a.u.b.'; + + @override + String get spotube_description => + 'Spotube, een lichtgewicht, cross-platform, vrij-voor-alles Spotify-client'; + + @override + String get version => 'Versie'; + + @override + String get build_number => 'Bouwnummer'; + + @override + String get founder => 'Grondlegger'; + + @override + String get repository => 'Opslagplaats'; + + @override + String get bug_issues => 'Bug+problemen'; + + @override + String get made_with => 'Met ❤️ gemaakt in Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licentie'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Maak je geen zorgen, je gegevens worden niet verzameld of gedeeld met anderen.'; + + @override + String get know_how_to_login => 'Weet je niet hoe je dit moet doen?'; + + @override + String get follow_step_by_step_guide => 'Volg de stapsgewijze handleiding'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => 'Vul alle velden in a.u.b.'; + + @override + String get submit => 'Verzenden'; + + @override + String get exit => 'Afronden'; + + @override + String get previous => 'Vorige'; + + @override + String get next => 'Volgende'; + + @override + String get done => 'Klaar'; + + @override + String get step_1 => 'Stap 1'; + + @override + String get first_go_to => 'Ga eerst naar'; + + @override + String get something_went_wrong => 'Er ging iets mis'; + + @override + String get piped_instance => 'Piped-serverinstantie'; + + @override + String get piped_description => + 'De Piped-serverinstantie die moet worden gebruikt voor overeenkomstige nummers'; + + @override + String get piped_warning => + 'Sommige werken misschien niet goed. Dus gebruik ze op eigen risico'; + + @override + String get invidious_instance => 'Invidious-serverinstantie'; + + @override + String get invidious_description => + 'De Invidious-serverinstantie die gebruikt wordt voor trackmatching'; + + @override + String get invidious_warning => + 'Sommigen werken mogelijk niet goed. Gebruik op eigen risico'; + + @override + String get generate => 'Genereren'; + + @override + String track_exists(Object track) { + return 'Nummer $track bestaat al'; + } + + @override + String get replace_downloaded_tracks => 'Alle gedownloade nummers vervangen'; + + @override + String get skip_download_tracks => + 'Downloaden van alle gedownloade nummers overslaan'; + + @override + String get do_you_want_to_replace => 'Wil je het bestaande nummer vervangen?'; + + @override + String get replace => 'Vervangen'; + + @override + String get skip => 'Overslaan'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Selecteer tot $count $type'; + } + + @override + String get select_genres => 'Genres selecteren'; + + @override + String get add_genres => 'Genres toevoegen'; + + @override + String get country => 'Land'; + + @override + String get number_of_tracks_generate => 'Aantal nummers om te genereren'; + + @override + String get acousticness => 'Akoestiek'; + + @override + String get danceability => 'Dansbaarheid'; + + @override + String get energy => 'Energie'; + + @override + String get instrumentalness => 'Instrumentaliteit'; + + @override + String get liveness => 'Levendigheid'; + + @override + String get loudness => 'Luidheid'; + + @override + String get speechiness => 'Spraak'; + + @override + String get valence => 'Valentie'; + + @override + String get popularity => 'Populariteit'; + + @override + String get key => 'Sleutel'; + + @override + String get duration => 'Tijdsduur (s)'; + + @override + String get tempo => 'Tempo (SPM)'; + + @override + String get mode => 'Modus'; + + @override + String get time_signature => 'Tijdsnotatie'; + + @override + String get short => 'Kort'; + + @override + String get medium => 'Middel'; + + @override + String get long => 'Lang'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Doel'; + + @override + String get moderate => 'Matig'; + + @override + String get deselect_all => 'Selectie opheffen'; + + @override + String get select_all => 'Alles selecteren'; + + @override + String get are_you_sure => 'Weet je het zeker?'; + + @override + String get generating_playlist => 'Aangepaste afspeellijst genereren…'; + + @override + String selected_count_tracks(Object count) { + return '$count nummers geselecteerd'; + } + + @override + String get download_warning => + 'Als je alle nummers in bulk downloadt, ben je duidelijk bezig met muziekpiraterij en breng je schade toe aan de creatieve muziekmaatschappij. Ik hoop dat je je hiervan bewust bent. Probeer altijd het harde werk van artiesten te respecteren en te steunen.'; + + @override + String get download_ip_ban_warning => + 'BTW, je IP-adres kan worden geblokkeerd op YouTube als gevolg van buitensporige downloadverzoeken. IP-blokkering betekent dat je YouTube niet kunt gebruiken (zelfs als je ingelogd bent) voor tenminste 2-3 maanden vanaf dat IP-apparaat. Spotube is niet verantwoordelijk als dit ooit gebeurt.'; + + @override + String get by_clicking_accept_terms => + 'Door op \'accepteren\' te klikken ga je akkoord met de volgende voorwaarden:'; + + @override + String get download_agreement_1 => + 'Ik weet dat ik muziek illegaal donload. Ik ben slecht.'; + + @override + String get download_agreement_2 => + 'Ik steun de artiest waar ik kan en ik doe dit alleen omdat ik geen geld heb om hun kunst te kopen.'; + + @override + String get download_agreement_3 => + 'Ik ben me er volledig van bewust dat mijn IP geblokkeerd kan worden op YouTube & ik houd Spotube of zijn eigenaars/contributeurs niet verantwoordelijk voor ongelukken die veroorzaakt worden door mijn huidige actie.'; + + @override + String get decline => 'Weigeren'; + + @override + String get accept => 'Accepteren'; + + @override + String get details => 'Bijzonderheden'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanaal'; + + @override + String get likes => 'Liefs'; + + @override + String get dislikes => 'Hekels'; + + @override + String get views => 'Weergaven'; + + @override + String get streamUrl => 'Stream-URL'; + + @override + String get stop => 'Stoppen'; + + @override + String get sort_newest => 'Sorteren op recent toegevoegd'; + + @override + String get sort_oldest => 'Sorteren op langst toegevoegd'; + + @override + String get sleep_timer => 'Slaaptimer'; + + @override + String mins(Object minutes) { + return '$minutes minuten'; + } + + @override + String hours(Object hours) { + return '$hours uren'; + } + + @override + String hour(Object hours) { + return '$hours uur'; + } + + @override + String get custom_hours => 'Aangepaste uren'; + + @override + String get logs => 'Logboeken'; + + @override + String get developers => 'Ontwikkelaars'; + + @override + String get not_logged_in => 'Je bent niet aangemeld'; + + @override + String get search_mode => 'Zoekmodus'; + + @override + String get audio_source => 'Audiobron'; + + @override + String get ok => 'Oké'; + + @override + String get failed_to_encrypt => 'Versleuteling mislukt'; + + @override + String get encryption_failed_warning => + 'Spotube gebruikt versleuteling om je gegevens veilig op te slaan. Maar dat is niet gelukt. Dus zal het terugvallen op onveilige opslag.\nAls je linux gebruikt, zorg er dan voor dat je een geheim-dienst (gnome-keyring, kde-wallet, keepassxc etc) hebt geïnstalleerd.'; + + @override + String get querying_info => 'Info opvragen…'; + + @override + String get piped_api_down => 'Piped API is uit'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'De Piped-instantie $pipedInstance is momenteel uitgevallen\n\nVerander de instantie of verander het \'API-type\' naar de officiële YouTube API.\n\nZorg ervoor dat u de app herstart na de wijziging'; + } + + @override + String get you_are_offline => 'Je bent momenteel offline'; + + @override + String get connection_restored => 'Je internetverbinding is hersteld'; + + @override + String get use_system_title_bar => 'Systeemtitelbalk gebruiken'; + + @override + String get crunching_results => 'Resultaten verwerken…'; + + @override + String get search_to_get_results => 'Zoeken naar resultaten'; + + @override + String get use_amoled_mode => 'Pikzwart donkerthema'; + + @override + String get pitch_dark_theme => 'AMOLED-modus'; + + @override + String get normalize_audio => 'Audio normaliseren'; + + @override + String get change_cover => 'Hoes aanpassen'; + + @override + String get add_cover => 'Hoes toevoegen'; + + @override + String get restore_defaults => 'Standaardwaarden herstellen'; + + @override + String get download_music_format => 'Download muziekformaat'; + + @override + String get streaming_music_format => 'Streaming muziekformaat'; + + @override + String get download_music_quality => 'Downloadkwaliteit'; + + @override + String get streaming_music_quality => 'Streamingkwaliteit'; + + @override + String get login_with_lastfm => 'Inloggen met Last.fm'; + + @override + String get connect => 'Verbinden'; + + @override + String get disconnect_lastfm => 'Last.fm verbreken'; + + @override + String get disconnect => 'Verbeken'; + + @override + String get username => 'Gebruikersnaam'; + + @override + String get password => 'Wachtwoord'; + + @override + String get login => 'Inloggen'; + + @override + String get login_with_your_lastfm => 'Inloggen met je Last.fm account'; + + @override + String get scrobble_to_lastfm => 'Scrobbelen naar Last.fm'; + + @override + String get go_to_album => 'Ga naar album'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'Alles doorbladeren'; + + @override + String get genres => 'Genres'; + + @override + String get explore_genres => 'Genres verkennen'; + + @override + String get friends => 'Vrienden'; + + @override + String get no_lyrics_available => + 'Sorry, geen teksten gevonden voor dit nummer'; + + @override + String get start_a_radio => 'Een radio starten'; + + @override + String get how_to_start_radio => 'Hoe wil je de radio starten?'; + + @override + String get replace_queue_question => + 'Wil je de huidige wachtrij vervangen of eraan toevoegen?'; + + @override + String get endless_playback => 'Oneindig afspelen'; + + @override + String get delete_playlist => 'Afspeellijst verwijderen'; + + @override + String get delete_playlist_confirmation => + 'Weet je zeker dat je deze afspeellijst wilt verwijderen?'; + + @override + String get local_tracks => 'Lokale nummers'; + + @override + String get local_tab => 'Lokaal'; + + @override + String get song_link => 'Song-link'; + + @override + String get skip_this_nonsense => 'Deze onzin overslaan'; + + @override + String get freedom_of_music => '“Vrijheid van muziek”'; + + @override + String get freedom_of_music_palm => '“Vrijheid van muziek in je hand”'; + + @override + String get get_started => 'Laten we beginnen'; + + @override + String get youtube_source_description => 'Aangeraden en werkt het best.'; + + @override + String get piped_source_description => + 'Voel je je vrij? Net als YouTube, maar meer vrij.'; + + @override + String get jiosaavn_source_description => + 'Het beste voor de regio Zuid-Azië.'; + + @override + String get invidious_source_description => + 'Vergelijkbaar met Piped, maar met een hogere beschikbaarheid.'; + + @override + String highest_quality(Object quality) { + return 'Hoogste kwaliteit: $quality'; + } + + @override + String get select_audio_source => 'Audiobron kiezen'; + + @override + String get endless_playback_description => + 'Nieuwe nummers automatisch achteraan de wachtrij toevoegen'; + + @override + String get choose_your_region => 'Kies je regio'; + + @override + String get choose_your_region_description => + 'Dit helpt Spotube om de juiste inhoud\nvoor jouw locatie te tonen.'; + + @override + String get choose_your_language => 'Kies je taal'; + + @override + String get help_project_grow => 'Help dit project met groeien'; + + @override + String get help_project_grow_description => + 'Spotube is een open-source project. Je kunt dit project helpen groeien door eraan bij te dragen, problemen te melden of nieuwe functies voor te stellen.'; + + @override + String get contribute_on_github => 'Bijdragen on GitHub'; + + @override + String get donate_on_open_collective => 'Doneren on Open Collective'; + + @override + String get browse_anonymously => 'Anoniem browsen'; + + @override + String get enable_connect => 'Verbinding inschakelen'; + + @override + String get enable_connect_description => + 'Spotube bedienen vanaf andere apparaten'; + + @override + String get devices => 'Apparaten'; + + @override + String get select => 'Selecteren'; + + @override + String connect_client_alert(Object client) { + return 'Je wordt gecontroleerd door $client'; + } + + @override + String get this_device => 'Dit apparaat'; + + @override + String get remote => 'Afstandsbediening'; + + @override + String get stats => 'Statistieken'; + + @override + String and_n_more(Object count) { + return 'en $count meer'; + } + + @override + String get recently_played => 'Onlangs afgespeeld'; + + @override + String get browse_more => 'Meer bekijken'; + + @override + String get no_title => 'Geen titel'; + + @override + String get not_playing => 'Niet aan het afspelen'; + + @override + String get epic_failure => 'Epische mislukking!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length nummers aan de wachtrij toegevoegd'; + } + + @override + String get spotube_has_an_update => 'Spotube heeft een update'; + + @override + String get download_now => 'Nu downloaden'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum is uitgebracht'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version is uitgebracht'; + } + + @override + String get read_the_latest => 'Lees de nieuwste '; + + @override + String get release_notes => 'release-opmerkingen'; + + @override + String get pick_color_scheme => 'Kies kleurenschema'; + + @override + String get save => 'Opslaan'; + + @override + String get choose_the_device => 'Kies het apparaat:'; + + @override + String get multiple_device_connected => + 'Er zijn meerdere apparaten verbonden.\nKies het apparaat waarop je deze actie wilt uitvoeren'; + + @override + String get nothing_found => 'Niets gevonden'; + + @override + String get the_box_is_empty => 'De doos is leeg'; + + @override + String get top_artists => 'Topartiesten'; + + @override + String get top_albums => 'Topalbums'; + + @override + String get this_week => 'Deze week'; + + @override + String get this_month => 'Deze maand'; + + @override + String get last_6_months => 'Laatste 6 maanden'; + + @override + String get this_year => 'Dit jaar'; + + @override + String get last_2_years => 'Laatste 2 jaar'; + + @override + String get all_time => 'All time'; + + @override + String powered_by_provider(Object providerName) { + return 'Aangedreven door $providerName'; + } + + @override + String get email => 'E-mail'; + + @override + String get profile_followers => 'Volgers'; + + @override + String get birthday => 'Verjaardag'; + + @override + String get subscription => 'Abonnement'; + + @override + String get not_born => 'Niet geboren'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profiel'; + + @override + String get no_name => 'Geen naam'; + + @override + String get edit => 'Bewerken'; + + @override + String get user_profile => 'Gebruikersprofiel'; + + @override + String count_plays(Object count) { + return '$count afspeelbeurten'; + } + + @override + String get streaming_fees_hypothetical => + '*Dit is berekend op basis van Spotify\'s uitbetaling per stream\nvan \$0.003 tot \$0.005. Dit is een hypothetische\nberekening om gebruikers inzicht te geven in hoeveel ze\naan de artiesten zouden hebben betaald als ze hun lied op Spotify zouden hebben beluisterd.'; + + @override + String get minutes_listened => 'Luistertijd'; + + @override + String get streamed_songs => 'Gestreamde nummers'; + + @override + String count_streams(Object count) { + return '$count streams'; + } + + @override + String get owned_by_you => 'Bezit door jou'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl gekopieerd naar klembord'; + } + + @override + String get hipotetical_calculation => + '*Dit is berekend op basis van de gemiddelde uitbetaling per stream van online muziekstreamingplatforms van \$0,003 tot \$0,005. Dit is een hypothetische berekening om de gebruiker inzicht te geven in hoeveel ze aan de artiesten zouden hebben betaald als ze hun nummer op een ander muziekstreamingplatform zouden beluisteren.'; + + @override + String count_mins(Object minutes) { + return '$minutes min'; + } + + @override + String get summary_minutes => 'minuten'; + + @override + String get summary_listened_to_music => 'Beluisterde muziek'; + + @override + String get summary_songs => 'nummers'; + + @override + String get summary_streamed_overall => 'Totaal gestreamd'; + + @override + String get summary_owed_to_artists => 'Te betalen aan artiesten\ndeze maand'; + + @override + String get summary_artists => 'van de artiest'; + + @override + String get summary_music_reached_you => 'Muziek heeft je bereikt'; + + @override + String get summary_full_albums => 'volledige albums'; + + @override + String get summary_got_your_love => 'Kreeg je liefde'; + + @override + String get summary_playlists => 'afspeellijsten'; + + @override + String get summary_were_on_repeat => 'Was op herhaling'; + + @override + String total_money(Object money) { + return 'Totaal $money'; + } + + @override + String get webview_not_found => 'Webview niet gevonden'; + + @override + String get webview_not_found_description => + 'Er is geen Webview-runtime geïnstalleerd op uw apparaat.\nAls het is geïnstalleerd, zorg ervoor dat het in het environment PATH staat\n\nHerstart de app na installatie'; + + @override + String get unsupported_platform => 'Niet ondersteund platform'; + + @override + String get cache_music => 'Cache muziek'; + + @override + String get open => 'Open'; + + @override + String get cache_folder => 'Cachemap'; + + @override + String get export => 'Exporteren'; + + @override + String get clear_cache => 'Cache wissen'; + + @override + String get clear_cache_confirmation => 'Wilt u de cache wissen?'; + + @override + String get export_cache_files => 'Gecacheerde bestanden exporteren'; + + @override + String found_n_files(Object count) { + return '$count bestanden gevonden'; + } + + @override + String get export_cache_confirmation => + 'Wilt u deze bestanden exporteren naar'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported van de $files bestanden geëxporteerd'; + } + + @override + String get undo => 'Ongedaan maken'; + + @override + String get download_all => 'Alles downloaden'; + + @override + String get add_all_to_playlist => 'Voeg alles toe aan afspeellijst'; + + @override + String get add_all_to_queue => 'Voeg alles toe aan wachtrij'; + + @override + String get play_all_next => 'Speel alles volgende'; + + @override + String get pause => 'Pauzeren'; + + @override + String get view_all => 'Bekijk alles'; + + @override + String get no_tracks_added_yet => + 'Het lijkt erop dat je nog geen nummers hebt toegevoegd'; + + @override + String get no_tracks => 'Het lijkt erop dat er hier geen nummers zijn'; + + @override + String get no_tracks_listened_yet => + 'Het lijkt erop dat je nog niets hebt beluisterd'; + + @override + String get not_following_artists => 'Je volgt geen artiesten'; + + @override + String get no_favorite_albums_yet => + 'Het lijkt erop dat je nog geen albums aan je favorieten hebt toegevoegd'; + + @override + String get no_logs_found => 'Geen logbestanden gevonden'; + + @override + String get youtube_engine => 'YouTube Engine'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine is niet geïnstalleerd'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine is niet geïnstalleerd op je systeem.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Zorg ervoor dat het beschikbaar is in de PATH-variabele of\nstel het absolute pad naar de $engine uitvoerbare bestanden in'; + } + + @override + String get youtube_engine_unix_issue_message => + 'Op macOS/Linux/unix-achtige besturingssystemen werkt het instellen van paden in .zshrc/.bashrc/.bash_profile enz. niet.\nJe moet het pad instellen in het shell-configuratiebestand'; + + @override + String get download => 'Downloaden'; + + @override + String get file_not_found => 'Bestand niet gevonden'; + + @override + String get custom => 'Aangepast'; + + @override + String get add_custom_url => 'Voeg aangepaste URL toe'; + + @override + String get edit_port => 'Poort bewerken'; + + @override + String get port_helper_msg => + 'Standaard is -1, wat een willekeurig nummer aangeeft. Als je een firewall hebt geconfigureerd, wordt aanbevolen dit in te stellen.'; + + @override + String connect_request(Object client) { + return 'Toestaan dat $client verbinding maakt?'; + } + + @override + String get connection_request_denied => + 'Verbinding geweigerd. Gebruiker heeft toegang geweigerd.'; + + @override + String get an_error_occurred => 'Er is een fout opgetreden'; + + @override + String get copy_to_clipboard => 'Kopiëren naar klembord'; + + @override + String get view_logs => 'Logboeken bekijken'; + + @override + String get retry => 'Opnieuw proberen'; + + @override + String get no_default_metadata_provider_selected => + 'U heeft geen standaard metadata-aanbieder ingesteld'; + + @override + String get manage_metadata_providers => 'Metadata-aanbieders beheren'; + + @override + String get open_link_in_browser => 'Link openen in browser?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Wilt u de volgende link openen'; + + @override + String get unsafe_url_warning => + 'Het kan onveilig zijn om links van onbetrouwbare bronnen te openen. Wees voorzichtig!\nU kunt de link ook naar uw klembord kopiëren.'; + + @override + String get copy_link => 'Link kopiëren'; + + @override + String get building_your_timeline => + 'Uw tijdlijn wordt opgebouwd op basis van uw luistergedrag...'; + + @override + String get official => 'Officieel'; + + @override + String author_name(Object author) { + return 'Auteur: $author'; + } + + @override + String get third_party => 'Derden'; + + @override + String get plugin_requires_authentication => 'Plugin vereist authenticatie'; + + @override + String get update_available => 'Update beschikbaar'; + + @override + String get supports_scrobbling => 'Ondersteunt scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Deze plugin scrobblet uw muziek om uw luistergeschiedenis te genereren.'; + + @override + String get default_metadata_source => 'Standaard metadata-bron'; + + @override + String get set_default_metadata_source => 'Standaard metadata-bron instellen'; + + @override + String get default_audio_source => 'Standaard audiobron'; + + @override + String get set_default_audio_source => 'Standaard audiobron instellen'; + + @override + String get set_default => 'Instellen als standaard'; + + @override + String get support => 'Ondersteuning'; + + @override + String get support_plugin_development => 'Ondersteun plugin-ontwikkeling'; + + @override + String can_access_name_api(Object name) { + return '- Kan de **$name** API benaderen'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Wilt u deze plugin installeren?'; + + @override + String get third_party_plugin_warning => + 'Deze plugin is afkomstig van een repository van derden. Zorg ervoor dat u de bron vertrouwt voordat u installeert.'; + + @override + String get author => 'Auteur'; + + @override + String get this_plugin_can_do_following => + 'Deze plugin kan het volgende doen'; + + @override + String get install => 'Installeren'; + + @override + String get install_a_metadata_provider => + 'Een metadata-aanbieder installeren'; + + @override + String get no_tracks_playing => 'Er wordt momenteel geen nummer afgespeeld'; + + @override + String get synced_lyrics_not_available => + 'Gesynchroniseerde songteksten zijn niet beschikbaar voor dit nummer. Gebruik in plaats daarvan het tabblad'; + + @override + String get plain_lyrics => 'Eenvoudige songteksten'; + + @override + String get tab_instead => 'in plaats daarvan.'; + + @override + String get disclaimer => 'Disclaimer'; + + @override + String get third_party_plugin_dmca_notice => + 'Het Spotube-team draagt geen enkele verantwoordelijkheid (inclusief juridische) voor \"derden\" plugins.\nGebruik ze op eigen risico. Voor bugs/problemen kunt u deze melden bij de plugin-repository.\n\nAls een \"derden\" plugin de ToS/DMCA van een service/juridische entiteit schendt, vraag dan de auteur van de \"derden\" plugin of het hostingplatform, bijvoorbeeld GitHub/Codeberg, om actie te ondernemen. De hierboven vermelde (gelabelde \"derden\") plugins zijn allemaal openbare/door de gemeenschap onderhouden plugins. We beheren ze niet, dus we kunnen geen actie tegen ze ondernemen.\n\n'; + + @override + String get input_does_not_match_format => + 'Invoer komt niet overeen met het vereiste formaat'; + + @override + String get plugins => 'Plug-ins'; + + @override + String get paste_plugin_download_url => + 'Plak de download-URL of de URL van de GitHub/Codeberg-repository of een directe link naar het .smplug-bestand'; + + @override + String get download_and_install_plugin_from_url => + 'Download en installeer de plugin via URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Kon de plugin niet toevoegen: $error'; + } + + @override + String get upload_plugin_from_file => 'Plugin uploaden vanuit bestand'; + + @override + String get installed => 'Geïnstalleerd'; + + @override + String get available_plugins => 'Beschikbare plugins'; + + @override + String get configure_plugins => + 'Configureer je eigen metadata- en audiobron-plug-ins'; + + @override + String get audio_scrobblers => 'Audioscrobblers'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Bron: '; + + @override + String get uncompressed => 'Ongecomprimeerd'; + + @override + String get dab_music_source_description => + 'Voor audiofielen. Biedt hoge kwaliteit/lossless audiostreams. Nauwkeurige trackmatching op basis van ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_pl.dart b/lib/l10n/generated/app_localizations_pl.dart new file mode 100644 index 00000000..5e185035 --- /dev/null +++ b/lib/l10n/generated/app_localizations_pl.dart @@ -0,0 +1,1572 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Polish (`pl`). +class AppLocalizationsPl extends AppLocalizations { + AppLocalizationsPl([String locale = 'pl']) : super(locale); + + @override + String get guest => 'Gość'; + + @override + String get browse => 'Przeglądaj'; + + @override + String get search => 'Szukaj'; + + @override + String get library => 'Biblioteka'; + + @override + String get lyrics => 'Tekst utworu'; + + @override + String get settings => 'Ustawienia'; + + @override + String get genre_categories_filter => 'Filtruj kategorie lub gatunki...'; + + @override + String get genre => 'Gatunki'; + + @override + String get personalized => 'Spersonalizowane'; + + @override + String get featured => 'Wyróżnione'; + + @override + String get new_releases => 'Nowo wydane'; + + @override + String get songs => 'Utwory'; + + @override + String playing_track(Object track) { + return 'Odtwarzanie $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'To spowoduje wyczyszczenie całej kolejki! $track_length pozycji zostanie usuniętych.\nCzy chcesz kontynuować?'; + } + + @override + String get load_more => 'Załaduj więcej'; + + @override + String get playlists => 'Playlisty'; + + @override + String get artists => 'Artyści'; + + @override + String get albums => 'Albumy'; + + @override + String get tracks => 'Utwory'; + + @override + String get downloads => 'Pobrane'; + + @override + String get filter_playlists => 'Filtruj swoje playlisty...'; + + @override + String get liked_tracks => 'Ulubione utwory'; + + @override + String get liked_tracks_description => 'Wszystkie twoje ulubione utwory'; + + @override + String get playlist => 'Playlista'; + + @override + String get create_a_playlist => 'Utwórz playlistę'; + + @override + String get update_playlist => 'Zaktualizuj playlistę'; + + @override + String get create => 'Utwórz'; + + @override + String get cancel => 'Anuluj'; + + @override + String get update => 'Aktualizuj'; + + @override + String get playlist_name => 'Nazwa playlisty'; + + @override + String get name_of_playlist => 'Nazwa playlisty'; + + @override + String get description => 'Opis'; + + @override + String get public => 'Publiczny'; + + @override + String get collaborative => 'Współpraca'; + + @override + String get search_local_tracks => 'Szukanie lokalnych utworów...'; + + @override + String get play => 'Odtwórz'; + + @override + String get delete => 'Usuń'; + + @override + String get none => 'Brak'; + + @override + String get sort_a_z => 'Sortuj od A do Z'; + + @override + String get sort_z_a => 'Sortuj od Z do A'; + + @override + String get sort_artist => 'Sortuj po Artyście'; + + @override + String get sort_album => 'Sortuj po Albumie'; + + @override + String get sort_duration => 'Sortuj według Czasu Trwania'; + + @override + String get sort_tracks => 'Sortuj Utwory'; + + @override + String currently_downloading(Object tracks_length) { + return 'Obecnie pobieram $tracks_length utworów.'; + } + + @override + String get cancel_all => 'Anuluj wszystkie'; + + @override + String get filter_artist => 'Filtruj artystów...'; + + @override + String followers(Object followers) { + return '$followers obserwujących'; + } + + @override + String get add_artist_to_blacklist => 'Dodaj artystę do czarnej listy'; + + @override + String get top_tracks => 'Popularne Utwory'; + + @override + String get fans_also_like => 'Fani lubią także'; + + @override + String get loading => 'Ładowanie...'; + + @override + String get artist => 'Artysta'; + + @override + String get blacklisted => 'Dodano do czarnej listy'; + + @override + String get following => 'Obserwujesz'; + + @override + String get follow => 'Zaobserwuj'; + + @override + String get artist_url_copied => 'Skopiowano URL artysty do schowka'; + + @override + String added_to_queue(Object tracks) { + return 'Dodano $tracks utworów do kolejki'; + } + + @override + String get filter_albums => 'Filtruj albumy...'; + + @override + String get synced => 'Zsynchronizowano'; + + @override + String get plain => 'Zwykły'; + + @override + String get shuffle => 'Losowe odtwarzanie'; + + @override + String get search_tracks => 'Szukam utworu...'; + + @override + String get released => 'Wydano'; + + @override + String error(Object error) { + return 'Błąd $error'; + } + + @override + String get title => 'Tytuł'; + + @override + String get time => 'Czas'; + + @override + String get more_actions => 'Więcej akcji'; + + @override + String download_count(Object count) { + return 'Pobrane ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Dodaj ($count) do Playlisty'; + } + + @override + String add_count_to_queue(Object count) { + return 'Dodaj ($count) do Kolejki'; + } + + @override + String play_count_next(Object count) { + return 'Odtwórz ($count) następne'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return 'Skopiowano $data do schowka'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Dodano $track do danych Playlist'; + } + + @override + String get add => 'Dodaj'; + + @override + String added_track_to_queue(Object track) { + return 'Dodano $track do kolejki'; + } + + @override + String get add_to_queue => 'Dodano do kolejki'; + + @override + String track_will_play_next(Object track) { + return '$track następny'; + } + + @override + String get play_next => 'Odtwórz następny'; + + @override + String removed_track_from_queue(Object track) { + return 'Usunięto $track z kolejki'; + } + + @override + String get remove_from_queue => 'Usunięto z kolejki'; + + @override + String get remove_from_favorites => 'Usunięto z ulubionych'; + + @override + String get save_as_favorite => 'Zapisz do ulubionych'; + + @override + String get add_to_playlist => 'Dodaj do playlisty'; + + @override + String get remove_from_playlist => 'Usuń z playlisty'; + + @override + String get add_to_blacklist => 'Dodaj do czarnej listy'; + + @override + String get remove_from_blacklist => 'Usuń z czarnej listy'; + + @override + String get share => 'Udostępnij'; + + @override + String get mini_player => 'Mały odwarzacz'; + + @override + String get slide_to_seek => 'Przesuń, aby przewinąć do przodu lub do tyłu.'; + + @override + String get shuffle_playlist => 'Odtwarzaj losowo z playlisty'; + + @override + String get unshuffle_playlist => 'Nie odtwarzaj losowo z playlisty'; + + @override + String get previous_track => 'Poprzedni utwór'; + + @override + String get next_track => 'Następny utwór'; + + @override + String get pause_playback => 'Zatrzymaj odwarzanie'; + + @override + String get resume_playback => 'Wznów odwarzanie'; + + @override + String get loop_track => 'Zapętl utwór'; + + @override + String get no_loop => 'Brak pętli'; + + @override + String get repeat_playlist => 'Powtarzaj playlistę'; + + @override + String get queue => 'Kolejka'; + + @override + String get alternative_track_sources => 'Alternatywne źródła utworów'; + + @override + String get download_track => 'Pobierz utwór'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks utworów w kolejce'; + } + + @override + String get clear_all => 'Wyczyść wszystko'; + + @override + String get show_hide_ui_on_hover => 'Pokaż/Ukryj unoszący się interfejs'; + + @override + String get always_on_top => 'Zawsze na wierzchu'; + + @override + String get exit_mini_player => 'Opuść Mały odtwarzacz'; + + @override + String get download_location => 'Zmień lokalizację'; + + @override + String get local_library => 'Biblioteka lokalna'; + + @override + String get add_library_location => 'Dodaj do biblioteki'; + + @override + String get remove_library_location => 'Usuń z biblioteki'; + + @override + String get account => 'Konto'; + + @override + String get logout => 'Wyloguj'; + + @override + String get logout_of_this_account => 'Wyloguj z tego konta'; + + @override + String get language_region => 'Język i Region'; + + @override + String get language => 'Język'; + + @override + String get system_default => 'Domyślny systemowy'; + + @override + String get market_place_region => 'Region Rynku'; + + @override + String get recommendation_country => 'Kraj rekomendacji'; + + @override + String get appearance => 'Wygląd'; + + @override + String get layout_mode => 'Tryb Układu'; + + @override + String get override_layout_settings => + 'Nadpisz responsywne ustawienia trybu układu'; + + @override + String get adaptive => 'Adaptacyjny'; + + @override + String get compact => 'Kompaktowy'; + + @override + String get extended => 'Rozszerzony'; + + @override + String get theme => 'Motyw'; + + @override + String get dark => 'Ciemny'; + + @override + String get light => 'Jasny'; + + @override + String get system => 'Systemowy'; + + @override + String get accent_color => 'Kolor Akcentu'; + + @override + String get sync_album_color => 'Synchronizuj kolor albumu'; + + @override + String get sync_album_color_description => + 'Używa dominującego koloru okładki albumu jako koloru akcentującego'; + + @override + String get playback => 'Odtwarzanie'; + + @override + String get audio_quality => 'Jakość dźwięku'; + + @override + String get high => 'Duża'; + + @override + String get low => 'Mała'; + + @override + String get pre_download_play => 'Wstępnie pobierz i odtwórz'; + + @override + String get pre_download_play_description => + 'Zamiast przesyłać strumieniowo dźwięk, pobiera odpowiedni bufor i odtwarza (zalecane dla użytkowników o większej przepustowości)'; + + @override + String get skip_non_music => 'Pomiń nie-muzyczne segmenty (SponsorBlock)'; + + @override + String get blacklist_description => 'Czarna lista utworów i artystów'; + + @override + String get wait_for_download_to_finish => + 'Proszę poczekać na zakończenie obecnego pobierania.'; + + @override + String get desktop => 'Pulpit'; + + @override + String get close_behavior => 'Zamknij'; + + @override + String get close => 'Zamknij'; + + @override + String get minimize_to_tray => 'Zminimalizuj do zasobnika'; + + @override + String get show_tray_icon => 'Pokazuj ikonę w zasobniku'; + + @override + String get about => 'O projekcie'; + + @override + String get u_love_spotube => 'Wiemy jak kochacie Spotube'; + + @override + String get check_for_updates => 'Sprawdź aktualizacje'; + + @override + String get about_spotube => 'O Spotube'; + + @override + String get blacklist => 'Czarna lista'; + + @override + String get please_sponsor => 'Proszę wesprzyj projekt'; + + @override + String get spotube_description => + 'Spotube, lekki, wieloplatformowy, darmowy dla wszystkich klient Spotify'; + + @override + String get version => 'Wersja'; + + @override + String get build_number => 'Numer Build\'a'; + + @override + String get founder => 'Twórca Założyciel'; + + @override + String get repository => 'Repozytorium'; + + @override + String get bug_issues => 'Błędy i propozycje'; + + @override + String get made_with => 'Stworzono z ❤️ w Bangladesh\'u 🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licencja'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Nie martw się, żadne dane logowania nie są zbierane ani udostępniane nikomu'; + + @override + String get know_how_to_login => 'Nie wiesz, jak się zalogować?'; + + @override + String get follow_step_by_step_guide => + 'Postępuj zgodnie z poradnikiem krok po kroku'; + + @override + String cookie_name_cookie(Object name) { + return '$name Ciasteczko'; + } + + @override + String get fill_in_all_fields => 'Proszę wypełnić wszystkie pola'; + + @override + String get submit => 'Zatwierdź'; + + @override + String get exit => 'Zamknij'; + + @override + String get previous => 'Poprzedni'; + + @override + String get next => 'Następny'; + + @override + String get done => 'Gotowe 🙂'; + + @override + String get step_1 => 'Krok 1'; + + @override + String get first_go_to => 'Po pierwsze przejdź do'; + + @override + String get something_went_wrong => 'Coś poszło nie tak 🙁'; + + @override + String get piped_instance => 'Instancja serwera Piped'; + + @override + String get piped_description => + 'Instancja serwera Piped używana jest do dopasowania utworów.'; + + @override + String get piped_warning => + 'Niektóre z nich mogą nie działać. Używasz na własną odpowiedzialność!'; + + @override + String get invidious_instance => 'Instancja serwera Invidious'; + + @override + String get invidious_description => + 'Instancja serwera Invidious do dopasowywania utworów'; + + @override + String get invidious_warning => + 'Niektóre z nich mogą nie działać dobrze. Używaj na własne ryzyko'; + + @override + String get generate => 'Generuj'; + + @override + String track_exists(Object track) { + return 'Utwór $track już istnieje'; + } + + @override + String get replace_downloaded_tracks => 'Zamień wszystkie pobrane utwory'; + + @override + String get skip_download_tracks => + 'Pomiń pobieranie wszystkich pobranych utworów'; + + @override + String get do_you_want_to_replace => 'Chcesz zamienić istniejący utwór ??'; + + @override + String get replace => 'Zamień'; + + @override + String get skip => 'Pomiń'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Wybierz do $count $type'; + } + + @override + String get select_genres => 'Wybierz Gatunki'; + + @override + String get add_genres => 'Dodaj Gatunki'; + + @override + String get country => 'Kraj'; + + @override + String get number_of_tracks_generate => 'Liczba utworów do wygenerowania'; + + @override + String get acousticness => 'Akustyczna'; + + @override + String get danceability => 'Taneczna'; + + @override + String get energy => 'Energiczna'; + + @override + String get instrumentalness => 'Instrumentalna'; + + @override + String get liveness => 'Żywa'; + + @override + String get loudness => 'Głośna'; + + @override + String get speechiness => 'Wymowna'; + + @override + String get valence => 'Wartościowa'; + + @override + String get popularity => 'Popularność'; + + @override + String get key => 'Kluczowa'; + + @override + String get duration => 'Długość (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Tryb'; + + @override + String get time_signature => 'Sygnatura Czasowa'; + + @override + String get short => 'Krótka'; + + @override + String get medium => 'Średnia'; + + @override + String get long => 'Długa'; + + @override + String get min => 'Minimalnie'; + + @override + String get max => 'Maksymalnie'; + + @override + String get target => 'Cel'; + + @override + String get moderate => 'Umiarkowanie'; + + @override + String get deselect_all => 'Odznacz wszystkie'; + + @override + String get select_all => 'Zaznacz wszystkie'; + + @override + String get are_you_sure => 'Jesteś pewny?'; + + @override + String get generating_playlist => 'Generowanie twojej własnej playlisty...'; + + @override + String selected_count_tracks(Object count) { + return 'Wybrano $count utworów'; + } + + @override + String get download_warning => + 'Jeśli hurtowo pobierasz wszystkie utwory, wyraźnie piracisz muzykę i wyrządzasz szkody kreatywnej społeczności muzycznej. Mam nadzieję, że jesteś tego świadomy. Zawsze staraj się szanować i wspierać ciężką pracę Artysty'; + + @override + String get download_ip_ban_warning => + 'Przy okazji, Twój adres IP może zostać zablokowany w YouTube z powodu nadmiernych żądań pobierania niż zwykle. Blokada IP oznacza, że nie możesz korzystać z YouTube (nawet jeśli jesteś zalogowany) przez co najmniej 2-3 miesiące z IP tego urządzenia. Spotube nie ponosi żadnej odpowiedzialności, jeśli tak się stanie'; + + @override + String get by_clicking_accept_terms => + 'Klikając \'Akceptuj\' zgadzasz się z następującymi warunkami:'; + + @override + String get download_agreement_1 => 'Wiem, że piracę muzykę. Jestem zły.'; + + @override + String get download_agreement_2 => + 'Będę wspierał artystę i robię to tylko dlatego, że nie mam pieniędzy na albumy wykonawcy. '; + + @override + String get download_agreement_3 => + 'Jestem całkowicie świadomy, że moje IP może zostać zablokowane w YouTube i nie pociągam Spotube ani jego właścicieli/współtwórców do odpowiedzialności za jakiekolwiek wypadki spowodowane moimi obecnymi działaniami'; + + @override + String get decline => 'Odrzuć'; + + @override + String get accept => 'Akceptuj'; + + @override + String get details => 'Szczegóły'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanał'; + + @override + String get likes => 'Polubienia'; + + @override + String get dislikes => 'Nie lubi'; + + @override + String get views => 'Wyświetlenia'; + + @override + String get streamUrl => 'URL strumienia'; + + @override + String get stop => 'Stop'; + + @override + String get sort_newest => 'Sortuj według ostatnio dodanych'; + + @override + String get sort_oldest => 'Sortuj według najstarszych dodanych'; + + @override + String get sleep_timer => 'Minutnik'; + + @override + String mins(Object minutes) { + return '$minutes Minuty'; + } + + @override + String hours(Object hours) { + return '$hours Godziny'; + } + + @override + String hour(Object hours) { + return '$hours Godzina'; + } + + @override + String get custom_hours => 'Własne godziny'; + + @override + String get logs => 'Logi'; + + @override + String get developers => 'Developerzy'; + + @override + String get not_logged_in => 'Nie jesteś zalogowany'; + + @override + String get search_mode => 'Tryb szukania'; + + @override + String get audio_source => 'Źródło dźwięku'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Nie można zaszyfrować :('; + + @override + String get encryption_failed_warning => + 'Spotube używa szyfrowania do bezpiecznego przechowywania danych. Ale nie udało się tego zrobić. Więc powróci do niezabezpieczonego przechowywania\nJeśli używasz Linuksa, upewnij się, że masz zainstalowane jakieś usługi do szyfrowania (gnome-keyring, kde-wallet, keepassxc itp.)'; + + @override + String get querying_info => 'Szukam informacji...'; + + @override + String get piped_api_down => 'API Piped jest niedostępne'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Instancja Piped $pipedInstance jest obecnie niedostępna\n\nZmień instancję lub zmień \'Rodzaj API\' na oficjalne API YouTube\n\nUpewnij się, że po zmianie zrestartujesz aplikację'; + } + + @override + String get you_are_offline => 'Obecnie jesteś offline'; + + @override + String get connection_restored => + 'Twoje połączenie z internetem zostało przywrócone'; + + @override + String get use_system_title_bar => 'Użyj paska tytułu systemu'; + + @override + String get crunching_results => 'Przetwarzanie wyników...'; + + @override + String get search_to_get_results => 'Szukaj, aby uzyskać wyniki'; + + @override + String get use_amoled_mode => 'Tryb AMOLED'; + + @override + String get pitch_dark_theme => 'Ciemny motyw'; + + @override + String get normalize_audio => 'Normalizuj dźwięk'; + + @override + String get change_cover => 'Zmień okładkę'; + + @override + String get add_cover => 'Dodaj okładkę'; + + @override + String get restore_defaults => 'Przywróć domyślne'; + + @override + String get download_music_format => 'Format pobierania muzyki'; + + @override + String get streaming_music_format => 'Format strumieniowania muzyki'; + + @override + String get download_music_quality => 'Jakość pobierania'; + + @override + String get streaming_music_quality => 'Jakość strumieniowania'; + + @override + String get login_with_lastfm => 'Zaloguj się z Last.fm'; + + @override + String get connect => 'Połącz'; + + @override + String get disconnect_lastfm => 'Rozłącz z Last.fm'; + + @override + String get disconnect => 'Rozłącz'; + + @override + String get username => 'Nazwa użytkownika'; + + @override + String get password => 'Hasło'; + + @override + String get login => 'Zaloguj'; + + @override + String get login_with_your_lastfm => 'Zaloguj się na swoje konto Last.fm'; + + @override + String get scrobble_to_lastfm => 'Scrobbluj do Last.fm'; + + @override + String get go_to_album => 'Przejdź do albumu'; + + @override + String get discord_rich_presence => 'Obecność na Discordzie'; + + @override + String get browse_all => 'Przeglądaj wszystko'; + + @override + String get genres => 'Gatunki muzyczne'; + + @override + String get explore_genres => 'Eksploruj gatunki'; + + @override + String get friends => 'Przyjaciele'; + + @override + String get no_lyrics_available => + 'Przepraszamy, nie można znaleźć tekstu dla tego utworu'; + + @override + String get start_a_radio => 'Uruchom radio'; + + @override + String get how_to_start_radio => 'Jak chcesz uruchomić radio?'; + + @override + String get replace_queue_question => + 'Czy chcesz zastąpić bieżącą kolejkę czy dodać do niej?'; + + @override + String get endless_playback => 'Nieskończona Odtwarzanie'; + + @override + String get delete_playlist => 'Usuń Playlistę'; + + @override + String get delete_playlist_confirmation => + 'Czy na pewno chcesz usunąć tę listę odtwarzania?'; + + @override + String get local_tracks => 'Lokalne Utwory'; + + @override + String get local_tab => 'Lokalny'; + + @override + String get song_link => 'Link do Utworu'; + + @override + String get skip_this_nonsense => 'Pomiń tę bzdurę'; + + @override + String get freedom_of_music => '“Wolność Muzyki”'; + + @override + String get freedom_of_music_palm => '“Wolność Muzyki w Twojej dłoni”'; + + @override + String get get_started => 'Zacznijmy'; + + @override + String get youtube_source_description => 'Polecane i działa najlepiej.'; + + @override + String get piped_source_description => + 'Czujesz się wolny? To samo co YouTube, ale dużo za darmo.'; + + @override + String get jiosaavn_source_description => + 'Najlepszy dla regionu Azji Południowej.'; + + @override + String get invidious_source_description => + 'Podobne do Piped, ale o wyższej dostępności.'; + + @override + String highest_quality(Object quality) { + return 'Najwyższa Jakość: $quality'; + } + + @override + String get select_audio_source => 'Wybierz Źródło Audio'; + + @override + String get endless_playback_description => + 'Automatycznie dodaj nowe utwory na koniec kolejki'; + + @override + String get choose_your_region => 'Wybierz swoją region'; + + @override + String get choose_your_region_description => + 'To pomoże Spotube pokazać Ci odpowiednią treść dla Twojej lokalizacji.'; + + @override + String get choose_your_language => 'Wybierz swój język'; + + @override + String get help_project_grow => 'Pomóż temu projektowi rosnąć'; + + @override + String get help_project_grow_description => + 'Spotube to projekt open-source. Możesz pomóc temu projektowi rosnąć, przyczyniając się do projektu, zgłaszając błędy lub sugerując nowe funkcje.'; + + @override + String get contribute_on_github => 'Przyczyniaj się na GitHubie'; + + @override + String get donate_on_open_collective => 'Dotuj na Open Collective'; + + @override + String get browse_anonymously => 'Przeglądaj Anonimowo'; + + @override + String get enable_connect => 'Włącz połączenie'; + + @override + String get enable_connect_description => + 'Kontroluj Spotube z innych urządzeń'; + + @override + String get devices => 'Urządzenia'; + + @override + String get select => 'Wybierz'; + + @override + String connect_client_alert(Object client) { + return 'Jesteś sterowany przez $client'; + } + + @override + String get this_device => 'To urządzenie'; + + @override + String get remote => 'Zdalny'; + + @override + String get stats => 'Statystyki'; + + @override + String and_n_more(Object count) { + return 'i $count więcej'; + } + + @override + String get recently_played => 'Ostatnio odtwarzane'; + + @override + String get browse_more => 'Zobacz więcej'; + + @override + String get no_title => 'Brak tytułu'; + + @override + String get not_playing => 'Nie odtwarzane'; + + @override + String get epic_failure => 'Epicka porażka!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Dodano $tracks_length utworów do kolejki'; + } + + @override + String get spotube_has_an_update => 'Spotube ma aktualizację'; + + @override + String get download_now => 'Pobierz teraz'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum został wydany'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version został wydany'; + } + + @override + String get read_the_latest => 'Przeczytaj najnowsze '; + + @override + String get release_notes => 'notatki o wersji'; + + @override + String get pick_color_scheme => 'Wybierz schemat kolorów'; + + @override + String get save => 'Zapisz'; + + @override + String get choose_the_device => 'Wybierz urządzenie:'; + + @override + String get multiple_device_connected => + 'Jest wiele urządzeń podłączonych.\nWybierz urządzenie, na którym chcesz wykonać tę akcję'; + + @override + String get nothing_found => 'Nic nie znaleziono'; + + @override + String get the_box_is_empty => 'Pudełko jest puste'; + + @override + String get top_artists => 'Najlepsi artyści'; + + @override + String get top_albums => 'Najlepsze albumy'; + + @override + String get this_week => 'W tym tygodniu'; + + @override + String get this_month => 'W tym miesiącu'; + + @override + String get last_6_months => 'Ostatnie 6 miesięcy'; + + @override + String get this_year => 'W tym roku'; + + @override + String get last_2_years => 'Ostatnie 2 lata'; + + @override + String get all_time => 'Wszystkie czasy'; + + @override + String powered_by_provider(Object providerName) { + return 'Napędzane przez $providerName'; + } + + @override + String get email => 'E-mail'; + + @override + String get profile_followers => 'Obserwujący'; + + @override + String get birthday => 'Data urodzenia'; + + @override + String get subscription => 'Subskrypcja'; + + @override + String get not_born => 'Nie urodzony'; + + @override + String get hacker => 'Haker'; + + @override + String get profile => 'Profil'; + + @override + String get no_name => 'Brak nazwy'; + + @override + String get edit => 'Edytuj'; + + @override + String get user_profile => 'Profil użytkownika'; + + @override + String count_plays(Object count) { + return '$count odtworzeń'; + } + + @override + String get streaming_fees_hypothetical => + '*Obliczone na podstawie wypłaty Spotify za stream\nod \$0.003 do \$0.005. Jest to hipotetyczne\nobliczenie, które ma na celu pokazanie, ile\nużytkownik zapłaciłby artystom, gdyby odsłuchał\ntych utworów na Spotify.'; + + @override + String get minutes_listened => 'Minuty odsłuchane'; + + @override + String get streamed_songs => 'Strumieniowane utwory'; + + @override + String count_streams(Object count) { + return '$count strumieni'; + } + + @override + String get owned_by_you => 'Własność Twoja'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl skopiowano do schowka'; + } + + @override + String get hipotetical_calculation => + '*Jest to obliczone na podstawie średniej wypłaty z internetowych platform streamingowych za jeden stream w wysokości 0,003 do 0,005 USD. Jest to hipotetyczne obliczenie, które ma na celu dać użytkownikowi wgląd w to, ile zapłaciłby artystom, gdyby słuchał ich piosenek na różnych platformach streamingowych.'; + + @override + String count_mins(Object minutes) { + return '$minutes min'; + } + + @override + String get summary_minutes => 'minuty'; + + @override + String get summary_listened_to_music => 'Słuchana muzyka'; + + @override + String get summary_songs => 'utwory'; + + @override + String get summary_streamed_overall => 'Ogółem streamowane'; + + @override + String get summary_owed_to_artists => 'Do zapłaty artystom\nw tym miesiącu'; + + @override + String get summary_artists => 'artystów'; + + @override + String get summary_music_reached_you => 'Muzyka dotarła do Ciebie'; + + @override + String get summary_full_albums => 'pełne albumy'; + + @override + String get summary_got_your_love => 'Otrzymał Twoją miłość'; + + @override + String get summary_playlists => 'playlisty'; + + @override + String get summary_were_on_repeat => 'Były na powtarzaniu'; + + @override + String total_money(Object money) { + return 'Łącznie $money'; + } + + @override + String get webview_not_found => 'Nie znaleziono Webview'; + + @override + String get webview_not_found_description => + 'Na twoim urządzeniu nie zainstalowano środowiska uruchomieniowego Webview.\nJeśli jest zainstalowany, upewnij się, że jest w environment PATH\n\nPo instalacji uruchom ponownie aplikację'; + + @override + String get unsupported_platform => 'Nieobsługiwana platforma'; + + @override + String get cache_music => 'Pamięć podręczna muzyki'; + + @override + String get open => 'Otwórz'; + + @override + String get cache_folder => 'Folder pamięci podręcznej'; + + @override + String get export => 'Eksportuj'; + + @override + String get clear_cache => 'Wyczyść pamięć podręczną'; + + @override + String get clear_cache_confirmation => + 'Czy chcesz wyczyścić pamięć podręczną?'; + + @override + String get export_cache_files => 'Eksportuj pliki z pamięci podręcznej'; + + @override + String found_n_files(Object count) { + return 'Znaleziono $count plików'; + } + + @override + String get export_cache_confirmation => + 'Czy chcesz wyeksportować te pliki do'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Wyeksportowano $filesExported z $files plików'; + } + + @override + String get undo => 'Cofnij'; + + @override + String get download_all => 'Pobierz wszystko'; + + @override + String get add_all_to_playlist => 'Dodaj wszystko do playlisty'; + + @override + String get add_all_to_queue => 'Dodaj wszystko do kolejki'; + + @override + String get play_all_next => 'Odtwórz wszystko następnie'; + + @override + String get pause => 'Pauza'; + + @override + String get view_all => 'Zobacz wszystko'; + + @override + String get no_tracks_added_yet => + 'Wygląda na to, że jeszcze nie dodałeś żadnych utworów'; + + @override + String get no_tracks => 'Wygląda na to, że tutaj nie ma żadnych utworów'; + + @override + String get no_tracks_listened_yet => + 'Wygląda na to, że jeszcze nic nie słuchałeś'; + + @override + String get not_following_artists => 'Nie obserwujesz żadnych artystów'; + + @override + String get no_favorite_albums_yet => + 'Wygląda na to, że jeszcze nie dodałeś żadnych albumów do ulubionych'; + + @override + String get no_logs_found => 'Nie znaleziono żadnych logów'; + + @override + String get youtube_engine => 'Silnik YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine nie jest zainstalowany'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine nie jest zainstalowany w systemie.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Upewnij się, że jest dostępny w zmiennej PATH lub\nustaw absolutną ścieżkę do pliku wykonywalnego $engine poniżej'; + } + + @override + String get youtube_engine_unix_issue_message => + 'W systemach macOS/Linux/unix, ustawianie ścieżki w .zshrc/.bashrc/.bash_profile itp. nie będzie działać.\nMusisz ustawić ścieżkę w pliku konfiguracyjnym powłoki'; + + @override + String get download => 'Pobierz'; + + @override + String get file_not_found => 'Plik nie znaleziony'; + + @override + String get custom => 'Niestandardowy'; + + @override + String get add_custom_url => 'Dodaj niestandardowy URL'; + + @override + String get edit_port => 'Edytuj port'; + + @override + String get port_helper_msg => + 'Domyślna wartość to -1, co oznacza losową liczbę. Jeśli masz skonfigurowany zaporę, zaleca się jej ustawienie.'; + + @override + String connect_request(Object client) { + return 'Zezwolić $client na połączenie?'; + } + + @override + String get connection_request_denied => + 'Połączenie odrzucone. Użytkownik odmówił dostępu.'; + + @override + String get an_error_occurred => 'Wystąpił błąd'; + + @override + String get copy_to_clipboard => 'Kopiuj do schowka'; + + @override + String get view_logs => 'Wyświetl logi'; + + @override + String get retry => 'Ponów'; + + @override + String get no_default_metadata_provider_selected => + 'Nie masz ustawionego domyślnego dostawcy metadanych'; + + @override + String get manage_metadata_providers => 'Zarządzaj dostawcami metadanych'; + + @override + String get open_link_in_browser => 'Otworzyć link w przeglądarce?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Czy chcesz otworzyć następujący link'; + + @override + String get unsafe_url_warning => + 'Otwieranie linków z niezaufanych źródeł może być niebezpieczne. Zachowaj ostrożność!\nMożesz również skopiować link do schowka.'; + + @override + String get copy_link => 'Kopiuj link'; + + @override + String get building_your_timeline => + 'Budowanie Twojej osi czasu na podstawie Twoich odsłuchań...'; + + @override + String get official => 'Oficjalny'; + + @override + String author_name(Object author) { + return 'Autor: $author'; + } + + @override + String get third_party => 'Zewnętrzny'; + + @override + String get plugin_requires_authentication => + 'Wtyczka wymaga uwierzytelnienia'; + + @override + String get update_available => 'Dostępna aktualizacja'; + + @override + String get supports_scrobbling => 'Obsługuje scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Ta wtyczka scrobbluje Twoją muzykę, aby wygenerować historię odsłuchań.'; + + @override + String get default_metadata_source => 'Domyślne źródło metadanych'; + + @override + String get set_default_metadata_source => 'Ustaw domyślne źródło metadanych'; + + @override + String get default_audio_source => 'Domyślne źródło audio'; + + @override + String get set_default_audio_source => 'Ustaw domyślne źródło audio'; + + @override + String get set_default => 'Ustaw jako domyślną'; + + @override + String get support => 'Wsparcie'; + + @override + String get support_plugin_development => 'Wspieraj rozwój wtyczki'; + + @override + String can_access_name_api(Object name) { + return '- Może uzyskać dostęp do API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Czy chcesz zainstalować tę wtyczkę?'; + + @override + String get third_party_plugin_warning => + 'Ta wtyczka pochodzi z zewnętrznego repozytorium. Upewnij się, że ufasz źródłu przed instalacją.'; + + @override + String get author => 'Autor'; + + @override + String get this_plugin_can_do_following => + 'Ta wtyczka może wykonywać następujące czynności'; + + @override + String get install => 'Instaluj'; + + @override + String get install_a_metadata_provider => 'Zainstaluj dostawcę metadanych'; + + @override + String get no_tracks_playing => 'Obecnie nie odtwarzany jest żaden utwór'; + + @override + String get synced_lyrics_not_available => + 'Zsynchronizowane teksty nie są dostępne dla tego utworu. Zamiast tego użyj zakładki'; + + @override + String get plain_lyrics => 'Zwykłe teksty'; + + @override + String get tab_instead => 'zamiast tego.'; + + @override + String get disclaimer => 'Zastrzeżenie'; + + @override + String get third_party_plugin_dmca_notice => + 'Zespół Spotube nie ponosi żadnej odpowiedzialności (w tym prawnej) za żadne wtyczki \"zewnętrzne\".\nUżywaj ich na własne ryzyko. Wszelkie błędy/problemy prosimy zgłaszać w repozytorium wtyczki.\n\nJeśli jakakolwiek wtyczka \"zewnętrzna\" narusza ToS/DMCA jakiejkolwiek usługi/podmiotu prawnego, prosimy o kontakt z autorem wtyczki \"zewnętrznej\" lub platformą hostingową, np. GitHub/Codeberg, w celu podjęcia działań. Wymienione powyżej (oznaczone jako \"zewnętrzne\") są publicznymi wtyczkami utrzymywanymi przez społeczność. Nie kuratujemy ich, więc nie możemy podjąć żadnych działań w ich sprawie.\n\n'; + + @override + String get input_does_not_match_format => + 'Wprowadzony tekst nie pasuje do wymaganego formatu'; + + @override + String get plugins => 'Wtyczki'; + + @override + String get paste_plugin_download_url => + 'Wklej adres URL do pobrania lub adres URL repozytorium GitHub/Codeberg lub bezpośredni link do pliku .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Pobierz i zainstaluj wtyczkę z adresu URL'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Nie udało się dodać wtyczki: $error'; + } + + @override + String get upload_plugin_from_file => 'Prześlij wtyczkę z pliku'; + + @override + String get installed => 'Zainstalowane'; + + @override + String get available_plugins => 'Dostępne wtyczki'; + + @override + String get configure_plugins => + 'Skonfiguruj własne wtyczki dostawców metadanych i źródeł audio'; + + @override + String get audio_scrobblers => 'Scrobblery audio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Źródło: '; + + @override + String get uncompressed => 'Nieskompresowany'; + + @override + String get dab_music_source_description => + 'Dla audiofilów. Oferuje strumienie audio wysokiej jakości/lossless. Precyzyjne dopasowanie utworów na podstawie ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_pt.dart b/lib/l10n/generated/app_localizations_pt.dart new file mode 100644 index 00000000..8d2eabe7 --- /dev/null +++ b/lib/l10n/generated/app_localizations_pt.dart @@ -0,0 +1,1569 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Portuguese (`pt`). +class AppLocalizationsPt extends AppLocalizations { + AppLocalizationsPt([String locale = 'pt']) : super(locale); + + @override + String get guest => 'Visitante'; + + @override + String get browse => 'Explorar'; + + @override + String get search => 'Buscar'; + + @override + String get library => 'Biblioteca'; + + @override + String get lyrics => 'Letras'; + + @override + String get settings => 'Configurações'; + + @override + String get genre_categories_filter => 'Filtrar categorias ou gêneros...'; + + @override + String get genre => 'Gênero'; + + @override + String get personalized => 'Personalizado'; + + @override + String get featured => 'Destaque'; + + @override + String get new_releases => 'Novos Lançamentos'; + + @override + String get songs => 'Músicas'; + + @override + String playing_track(Object track) { + return 'Tocando $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Isso irá limpar a fila atual. $track_length músicas serão removidas.\nDeseja continuar?'; + } + + @override + String get load_more => 'Carregar mais'; + + @override + String get playlists => 'Playlists'; + + @override + String get artists => 'Artistas'; + + @override + String get albums => 'Álbuns'; + + @override + String get tracks => 'Faixas'; + + @override + String get downloads => 'Downloads'; + + @override + String get filter_playlists => 'Filtrar suas playlists...'; + + @override + String get liked_tracks => 'Músicas Curtidas'; + + @override + String get liked_tracks_description => 'Todas as suas músicas curtidas'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Criar uma playlist'; + + @override + String get update_playlist => 'Atualizar lista de reprodução'; + + @override + String get create => 'Criar'; + + @override + String get cancel => 'Cancelar'; + + @override + String get update => 'Atualizar'; + + @override + String get playlist_name => 'Nome da Playlist'; + + @override + String get name_of_playlist => 'Nome da playlist'; + + @override + String get description => 'Descrição'; + + @override + String get public => 'Pública'; + + @override + String get collaborative => 'Colaborativa'; + + @override + String get search_local_tracks => 'Buscar músicas locais...'; + + @override + String get play => 'Reproduzir'; + + @override + String get delete => 'Excluir'; + + @override + String get none => 'Nenhum'; + + @override + String get sort_a_z => 'Ordenar de A-Z'; + + @override + String get sort_z_a => 'Ordenar de Z-A'; + + @override + String get sort_artist => 'Ordenar por Artista'; + + @override + String get sort_album => 'Ordenar por Álbum'; + + @override + String get sort_duration => 'Ordenar por Duração'; + + @override + String get sort_tracks => 'Ordenar Faixas'; + + @override + String currently_downloading(Object tracks_length) { + return 'Baixando no momento ($tracks_length)'; + } + + @override + String get cancel_all => 'Cancelar Tudo'; + + @override + String get filter_artist => 'Filtrar artistas...'; + + @override + String followers(Object followers) { + return '$followers Seguidores'; + } + + @override + String get add_artist_to_blacklist => 'Adicionar artista à lista negra'; + + @override + String get top_tracks => 'Principais Músicas'; + + @override + String get fans_also_like => 'Fãs também curtiram'; + + @override + String get loading => 'Carregando...'; + + @override + String get artist => 'Artista'; + + @override + String get blacklisted => 'Na Lista Negra'; + + @override + String get following => 'Seguindo'; + + @override + String get follow => 'Seguir'; + + @override + String get artist_url_copied => + 'URL do artista copiada para a área de transferência'; + + @override + String added_to_queue(Object tracks) { + return 'Adicionadas $tracks músicas à fila'; + } + + @override + String get filter_albums => 'Filtrar álbuns...'; + + @override + String get synced => 'Sincronizado'; + + @override + String get plain => 'Simples'; + + @override + String get shuffle => 'Aleatório'; + + @override + String get search_tracks => 'Buscar músicas...'; + + @override + String get released => 'Lançado'; + + @override + String error(Object error) { + return 'Erro $error'; + } + + @override + String get title => 'Título'; + + @override + String get time => 'Tempo'; + + @override + String get more_actions => 'Mais ações'; + + @override + String download_count(Object count) { + return 'Baixar ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Adicionar ($count) à Playlist'; + } + + @override + String add_count_to_queue(Object count) { + return 'Adicionar ($count) à Fila'; + } + + @override + String play_count_next(Object count) { + return 'Reproduzir ($count) em seguida'; + } + + @override + String get album => 'Álbum'; + + @override + String copied_to_clipboard(Object data) { + return '$data copiado para a área de transferência'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Adicionar $track às Playlists Seguintes'; + } + + @override + String get add => 'Adicionar'; + + @override + String added_track_to_queue(Object track) { + return 'Adicionada $track à fila'; + } + + @override + String get add_to_queue => 'Adicionar à fila'; + + @override + String track_will_play_next(Object track) { + return '$track será reproduzida em seguida'; + } + + @override + String get play_next => 'Reproduzir em seguida'; + + @override + String removed_track_from_queue(Object track) { + return '$track removida da fila'; + } + + @override + String get remove_from_queue => 'Remover da fila'; + + @override + String get remove_from_favorites => 'Remover dos favoritos'; + + @override + String get save_as_favorite => 'Salvar como favorita'; + + @override + String get add_to_playlist => 'Adicionar à playlist'; + + @override + String get remove_from_playlist => 'Remover da playlist'; + + @override + String get add_to_blacklist => 'Adicionar à lista negra'; + + @override + String get remove_from_blacklist => 'Remover da lista negra'; + + @override + String get share => 'Compartilhar'; + + @override + String get mini_player => 'Mini Player'; + + @override + String get slide_to_seek => 'Arraste para avançar ou retroceder'; + + @override + String get shuffle_playlist => 'Embaralhar playlist'; + + @override + String get unshuffle_playlist => 'Desembaralhar playlist'; + + @override + String get previous_track => 'Faixa anterior'; + + @override + String get next_track => 'Próxima faixa'; + + @override + String get pause_playback => 'Pausar Reprodução'; + + @override + String get resume_playback => 'Continuar Reprodução'; + + @override + String get loop_track => 'Repetir faixa'; + + @override + String get no_loop => 'Sem loop'; + + @override + String get repeat_playlist => 'Repetir playlist'; + + @override + String get queue => 'Fila'; + + @override + String get alternative_track_sources => 'Fontes alternativas de faixas'; + + @override + String get download_track => 'Baixar faixa'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks músicas na fila'; + } + + @override + String get clear_all => 'Limpar tudo'; + + @override + String get show_hide_ui_on_hover => 'Mostrar/Ocultar UI ao passar o mouse'; + + @override + String get always_on_top => 'Sempre no topo'; + + @override + String get exit_mini_player => 'Sair do Mini player'; + + @override + String get download_location => 'Local de download'; + + @override + String get local_library => 'Biblioteca local'; + + @override + String get add_library_location => 'Adicionar à biblioteca'; + + @override + String get remove_library_location => 'Remover da biblioteca'; + + @override + String get account => 'Conta'; + + @override + String get logout => 'Sair'; + + @override + String get logout_of_this_account => 'Sair desta conta'; + + @override + String get language_region => 'Idioma e Região'; + + @override + String get language => 'Idioma'; + + @override + String get system_default => 'Padrão do Sistema'; + + @override + String get market_place_region => 'Região da Loja'; + + @override + String get recommendation_country => 'País de Recomendação'; + + @override + String get appearance => 'Aparência'; + + @override + String get layout_mode => 'Modo de Layout'; + + @override + String get override_layout_settings => + 'Substituir configurações do modo de layout responsivo'; + + @override + String get adaptive => 'Adaptável'; + + @override + String get compact => 'Compacto'; + + @override + String get extended => 'Estendido'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Escuro'; + + @override + String get light => 'Claro'; + + @override + String get system => 'Sistema'; + + @override + String get accent_color => 'Cor de Destaque'; + + @override + String get sync_album_color => 'Sincronizar cor do álbum'; + + @override + String get sync_album_color_description => + 'Usa a cor predominante da capa do álbum como cor de destaque'; + + @override + String get playback => 'Reprodução'; + + @override + String get audio_quality => 'Qualidade do Áudio'; + + @override + String get high => 'Alta'; + + @override + String get low => 'Baixa'; + + @override + String get pre_download_play => 'Pré-download e reprodução'; + + @override + String get pre_download_play_description => + 'Em vez de transmitir áudio, baixar bytes e reproduzir (recomendado para usuários com maior largura de banda)'; + + @override + String get skip_non_music => 'Pular segmentos não musicais (SponsorBlock)'; + + @override + String get blacklist_description => 'Faixas e artistas na lista negra'; + + @override + String get wait_for_download_to_finish => + 'Aguarde o download atual ser concluído'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Comportamento de Fechamento'; + + @override + String get close => 'Fechar'; + + @override + String get minimize_to_tray => 'Minimizar para a bandeja'; + + @override + String get show_tray_icon => 'Mostrar ícone na bandeja do sistema'; + + @override + String get about => 'Sobre'; + + @override + String get u_love_spotube => 'Sabemos que você adora o Spotube'; + + @override + String get check_for_updates => 'Verificar atualizações'; + + @override + String get about_spotube => 'Sobre o Spotube'; + + @override + String get blacklist => 'Lista Negra'; + + @override + String get please_sponsor => 'Por favor, patrocine/doe'; + + @override + String get spotube_description => + 'Spotube, um cliente leve, multiplataforma e gratuito para o Spotify'; + + @override + String get version => 'Versão'; + + @override + String get build_number => 'Número de Build'; + + @override + String get founder => 'Fundador'; + + @override + String get repository => 'Repositório'; + + @override + String get bug_issues => 'Bugs/Problemas'; + + @override + String get made_with => 'Feito com ❤️ em Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Licença'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Não se preocupe, suas credenciais não serão coletadas nem compartilhadas com ninguém'; + + @override + String get know_how_to_login => 'Não sabe como fazer isso?'; + + @override + String get follow_step_by_step_guide => 'Siga o guia passo a passo'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Preencha todos os campos, por favor'; + + @override + String get submit => 'Enviar'; + + @override + String get exit => 'Sair'; + + @override + String get previous => 'Anterior'; + + @override + String get next => 'Próximo'; + + @override + String get done => 'Concluído'; + + @override + String get step_1 => 'Passo 1'; + + @override + String get first_go_to => 'Primeiro, vá para'; + + @override + String get something_went_wrong => 'Algo deu errado'; + + @override + String get piped_instance => 'Instância do Servidor Piped'; + + @override + String get piped_description => + 'A instância do servidor Piped a ser usada para correspondência de faixas'; + + @override + String get piped_warning => + 'Algumas delas podem não funcionar bem. Use por sua conta e risco'; + + @override + String get invidious_instance => 'Instância do Servidor Invidious'; + + @override + String get invidious_description => + 'A instância do servidor Invidious a ser usada para correspondência de faixas'; + + @override + String get invidious_warning => + 'Alguns podem não funcionar bem. Use por sua conta e risco'; + + @override + String get generate => 'Gerar'; + + @override + String track_exists(Object track) { + return 'A faixa $track já existe'; + } + + @override + String get replace_downloaded_tracks => 'Substituir todas as faixas baixadas'; + + @override + String get skip_download_tracks => + 'Pular o download de todas as faixas baixadas'; + + @override + String get do_you_want_to_replace => 'Deseja substituir a faixa existente?'; + + @override + String get replace => 'Substituir'; + + @override + String get skip => 'Pular'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Selecione até $count $type'; + } + + @override + String get select_genres => 'Selecionar Gêneros'; + + @override + String get add_genres => 'Adicionar Gêneros'; + + @override + String get country => 'País'; + + @override + String get number_of_tracks_generate => 'Número de faixas a gerar'; + + @override + String get acousticness => 'Acústica'; + + @override + String get danceability => 'Dançabilidade'; + + @override + String get energy => 'Energia'; + + @override + String get instrumentalness => 'Instrumentalidade'; + + @override + String get liveness => 'Vivacidade'; + + @override + String get loudness => 'Volume'; + + @override + String get speechiness => 'Discurso'; + + @override + String get valence => 'Valência'; + + @override + String get popularity => 'Popularidade'; + + @override + String get key => 'Tonalidade'; + + @override + String get duration => 'Duração (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Modo'; + + @override + String get time_signature => 'Assinatura de tempo'; + + @override + String get short => 'Curto'; + + @override + String get medium => 'Médio'; + + @override + String get long => 'Longo'; + + @override + String get min => 'Min'; + + @override + String get max => 'Máx'; + + @override + String get target => 'Alvo'; + + @override + String get moderate => 'Moderado'; + + @override + String get deselect_all => 'Desmarcar Todos'; + + @override + String get select_all => 'Selecionar Todos'; + + @override + String get are_you_sure => 'Tem certeza?'; + + @override + String get generating_playlist => 'Gerando sua playlist personalizada...'; + + @override + String selected_count_tracks(Object count) { + return '$count faixas selecionadas'; + } + + @override + String get download_warning => + 'Se você baixar todas as faixas em massa, estará claramente pirateando música e causando danos à sociedade criativa da música. Espero que você esteja ciente disso. Sempre tente respeitar e apoiar o trabalho árduo dos artistas'; + + @override + String get download_ip_ban_warning => + 'Além disso, seu IP pode ser bloqueado no YouTube devido a solicitações de download excessivas. O bloqueio de IP significa que você não poderá usar o YouTube (mesmo se estiver conectado) por pelo menos 2-3 meses a partir do dispositivo IP. E o Spotube não se responsabiliza se isso acontecer'; + + @override + String get by_clicking_accept_terms => + 'Ao clicar em \'aceitar\', você concorda com os seguintes termos:'; + + @override + String get download_agreement_1 => + 'Eu sei que estou pirateando música. Sou mau'; + + @override + String get download_agreement_2 => + 'Vou apoiar o artista onde puder e estou fazendo isso porque não tenho dinheiro para comprar sua arte'; + + @override + String get download_agreement_3 => + 'Estou completamente ciente de que meu IP pode ser bloqueado no YouTube e não responsabilizo o Spotube ou seus proprietários/colaboradores por quaisquer acidentes causados pela minha ação atual'; + + @override + String get decline => 'Recusar'; + + @override + String get accept => 'Aceitar'; + + @override + String get details => 'Detalhes'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Canal'; + + @override + String get likes => 'Curtidas'; + + @override + String get dislikes => 'Descurtidas'; + + @override + String get views => 'Visualizações'; + + @override + String get streamUrl => 'URL do Stream'; + + @override + String get stop => 'Parar'; + + @override + String get sort_newest => 'Ordenar por mais recente adicionado'; + + @override + String get sort_oldest => 'Ordenar por mais antigo adicionado'; + + @override + String get sleep_timer => 'Temporizador de Sono'; + + @override + String mins(Object minutes) { + return '$minutes Minutos'; + } + + @override + String hours(Object hours) { + return '$hours Horas'; + } + + @override + String hour(Object hours) { + return '$hours Hora'; + } + + @override + String get custom_hours => 'Horas Personalizadas'; + + @override + String get logs => 'Registros'; + + @override + String get developers => 'Desenvolvedores'; + + @override + String get not_logged_in => 'Você não está logado'; + + @override + String get search_mode => 'Modo de Busca'; + + @override + String get audio_source => 'Fonte de Áudio'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Falha ao criptografar'; + + @override + String get encryption_failed_warning => + 'O Spotube usa criptografia para armazenar seus dados com segurança, mas falhou em fazê-lo. Portanto, ele voltará para o armazenamento não seguro.\nSe você estiver usando o Linux, certifique-se de ter algum serviço secreto (gnome-keyring, kde-wallet, keepassxc, etc.) instalado'; + + @override + String get querying_info => 'Consultando informações...'; + + @override + String get piped_api_down => 'A API do Piped está indisponível'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'A instância do Piped $pipedInstance está atualmente indisponível\n\nMude a instância ou mude o \'Tipo de API\' para a API oficial do YouTube\n\nCertifique-se de reiniciar o aplicativo após a alteração'; + } + + @override + String get you_are_offline => 'Você está offline no momento'; + + @override + String get connection_restored => 'Sua conexão com a internet foi restaurada'; + + @override + String get use_system_title_bar => 'Usar a barra de título do sistema'; + + @override + String get crunching_results => 'Processando resultados...'; + + @override + String get search_to_get_results => 'Pesquisar para obter resultados'; + + @override + String get use_amoled_mode => 'Modo AMOLED'; + + @override + String get pitch_dark_theme => 'Tema escuro'; + + @override + String get normalize_audio => 'Normalizar áudio'; + + @override + String get change_cover => 'Alterar capa'; + + @override + String get add_cover => 'Adicionar capa'; + + @override + String get restore_defaults => 'Restaurar padrões'; + + @override + String get download_music_format => 'Formato de download de música'; + + @override + String get streaming_music_format => 'Formato de streaming de música'; + + @override + String get download_music_quality => 'Qualidade de download'; + + @override + String get streaming_music_quality => 'Qualidade de streaming'; + + @override + String get login_with_lastfm => 'Iniciar sessão com o Last.fm'; + + @override + String get connect => 'Ligar'; + + @override + String get disconnect_lastfm => 'Desligar do Last.fm'; + + @override + String get disconnect => 'Desligar'; + + @override + String get username => 'Nome de utilizador'; + + @override + String get password => 'Palavra-passe'; + + @override + String get login => 'Iniciar sessão'; + + @override + String get login_with_your_lastfm => 'Inicie sessão na sua conta Last.fm'; + + @override + String get scrobble_to_lastfm => 'Scrobble para o Last.fm'; + + @override + String get go_to_album => 'Ir para o álbum'; + + @override + String get discord_rich_presence => 'Presença rica no Discord'; + + @override + String get browse_all => 'Navegar por tudo'; + + @override + String get genres => 'Gêneros'; + + @override + String get explore_genres => 'Explorar gêneros'; + + @override + String get friends => 'Amigos'; + + @override + String get no_lyrics_available => + 'Desculpe, não foi possível encontrar a letra desta faixa'; + + @override + String get start_a_radio => 'Iniciar uma Rádio'; + + @override + String get how_to_start_radio => 'Como você deseja iniciar a rádio?'; + + @override + String get replace_queue_question => + 'Você deseja substituir a fila atual ou acrescentar a ela?'; + + @override + String get endless_playback => 'Reprodução sem fim'; + + @override + String get delete_playlist => 'Excluir Lista de Reprodução'; + + @override + String get delete_playlist_confirmation => + 'Tem certeza de que deseja excluir esta lista de reprodução?'; + + @override + String get local_tracks => 'Faixas Locais'; + + @override + String get local_tab => 'Local'; + + @override + String get song_link => 'Link da Música'; + + @override + String get skip_this_nonsense => 'Pular essa bobagem'; + + @override + String get freedom_of_music => '“Liberdade da Música”'; + + @override + String get freedom_of_music_palm => + '“Liberdade da Música na palma da sua mão”'; + + @override + String get get_started => 'Vamos começar'; + + @override + String get youtube_source_description => 'Recomendado e funciona melhor.'; + + @override + String get piped_source_description => + 'Sentindo-se livre? Igual ao YouTube, mas muito mais grátis.'; + + @override + String get jiosaavn_source_description => + 'Melhor para a região da Ásia do Sul.'; + + @override + String get invidious_source_description => + 'Semelhante ao Piped, mas com maior disponibilidade.'; + + @override + String highest_quality(Object quality) { + return 'Melhor Qualidade: $quality'; + } + + @override + String get select_audio_source => 'Selecionar Fonte de Áudio'; + + @override + String get endless_playback_description => + 'Adicionar automaticamente novas músicas\nao final da fila'; + + @override + String get choose_your_region => 'Escolha sua região'; + + @override + String get choose_your_region_description => + 'Isso ajudará o Spotube a mostrar o conteúdo certo\npara sua localização.'; + + @override + String get choose_your_language => 'Escolha seu idioma'; + + @override + String get help_project_grow => 'Ajude este projeto a crescer'; + + @override + String get help_project_grow_description => + 'Spotube é um projeto de código aberto. Você pode ajudar este projeto a crescer contribuindo para o projeto, relatando bugs ou sugerindo novos recursos.'; + + @override + String get contribute_on_github => 'Contribuir no GitHub'; + + @override + String get donate_on_open_collective => 'Doar no Open Collective'; + + @override + String get browse_anonymously => 'Navegar Anonimamente'; + + @override + String get enable_connect => 'Ativar conexão'; + + @override + String get enable_connect_description => + 'Controle o Spotube a partir de outros dispositivos'; + + @override + String get devices => 'Dispositivos'; + + @override + String get select => 'Selecionar'; + + @override + String connect_client_alert(Object client) { + return 'Você está sendo controlado por $client'; + } + + @override + String get this_device => 'Este dispositivo'; + + @override + String get remote => 'Remoto'; + + @override + String get stats => 'Estatísticas'; + + @override + String and_n_more(Object count) { + return 'e $count mais'; + } + + @override + String get recently_played => 'Reproduzido Recentemente'; + + @override + String get browse_more => 'Ver Mais'; + + @override + String get no_title => 'Sem Título'; + + @override + String get not_playing => 'Não está a reproduzir'; + + @override + String get epic_failure => 'Fracasso épico!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Adicionados $tracks_length faixas à fila'; + } + + @override + String get spotube_has_an_update => 'Spotube tem uma atualização'; + + @override + String get download_now => 'Baixar Agora'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum foi lançado'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version foi lançado'; + } + + @override + String get read_the_latest => 'Leia o mais recente '; + + @override + String get release_notes => 'notas de versão'; + + @override + String get pick_color_scheme => 'Escolha o esquema de cores'; + + @override + String get save => 'Salvar'; + + @override + String get choose_the_device => 'Escolha o dispositivo:'; + + @override + String get multiple_device_connected => + 'Há vários dispositivos conectados.\nEscolha o dispositivo no qual deseja executar esta ação'; + + @override + String get nothing_found => 'Nada encontrado'; + + @override + String get the_box_is_empty => 'A caixa está vazia'; + + @override + String get top_artists => 'Principais Artistas'; + + @override + String get top_albums => 'Principais Álbuns'; + + @override + String get this_week => 'Esta semana'; + + @override + String get this_month => 'Este mês'; + + @override + String get last_6_months => 'Últimos 6 meses'; + + @override + String get this_year => 'Este ano'; + + @override + String get last_2_years => 'Últimos 2 anos'; + + @override + String get all_time => 'De todos os tempos'; + + @override + String powered_by_provider(Object providerName) { + return 'Desenvolvido por $providerName'; + } + + @override + String get email => 'E-mail'; + + @override + String get profile_followers => 'Seguidores'; + + @override + String get birthday => 'Aniversário'; + + @override + String get subscription => 'Assinatura'; + + @override + String get not_born => 'Não nascido'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Perfil'; + + @override + String get no_name => 'Sem Nome'; + + @override + String get edit => 'Editar'; + + @override + String get user_profile => 'Perfil do Usuário'; + + @override + String count_plays(Object count) { + return '$count reproduzidos'; + } + + @override + String get streaming_fees_hypothetical => + '*Calculado com base no pagamento por stream do Spotify\nque varia de \$0.003 a \$0.005. Isso é um cálculo hipotético\npara fornecer uma visão ao usuário sobre quanto eles\nteriam pago aos artistas se estivessem ouvindo\no seu som no Spotify.'; + + @override + String get minutes_listened => 'Minutos ouvidos'; + + @override + String get streamed_songs => 'Músicas transmitidas'; + + @override + String count_streams(Object count) { + return '$count streams'; + } + + @override + String get owned_by_you => 'De sua propriedade'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl copiado para a área de transferência'; + } + + @override + String get hipotetical_calculation => + '*Isso é calculado com base no pagamento médio por stream de plataformas de streaming de música online de US\$ 0,003 a US\$ 0,005. Esta é uma estimativa hipotética para dar ao usuário uma ideia de quanto ele teria pago aos artistas se ouvisse sua música em diferentes plataformas de streaming de música.'; + + @override + String count_mins(Object minutes) { + return '$minutes min'; + } + + @override + String get summary_minutes => 'minutos'; + + @override + String get summary_listened_to_music => 'Música ouvida'; + + @override + String get summary_songs => 'faixas'; + + @override + String get summary_streamed_overall => 'Total de streams'; + + @override + String get summary_owed_to_artists => 'Devido aos artistas\neste mês'; + + @override + String get summary_artists => 'artista'; + + @override + String get summary_music_reached_you => 'A música chegou até você'; + + @override + String get summary_full_albums => 'álbuns completos'; + + @override + String get summary_got_your_love => 'Recebeu seu amor'; + + @override + String get summary_playlists => 'playlists'; + + @override + String get summary_were_on_repeat => 'Estavam em repetição'; + + @override + String total_money(Object money) { + return 'Total $money'; + } + + @override + String get webview_not_found => 'Webview não encontrado'; + + @override + String get webview_not_found_description => + 'Nenhum runtime Webview está instalado no seu dispositivo.\nSe estiver instalado, certifique-se de que está no environment PATH\n\nApós a instalação, reinicie o aplicativo'; + + @override + String get unsupported_platform => 'Plataforma não suportada'; + + @override + String get cache_music => 'Música em cache'; + + @override + String get open => 'Abrir'; + + @override + String get cache_folder => 'Pasta de cache'; + + @override + String get export => 'Exportar'; + + @override + String get clear_cache => 'Limpar cache'; + + @override + String get clear_cache_confirmation => 'Deseja limpar o cache?'; + + @override + String get export_cache_files => 'Exportar Arquivos em Cache'; + + @override + String found_n_files(Object count) { + return 'Encontrados $count arquivos'; + } + + @override + String get export_cache_confirmation => 'Deseja exportar estes arquivos para'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Exportados $filesExported de $files arquivos'; + } + + @override + String get undo => 'Desfazer'; + + @override + String get download_all => 'Baixar tudo'; + + @override + String get add_all_to_playlist => 'Adicionar tudo à playlist'; + + @override + String get add_all_to_queue => 'Adicionar tudo à fila'; + + @override + String get play_all_next => 'Reproduzir tudo a seguir'; + + @override + String get pause => 'Pausar'; + + @override + String get view_all => 'Ver tudo'; + + @override + String get no_tracks_added_yet => + 'Parece que você ainda não adicionou nenhuma faixa'; + + @override + String get no_tracks => 'Parece que não há faixas aqui'; + + @override + String get no_tracks_listened_yet => 'Parece que você ainda não ouviu nada'; + + @override + String get not_following_artists => 'Você não está seguindo nenhum artista'; + + @override + String get no_favorite_albums_yet => + 'Parece que você ainda não adicionou nenhum álbum aos favoritos'; + + @override + String get no_logs_found => 'Nenhum log encontrado'; + + @override + String get youtube_engine => 'Motor YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine não está instalado'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine não está instalado no seu sistema.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Certifique-se de que está disponível na variável PATH ou\ndefina o caminho absoluto para o executável $engine abaixo'; + } + + @override + String get youtube_engine_unix_issue_message => + 'Em sistemas macOS/Linux/unix, definir o caminho no .zshrc/.bashrc/.bash_profile etc. não funcionará.\nVocê precisa definir o caminho no arquivo de configuração do shell'; + + @override + String get download => 'Baixar'; + + @override + String get file_not_found => 'Arquivo não encontrado'; + + @override + String get custom => 'Personalizado'; + + @override + String get add_custom_url => 'Adicionar URL personalizada'; + + @override + String get edit_port => 'Editar porta'; + + @override + String get port_helper_msg => + 'O padrão é -1, que indica um número aleatório. Se você tiver um firewall configurado, é recomendável definir isso.'; + + @override + String connect_request(Object client) { + return 'Permitir que $client se conecte?'; + } + + @override + String get connection_request_denied => + 'Conexão negada. O usuário negou o acesso .'; + + @override + String get an_error_occurred => 'Ocorreu um erro'; + + @override + String get copy_to_clipboard => 'Copiar para a área de transferência'; + + @override + String get view_logs => 'Ver logs'; + + @override + String get retry => 'Tentar novamente'; + + @override + String get no_default_metadata_provider_selected => + 'Você não tem um provedor de metadados padrão definido'; + + @override + String get manage_metadata_providers => 'Gerenciar provedores de metadados'; + + @override + String get open_link_in_browser => 'Abrir link no navegador?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Você deseja abrir o seguinte link'; + + @override + String get unsafe_url_warning => + 'Pode ser inseguro abrir links de fontes não confiáveis. Tenha cautela!\nVocê também pode copiar o link para sua área de transferência.'; + + @override + String get copy_link => 'Copiar link'; + + @override + String get building_your_timeline => + 'Construindo sua linha do tempo com base em suas audições...'; + + @override + String get official => 'Oficial'; + + @override + String author_name(Object author) { + return 'Autor: $author'; + } + + @override + String get third_party => 'Terceiros'; + + @override + String get plugin_requires_authentication => 'Plugin requer autenticação'; + + @override + String get update_available => 'Atualização disponível'; + + @override + String get supports_scrobbling => 'Suporta scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Este plugin faz o scrobbling de sua música para gerar seu histórico de audição.'; + + @override + String get default_metadata_source => 'Fonte padrão de metadados'; + + @override + String get set_default_metadata_source => 'Definir fonte padrão de metadados'; + + @override + String get default_audio_source => 'Fonte de áudio padrão'; + + @override + String get set_default_audio_source => 'Definir fonte de áudio padrão'; + + @override + String get set_default => 'Definir como padrão'; + + @override + String get support => 'Suporte'; + + @override + String get support_plugin_development => 'Apoiar o desenvolvimento do plugin'; + + @override + String can_access_name_api(Object name) { + return '- Pode acessar a API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Você deseja instalar este plugin?'; + + @override + String get third_party_plugin_warning => + 'Este plugin é de um repositório de terceiros. Certifique-se de que você confia na fonte antes de instalá-lo.'; + + @override + String get author => 'Autor'; + + @override + String get this_plugin_can_do_following => + 'Este plugin pode fazer o seguinte'; + + @override + String get install => 'Instalar'; + + @override + String get install_a_metadata_provider => 'Instalar um provedor de metadados'; + + @override + String get no_tracks_playing => 'Nenhuma música sendo reproduzida no momento'; + + @override + String get synced_lyrics_not_available => + 'As letras sincronizadas não estão disponíveis para esta música. Por favor, use a aba'; + + @override + String get plain_lyrics => 'Letras simples'; + + @override + String get tab_instead => 'em vez disso.'; + + @override + String get disclaimer => 'Aviso'; + + @override + String get third_party_plugin_dmca_notice => + 'A equipe Spotube não se responsabiliza (incluindo legalmente) por quaisquer plugins de \"terceiros\".\nUse-os por sua conta e risco. Para quaisquer bugs/problemas, por favor, relate-os ao repositório do plugin.\n\nSe algum plugin de \"terceiros\" estiver violando os Termos de Serviço/DMCA de qualquer serviço/entidade legal, por favor, peça ao autor do plugin \"terceiro\" ou à plataforma de hospedagem, por exemplo, GitHub/Codeberg, para tomar medidas. Os plugins listados acima (rotulados como \"terceiros\") são todos plugins públicos/mantidos pela comunidade. Não os estamos curando, então não podemos tomar nenhuma medida sobre eles.\n\n'; + + @override + String get input_does_not_match_format => + 'A entrada não corresponde ao formato exigido'; + + @override + String get plugins => 'Plugins'; + + @override + String get paste_plugin_download_url => + 'Cole a url de download ou a url do repositório GitHub/Codeberg ou o link direto para o arquivo .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Baixar e instalar o plugin a partir da url'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Falha ao adicionar plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'Carregar plugin a partir de arquivo'; + + @override + String get installed => 'Instalado'; + + @override + String get available_plugins => 'Plugins disponíveis'; + + @override + String get configure_plugins => + 'Configure seus próprios plugins de provedores de metadados e fontes de áudio'; + + @override + String get audio_scrobblers => 'Scrobblers de áudio'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Fonte: '; + + @override + String get uncompressed => 'Não comprimido'; + + @override + String get dab_music_source_description => + 'Para audiófilos. Fornece streams de áudio de alta qualidade/sem perdas. Correspondência precisa de faixas baseada em ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_ru.dart b/lib/l10n/generated/app_localizations_ru.dart new file mode 100644 index 00000000..31be6a7b --- /dev/null +++ b/lib/l10n/generated/app_localizations_ru.dart @@ -0,0 +1,1573 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Russian (`ru`). +class AppLocalizationsRu extends AppLocalizations { + AppLocalizationsRu([String locale = 'ru']) : super(locale); + + @override + String get guest => 'Гость'; + + @override + String get browse => 'Обзор'; + + @override + String get search => 'Поиск'; + + @override + String get library => 'Библиотека'; + + @override + String get lyrics => 'Текст'; + + @override + String get settings => 'Настройки'; + + @override + String get genre_categories_filter => 'Фильтр по категориям или жанрам...'; + + @override + String get genre => 'Жанр'; + + @override + String get personalized => 'Персонализированный'; + + @override + String get featured => 'Популярное'; + + @override + String get new_releases => 'Новое'; + + @override + String get songs => 'Треки'; + + @override + String playing_track(Object track) { + return 'Играет $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Это удалит текущую очередь. $track_length треков будет удалено. Вы хотите продолжить?'; + } + + @override + String get load_more => 'Загрузить больше'; + + @override + String get playlists => 'Плейлисты'; + + @override + String get artists => 'Исполнители'; + + @override + String get albums => 'Альбомы'; + + @override + String get tracks => 'Треки'; + + @override + String get downloads => 'Загрузки'; + + @override + String get filter_playlists => 'Применить фильтры к вашим плейлистам...'; + + @override + String get liked_tracks => 'Понравившиеся треки'; + + @override + String get liked_tracks_description => 'Все понравившиеся треки'; + + @override + String get playlist => 'Плейлист'; + + @override + String get create_a_playlist => 'Создать плейлист'; + + @override + String get update_playlist => 'Обновить плейлист'; + + @override + String get create => 'Создать'; + + @override + String get cancel => 'Отмена'; + + @override + String get update => 'Обновить'; + + @override + String get playlist_name => 'Назвать плейлист'; + + @override + String get name_of_playlist => 'Название плейлиста'; + + @override + String get description => 'Описание'; + + @override + String get public => 'Публичный'; + + @override + String get collaborative => 'Совместный'; + + @override + String get search_local_tracks => 'Поиск песен на вашем устройстве...'; + + @override + String get play => 'Играть'; + + @override + String get delete => 'Удалить'; + + @override + String get none => 'Пусто'; + + @override + String get sort_a_z => 'Сортировка по алфавиту'; + + @override + String get sort_z_a => 'Сортировка по алфавиту в обратную сторону'; + + @override + String get sort_artist => 'Сортировать по исполнителю'; + + @override + String get sort_album => 'Сортировать по альбомам'; + + @override + String get sort_duration => 'Сортировать по длительности'; + + @override + String get sort_tracks => 'Сортировать треки'; + + @override + String currently_downloading(Object tracks_length) { + return 'Загружается ($tracks_length)'; + } + + @override + String get cancel_all => 'Отменить все'; + + @override + String get filter_artist => 'Фильтровать по исполнителю...'; + + @override + String followers(Object followers) { + return '$followers Подписчики'; + } + + @override + String get add_artist_to_blacklist => 'Добавить исполнителя в черный список'; + + @override + String get top_tracks => 'Чарт'; + + @override + String get fans_also_like => 'Поклонникам также нравится'; + + @override + String get loading => 'Загрузка...'; + + @override + String get artist => 'Исполнитель'; + + @override + String get blacklisted => 'Внесен в черный список'; + + @override + String get following => 'Подписаны'; + + @override + String get follow => 'Подписаться'; + + @override + String get artist_url_copied => + 'URL-адрес исполнителя скопирован в буфер обмена'; + + @override + String added_to_queue(Object tracks) { + return 'Добавлено $tracks треков в очередь'; + } + + @override + String get filter_albums => 'Фильтровать альбомы...'; + + @override + String get synced => 'Синхронизировано'; + + @override + String get plain => 'Обычный'; + + @override + String get shuffle => 'Перемешать'; + + @override + String get search_tracks => 'Поиск треков...'; + + @override + String get released => 'Дата выхода'; + + @override + String error(Object error) { + return 'Ошибка $error'; + } + + @override + String get title => 'Заголовок'; + + @override + String get time => 'Время'; + + @override + String get more_actions => 'Больше действий'; + + @override + String download_count(Object count) { + return 'Скачать ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Добавить ($count) в плейлист'; + } + + @override + String add_count_to_queue(Object count) { + return 'Добавить ($count) в очередь'; + } + + @override + String play_count_next(Object count) { + return 'Воспроизвести ($count) следующий'; + } + + @override + String get album => 'Альбом'; + + @override + String copied_to_clipboard(Object data) { + return 'Скопировано $data в буфер обмена'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Добавить $track в этот плейлист'; + } + + @override + String get add => 'Добавить'; + + @override + String added_track_to_queue(Object track) { + return 'Добавлен $track в очередь'; + } + + @override + String get add_to_queue => 'Добавить в очередь'; + + @override + String track_will_play_next(Object track) { + return '$track будет воспроизведен следующим'; + } + + @override + String get play_next => 'Воспроизвести следующий'; + + @override + String removed_track_from_queue(Object track) { + return '$track удален из очереди'; + } + + @override + String get remove_from_queue => 'Удалить из очереди'; + + @override + String get remove_from_favorites => 'Удалить из избранного'; + + @override + String get save_as_favorite => 'Сохранить в избранное'; + + @override + String get add_to_playlist => 'Добавить в плейлист'; + + @override + String get remove_from_playlist => 'Удалить из плейлиста'; + + @override + String get add_to_blacklist => 'Добавить в черный список'; + + @override + String get remove_from_blacklist => 'Удалить из черного списка'; + + @override + String get share => 'Поделиться'; + + @override + String get mini_player => 'Мини-плеер'; + + @override + String get slide_to_seek => 'Потяните для перемотки вперед или назад'; + + @override + String get shuffle_playlist => 'Перемешать плейлист'; + + @override + String get unshuffle_playlist => 'Снять перемешивание плейлиста'; + + @override + String get previous_track => 'Предыдущий трек'; + + @override + String get next_track => 'Следующий трек'; + + @override + String get pause_playback => 'Пауза воспроизведения'; + + @override + String get resume_playback => 'Возобновить воспроизведение'; + + @override + String get loop_track => 'Циклический трек'; + + @override + String get no_loop => 'Без повтора'; + + @override + String get repeat_playlist => 'Повторите плейлист'; + + @override + String get queue => 'Очередь'; + + @override + String get alternative_track_sources => 'Альтернативные источники треков'; + + @override + String get download_track => 'Скачать трек'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks треков в очереди'; + } + + @override + String get clear_all => 'Очистить все'; + + @override + String get show_hide_ui_on_hover => 'Показать/Скрыть интерфейс при наведении'; + + @override + String get always_on_top => 'Всегда сверху'; + + @override + String get exit_mini_player => 'Выйти из мини-плеера'; + + @override + String get download_location => 'Место загрузки'; + + @override + String get local_library => 'Локальная библиотека'; + + @override + String get add_library_location => 'Добавить в библиотеку'; + + @override + String get remove_library_location => 'Удалить из библиотеки'; + + @override + String get account => 'Аккаунт'; + + @override + String get logout => 'Выйти'; + + @override + String get logout_of_this_account => 'Выйдите из этого аккаунта'; + + @override + String get language_region => 'Язык и регион'; + + @override + String get language => 'Язык'; + + @override + String get system_default => 'Системное значение по умолчанию'; + + @override + String get market_place_region => 'Региональное пространство'; + + @override + String get recommendation_country => 'Страна рекомендаций'; + + @override + String get appearance => 'Внешний вид'; + + @override + String get layout_mode => 'Режим компоновки'; + + @override + String get override_layout_settings => + 'Изменить настройки режима адаптивной компоновки'; + + @override + String get adaptive => 'Адаптивный'; + + @override + String get compact => 'Компактный'; + + @override + String get extended => 'Расширенный'; + + @override + String get theme => 'Тема'; + + @override + String get dark => 'Тёмная'; + + @override + String get light => 'Светлая'; + + @override + String get system => 'Системная'; + + @override + String get accent_color => 'Акцентный цвет'; + + @override + String get sync_album_color => 'Синхронизировать цвет альбома'; + + @override + String get sync_album_color_description => + 'Использует основной цвет обложки альбома как цвет акцента'; + + @override + String get playback => 'Воспроизведение'; + + @override + String get audio_quality => 'Качество звука'; + + @override + String get high => 'Высокое'; + + @override + String get low => 'Низкое'; + + @override + String get pre_download_play => 'Предварительная загрузка и воспроизведение'; + + @override + String get pre_download_play_description => + 'Вместо потоковой передачи аудио используйте загруженные байты и воспроизводьте их (рекомендуется для пользователей с высокой пропускной способностью)'; + + @override + String get skip_non_music => + 'Пропускать немузыкальные сегменты (SponsorBlock)'; + + @override + String get blacklist_description => 'Черный список треков и артистов'; + + @override + String get wait_for_download_to_finish => + 'Пожалуйста, дождитесь завершения текущей загрузки'; + + @override + String get desktop => 'Компьютер'; + + @override + String get close_behavior => 'Поведение при закрытии'; + + @override + String get close => 'Закрыть'; + + @override + String get minimize_to_tray => 'Свернуть'; + + @override + String get show_tray_icon => 'Показать значок на панели задач'; + + @override + String get about => 'О нас'; + + @override + String get u_love_spotube => 'Мы знаем что вам нравится Spotube'; + + @override + String get check_for_updates => 'Проверьте наличие обновлений'; + + @override + String get about_spotube => 'О Spotube'; + + @override + String get blacklist => 'Чёрный список'; + + @override + String get please_sponsor => 'Стать спосором/поддержать'; + + @override + String get spotube_description => + 'Spotube – это легкий, кросс-платформенный клиент Spotify, предоставляющий бесплатный доступ для всех пользователей'; + + @override + String get version => 'Версия'; + + @override + String get build_number => 'Номер сборки'; + + @override + String get founder => 'Создатель'; + + @override + String get repository => 'Репозиторий'; + + @override + String get bug_issues => 'Ошибки и проблемы'; + + @override + String get made_with => 'Сделано Bangladesh🇧🇩 с ❤️'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Лицензия'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Не беспокойся, никакая личная информация не собирается и не передается'; + + @override + String get know_how_to_login => 'Не знаете, как это сделать?'; + + @override + String get follow_step_by_step_guide => 'Следуйте пошаговому руководству'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => 'Пожалуйста, заполните все поля'; + + @override + String get submit => 'Отправить'; + + @override + String get exit => 'Выйти'; + + @override + String get previous => 'Предыдущий'; + + @override + String get next => 'Следующий'; + + @override + String get done => 'Готово'; + + @override + String get step_1 => 'Шаг 1'; + + @override + String get first_go_to => 'Сначала перейдите в'; + + @override + String get something_went_wrong => 'Что-то пошло не так'; + + @override + String get piped_instance => 'Экземпляр сервера Piped'; + + @override + String get piped_description => + 'Серверный экземпляр Piped для сопоставления треков'; + + @override + String get piped_warning => + 'Некоторые из них могут работать неправильно, поэтому используйте на свой страх и риск'; + + @override + String get invidious_instance => 'Экземпляр сервера Invidious'; + + @override + String get invidious_description => + 'Экземпляр сервера Invidious для сопоставления треков'; + + @override + String get invidious_warning => + 'Некоторые могут работать не очень хорошо. Используйте на свой страх и риск'; + + @override + String get generate => 'Генерировать'; + + @override + String track_exists(Object track) { + return 'Трек $track уже существует'; + } + + @override + String get replace_downloaded_tracks => 'Заменить все ранее скачанные треки'; + + @override + String get skip_download_tracks => + 'Пропустить загрузку всех ранее скачанных треков'; + + @override + String get do_you_want_to_replace => 'Хотите заменить существующий трек??'; + + @override + String get replace => 'Заменить'; + + @override + String get skip => 'Пропустить'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Выберите до $count $type'; + } + + @override + String get select_genres => 'Выберите жанр'; + + @override + String get add_genres => 'Добавить жанр'; + + @override + String get country => 'Страна'; + + @override + String get number_of_tracks_generate => 'Количество треков для создания'; + + @override + String get acousticness => 'Акустичность'; + + @override + String get danceability => 'Ритмичность'; + + @override + String get energy => 'Энергичность'; + + @override + String get instrumentalness => 'Инструментальность'; + + @override + String get liveness => 'Живость'; + + @override + String get loudness => 'Громкость'; + + @override + String get speechiness => 'Речевой характер'; + + @override + String get valence => 'Значимость'; + + @override + String get popularity => 'Популярность'; + + @override + String get key => 'Ключ'; + + @override + String get duration => 'Продолжительность (с)'; + + @override + String get tempo => 'Темп (BPM)'; + + @override + String get mode => 'Режим'; + + @override + String get time_signature => 'Тактовый размер'; + + @override + String get short => 'Короткий'; + + @override + String get medium => 'Средний'; + + @override + String get long => 'Длинный'; + + @override + String get min => 'Минимум'; + + @override + String get max => 'Максимум'; + + @override + String get target => 'Цель'; + + @override + String get moderate => 'Отобрать'; + + @override + String get deselect_all => 'Убрать выделение со всех'; + + @override + String get select_all => 'Выделить все'; + + @override + String get are_you_sure => 'Вы уверены?'; + + @override + String get generating_playlist => 'Создание собственного плейлиста...'; + + @override + String selected_count_tracks(Object count) { + return 'Выбрано $count треков'; + } + + @override + String get download_warning => + 'При скачивании всех треков пакетом вы фактически занимаетесь пиратством и наносите ущерб творческому обществу музыки. Надеюсь, что вы осознаете это. Всегда старайтесь уважать и поддерживать усилия исполнителей, вложенные в их творчество'; + + @override + String get download_ip_ban_warning => + 'Кроме того, стоит учитывать, что из-за чрезмерного количества запросов на скачивание ваш IP-адрес может быть заблокирован на YouTube. Блокировка IP означает, что вы не сможете использовать YouTube (даже если вы вошли в свою учетную запись) в течение, как минимум, 2-3 месяцев с того устройства, с которого были сделаны эти запросы. Важно заметить, что Spotube не несет ответственности за такие события'; + + @override + String get by_clicking_accept_terms => + 'Нажимая \'принять\', вы соглашаетесь с следующими условиями:'; + + @override + String get download_agreement_1 => + 'Я осознаю, что я использую музыку незаконно. Это плохо.'; + + @override + String get download_agreement_2 => + 'Я бы поддержал исполнителей, где только смог, и делаю это, так как не имею средств на приобретение их творчества'; + + @override + String get download_agreement_3 => + 'Я полностью осознаю, что мой IP-адрес может быть заблокирован на YouTube, и я не считаю Spotube или его владельцев/соавторов ответственными за какие-либо неприятности, вызванные моими текущими действиями'; + + @override + String get decline => 'Отклонить'; + + @override + String get accept => 'Принять'; + + @override + String get details => 'Детали'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Канал'; + + @override + String get likes => 'Нравится'; + + @override + String get dislikes => 'Не нравится'; + + @override + String get views => 'Просмотров'; + + @override + String get streamUrl => 'URL-адрес потока'; + + @override + String get stop => 'Остановить'; + + @override + String get sort_newest => 'Сортировать по самым новым добавленным'; + + @override + String get sort_oldest => 'Сортировать по самым старым добавленным'; + + @override + String get sleep_timer => 'Таймер сна'; + + @override + String mins(Object minutes) { + return '$minutes Минут'; + } + + @override + String hours(Object hours) { + return '$hours Часы'; + } + + @override + String hour(Object hours) { + return '$hours Час'; + } + + @override + String get custom_hours => 'Пользовательские часы'; + + @override + String get logs => 'Журналы'; + + @override + String get developers => 'Разработчики'; + + @override + String get not_logged_in => 'Вы не выполнили вход'; + + @override + String get search_mode => 'Режим поиска'; + + @override + String get audio_source => 'Источник аудио'; + + @override + String get ok => 'Ок'; + + @override + String get failed_to_encrypt => 'Не удалось зашифровать'; + + @override + String get encryption_failed_warning => + 'Spotube использует шифрование для безопасного хранения ваших данных. Однако в этом случае произошла ошибка. Поэтому будет использовано небезопасное хранилище.\nЕсли вы используете Linux, убедитесь, что у вас установлен какой-либо инструмент для работы с секретами (gnome-keyring, kde-wallet, keepassxc и т.д.)'; + + @override + String get querying_info => 'Запрос информации...'; + + @override + String get piped_api_down => 'Piped API не отвечает'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Экземпляр Piped $pipedInstance в данный момент недоступен.\n\nВы можете либо изменить экземпляр, либо переключиться на использование официального API YouTube.\n\nНе забудьте перезапустить приложение после внесенных изменений'; + } + + @override + String get you_are_offline => 'Нет доступа к сети'; + + @override + String get connection_restored => 'Ваше интернет-соединение восстановлено'; + + @override + String get use_system_title_bar => 'Использовать системную панель заголовка'; + + @override + String get crunching_results => 'Обработка результатов...'; + + @override + String get search_to_get_results => 'Поиск для получения результатов'; + + @override + String get use_amoled_mode => 'Режим AMOLED'; + + @override + String get pitch_dark_theme => 'Темная тема'; + + @override + String get normalize_audio => 'Нормализовать звук'; + + @override + String get change_cover => 'Изменить обложку'; + + @override + String get add_cover => 'Добавить обложку'; + + @override + String get restore_defaults => 'Восстановить настройки по умолчанию'; + + @override + String get download_music_format => 'Формат загрузки музыки'; + + @override + String get streaming_music_format => 'Формат потоковой музыки'; + + @override + String get download_music_quality => 'Качество загрузки'; + + @override + String get streaming_music_quality => 'Качество стриминга'; + + @override + String get login_with_lastfm => 'Войти с помощью Last.fm'; + + @override + String get connect => 'Подключить'; + + @override + String get disconnect_lastfm => 'Отключиться от Last.fm'; + + @override + String get disconnect => 'Отключить'; + + @override + String get username => 'Имя пользователя'; + + @override + String get password => 'Пароль'; + + @override + String get login => 'Войти'; + + @override + String get login_with_your_lastfm => 'Войти в свою учетную запись Last.fm'; + + @override + String get scrobble_to_lastfm => 'Скробблинг на Last.fm'; + + @override + String get go_to_album => 'Перейти к альбому'; + + @override + String get discord_rich_presence => 'Богатое присутствие в Discord'; + + @override + String get browse_all => 'Просмотреть все'; + + @override + String get genres => 'Жанры'; + + @override + String get explore_genres => 'Исследовать жанры'; + + @override + String get friends => 'Друзья'; + + @override + String get no_lyrics_available => + 'Извините, не удается найти текст для этого трека'; + + @override + String get start_a_radio => 'Запустить радио'; + + @override + String get how_to_start_radio => 'Как вы хотите запустить радио?'; + + @override + String get replace_queue_question => + 'Хотите заменить текущую очередь или добавить к ней?'; + + @override + String get endless_playback => 'Бесконечное воспроизведение'; + + @override + String get delete_playlist => 'Удалить плейлист'; + + @override + String get delete_playlist_confirmation => + 'Вы уверены, что хотите удалить этот плейлист?'; + + @override + String get local_tracks => 'Локальные треки'; + + @override + String get local_tab => 'Локальное'; + + @override + String get song_link => 'Ссылка на песню'; + + @override + String get skip_this_nonsense => 'Пропустить этот бред'; + + @override + String get freedom_of_music => '“Свобода музыки”'; + + @override + String get freedom_of_music_palm => '“Свобода музыки в вашей ладони”'; + + @override + String get get_started => 'Начнем'; + + @override + String get youtube_source_description => + 'Рекомендуется и лучше всего работает.'; + + @override + String get piped_source_description => + 'Чувствуете себя свободно? То же самое, что и YouTube, но намного бесплатно.'; + + @override + String get jiosaavn_source_description => + 'Лучший для Южно-Азиатского региона.'; + + @override + String get invidious_source_description => + 'Похож на Piped, но с более высокой доступностью.'; + + @override + String highest_quality(Object quality) { + return 'Наивысшее качество: $quality'; + } + + @override + String get select_audio_source => 'Выберите аудиоисточник'; + + @override + String get endless_playback_description => + 'Автоматически добавляйте новые песни\nв конец очереди'; + + @override + String get choose_your_region => 'Выберите ваш регион'; + + @override + String get choose_your_region_description => + 'Это поможет Spotube показать вам правильный контент\nдля вашего местоположения.'; + + @override + String get choose_your_language => 'Выберите ваш язык'; + + @override + String get help_project_grow => 'Помогите этому проекту расти'; + + @override + String get help_project_grow_description => + 'Spotube - это проект с открытым исходным кодом. Вы можете помочь этому проекту развиваться, внося вклад в проект, сообщая ошибках или предлагая новые функции.'; + + @override + String get contribute_on_github => 'Внести вклад на GitHub'; + + @override + String get donate_on_open_collective => 'Пожертвовать на Open Collective'; + + @override + String get browse_anonymously => 'Анонимно просматривать'; + + @override + String get enable_connect => 'Включить подключение'; + + @override + String get enable_connect_description => + 'Управление Spotube с других устройств'; + + @override + String get devices => 'Устройства'; + + @override + String get select => 'Выбрать'; + + @override + String connect_client_alert(Object client) { + return 'Вас контролирует $client'; + } + + @override + String get this_device => 'Это устройство'; + + @override + String get remote => 'Дистанционное управление'; + + @override + String get stats => 'Статистика'; + + @override + String and_n_more(Object count) { + return 'и $count еще'; + } + + @override + String get recently_played => 'Недавно воспроизведено'; + + @override + String get browse_more => 'Посмотреть больше'; + + @override + String get no_title => 'Без названия'; + + @override + String get not_playing => 'Не воспроизводится'; + + @override + String get epic_failure => 'Эпическое фиаско!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Добавлено $tracks_length треков в очередь'; + } + + @override + String get spotube_has_an_update => 'В Spotube доступно обновление'; + + @override + String get download_now => 'Скачать сейчас'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum выпущен'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version выпущен'; + } + + @override + String get read_the_latest => 'Читать последние '; + + @override + String get release_notes => 'заметки о версии'; + + @override + String get pick_color_scheme => 'Выберите цветовую схему'; + + @override + String get save => 'Сохранить'; + + @override + String get choose_the_device => 'Выберите устройство:'; + + @override + String get multiple_device_connected => + 'Подключено несколько устройств.\nВыберите устройство, на котором вы хотите выполнить это действие'; + + @override + String get nothing_found => 'Ничего не найдено'; + + @override + String get the_box_is_empty => 'Коробка пуста'; + + @override + String get top_artists => 'Лучшие артисты'; + + @override + String get top_albums => 'Лучшие альбомы'; + + @override + String get this_week => 'На этой неделе'; + + @override + String get this_month => 'В этом месяце'; + + @override + String get last_6_months => 'Последние 6 месяцев'; + + @override + String get this_year => 'В этом году'; + + @override + String get last_2_years => 'Последние 2 года'; + + @override + String get all_time => 'Все время'; + + @override + String powered_by_provider(Object providerName) { + return 'При поддержке $providerName'; + } + + @override + String get email => 'Электронная почта'; + + @override + String get profile_followers => 'Подписчики'; + + @override + String get birthday => 'День рождения'; + + @override + String get subscription => 'Подписка'; + + @override + String get not_born => 'Не рожден'; + + @override + String get hacker => 'Хакер'; + + @override + String get profile => 'Профиль'; + + @override + String get no_name => 'Без имени'; + + @override + String get edit => 'Редактировать'; + + @override + String get user_profile => 'Профиль пользователя'; + + @override + String count_plays(Object count) { + return '$count воспроизведений'; + } + + @override + String get streaming_fees_hypothetical => + '*Рассчитано на основе выплат Spotify за стрим\nот \$0.003 до \$0.005. Это гипотетический\nрасчет, чтобы показать пользователю, сколько бы он\nзаплатил артистам, если бы слушал их песни на Spotify.'; + + @override + String get minutes_listened => 'Минут прослушивания'; + + @override + String get streamed_songs => 'Стримленные песни'; + + @override + String count_streams(Object count) { + return '$count стримов'; + } + + @override + String get owned_by_you => 'Ваша собственность'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl скопировано в буфер обмена'; + } + + @override + String get hipotetical_calculation => + '*Это рассчитано на основе средней выплаты за прослушивание на онлайн-платформах для потоковой передачи музыки в размере от 0,003 до 0,005 долларов США. Это гипотетический расчет, чтобы дать пользователю представление о том, сколько бы они заплатили артистам, если бы слушали их песни на разных музыкальных стриминговых платформах.'; + + @override + String count_mins(Object minutes) { + return '$minutes мин'; + } + + @override + String get summary_minutes => 'минуты'; + + @override + String get summary_listened_to_music => 'Слушанная музыка'; + + @override + String get summary_songs => 'песни'; + + @override + String get summary_streamed_overall => 'Всего стримов'; + + @override + String get summary_owed_to_artists => 'К выплате артистам\nв этом месяце'; + + @override + String get summary_artists => 'артиста'; + + @override + String get summary_music_reached_you => 'Музыка дошла до вас'; + + @override + String get summary_full_albums => 'полные альбомы'; + + @override + String get summary_got_your_love => 'Получил вашу любовь'; + + @override + String get summary_playlists => 'плейлисты'; + + @override + String get summary_were_on_repeat => 'Были на повторе'; + + @override + String total_money(Object money) { + return 'Всего $money'; + } + + @override + String get webview_not_found => 'Webview не найден'; + + @override + String get webview_not_found_description => + 'На вашем устройстве не установлена среда выполнения Webview.\nЕсли он установлен, убедитесь, что он находится в environment PATH\n\nПосле установки перезапустите приложение'; + + @override + String get unsupported_platform => 'Платформа не поддерживается'; + + @override + String get cache_music => 'Кэшировать музыку'; + + @override + String get open => 'Открыть'; + + @override + String get cache_folder => 'Папка кэша'; + + @override + String get export => 'Экспорт'; + + @override + String get clear_cache => 'Очистить кэш'; + + @override + String get clear_cache_confirmation => 'Вы хотите очистить кэш?'; + + @override + String get export_cache_files => 'Экспортировать кэшированные файлы'; + + @override + String found_n_files(Object count) { + return 'Найдено $count файлов'; + } + + @override + String get export_cache_confirmation => + 'Вы хотите экспортировать эти файлы в'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Экспортировано $filesExported из $files файлов'; + } + + @override + String get undo => 'Отменить'; + + @override + String get download_all => 'Скачать все'; + + @override + String get add_all_to_playlist => 'Добавить все в плейлист'; + + @override + String get add_all_to_queue => 'Добавить все в очередь'; + + @override + String get play_all_next => 'Воспроизвести все следующее'; + + @override + String get pause => 'Пауза'; + + @override + String get view_all => 'Просмотреть все'; + + @override + String get no_tracks_added_yet => + 'Похоже, вы ещё не добавили ни одного трека'; + + @override + String get no_tracks => 'Похоже, здесь нет треков'; + + @override + String get no_tracks_listened_yet => 'Похоже, вы ещё ничего не слушали'; + + @override + String get not_following_artists => 'Вы не подписаны на художников'; + + @override + String get no_favorite_albums_yet => + 'Похоже, вы ещё не добавили ни одного альбома в избранное'; + + @override + String get no_logs_found => 'Логи не найдены'; + + @override + String get youtube_engine => 'YouTube Движок'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine не установлен'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine не установлен в вашей системе.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Убедитесь, что он доступен в переменной PATH или\nустановите абсолютный путь к исполнимому файлу $engine ниже'; + } + + @override + String get youtube_engine_unix_issue_message => + 'В macOS/Linux/Unix-подобных ОС, установка пути в .zshrc/.bashrc/.bash_profile и т.д. не будет работать.\nВы должны установить путь в файле конфигурации оболочки'; + + @override + String get download => 'Скачать'; + + @override + String get file_not_found => 'Файл не найден'; + + @override + String get custom => 'Пользовательский'; + + @override + String get add_custom_url => 'Добавить пользовательский URL'; + + @override + String get edit_port => 'Редактировать порт'; + + @override + String get port_helper_msg => + 'По умолчанию -1, что означает случайное число. Если у вас настроен брандмауэр, рекомендуется установить это.'; + + @override + String connect_request(Object client) { + return 'Разрешить $client подключение?'; + } + + @override + String get connection_request_denied => + 'Подключение отклонено. Пользователь отказал в доступе.'; + + @override + String get an_error_occurred => 'Произошла ошибка'; + + @override + String get copy_to_clipboard => 'Скопировать в буфер обмена'; + + @override + String get view_logs => 'Просмотреть журналы'; + + @override + String get retry => 'Повторить'; + + @override + String get no_default_metadata_provider_selected => + 'Вы не выбрали поставщика метаданных по умолчанию'; + + @override + String get manage_metadata_providers => 'Управление поставщиками метаданных'; + + @override + String get open_link_in_browser => 'Открыть ссылку в браузере?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Вы хотите открыть следующую ссылку'; + + @override + String get unsafe_url_warning => + 'Открытие ссылок из ненадежных источников может быть небезопасным. Будьте осторожны!\nВы также можете скопировать ссылку в буфер обмена.'; + + @override + String get copy_link => 'Копировать ссылку'; + + @override + String get building_your_timeline => + 'Создание вашей временной шкалы на основе ваших прослушиваний...'; + + @override + String get official => 'Официальный'; + + @override + String author_name(Object author) { + return 'Автор: $author'; + } + + @override + String get third_party => 'Сторонний'; + + @override + String get plugin_requires_authentication => 'Плагин требует аутентификации'; + + @override + String get update_available => 'Доступно обновление'; + + @override + String get supports_scrobbling => 'Поддерживает скробблинг'; + + @override + String get plugin_scrobbling_info => + 'Этот плагин скробблит вашу музыку для создания вашей истории прослушиваний.'; + + @override + String get default_metadata_source => 'Источник метаданных по умолчанию'; + + @override + String get set_default_metadata_source => + 'Задать источник метаданных по умолчанию'; + + @override + String get default_audio_source => 'Источник аудио по умолчанию'; + + @override + String get set_default_audio_source => 'Задать источник аудио по умолчанию'; + + @override + String get set_default => 'Установить по умолчанию'; + + @override + String get support => 'Поддержка'; + + @override + String get support_plugin_development => 'Поддержать разработку плагина'; + + @override + String can_access_name_api(Object name) { + return '- Может получить доступ к API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Вы хотите установить этот плагин?'; + + @override + String get third_party_plugin_warning => + 'Этот плагин из стороннего репозитория. Пожалуйста, убедитесь, что вы доверяете источнику перед установкой.'; + + @override + String get author => 'Автор'; + + @override + String get this_plugin_can_do_following => + 'Этот плагин может выполнять следующее'; + + @override + String get install => 'Установить'; + + @override + String get install_a_metadata_provider => 'Установить поставщика метаданных'; + + @override + String get no_tracks_playing => + 'В настоящее время не воспроизводится ни один трек'; + + @override + String get synced_lyrics_not_available => + 'Синхронизированные тексты недоступны для этой песни. Пожалуйста, используйте вкладку'; + + @override + String get plain_lyrics => 'Простые тексты'; + + @override + String get tab_instead => 'вместо этого.'; + + @override + String get disclaimer => 'Отказ от ответственности'; + + @override + String get third_party_plugin_dmca_notice => + 'Команда Spotube не несет никакой ответственности (в том числе юридической) за какие-либо \"сторонние\" плагины.\nПожалуйста, используйте их на свой страх и риск. О любых ошибках/проблемах сообщайте в репозиторий плагина.\n\nЕсли какой-либо \"сторонний\" плагин нарушает ToS/DMCA какого-либо сервиса/юридического лица, пожалуйста, попросите автора плагина \"стороннего\" или хостинговую платформу, например, GitHub/Codeberg, принять меры. Перечисленные выше (помеченные как \"сторонние\") являются общедоступными/поддерживаемыми сообществом плагинами. Мы не курируем их, поэтому не можем принимать по ним никаких мер.\n\n'; + + @override + String get input_does_not_match_format => + 'Введенные данные не соответствуют требуемому формату'; + + @override + String get plugins => 'Плагины'; + + @override + String get paste_plugin_download_url => + 'Вставьте URL-адрес для загрузки или URL-адрес репозитория GitHub/Codeberg или прямую ссылку на файл .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Загрузить и установить плагин по URL-адресу'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Не удалось добавить плагин: $error'; + } + + @override + String get upload_plugin_from_file => 'Загрузить плагин из файла'; + + @override + String get installed => 'Установлено'; + + @override + String get available_plugins => 'Доступные плагины'; + + @override + String get configure_plugins => + 'Настройте собственные плагины провайдеров метаданных и источников аудио'; + + @override + String get audio_scrobblers => 'Аудио скробблеры'; + + @override + String get scrobbling => 'Скробблинг'; + + @override + String get source => 'Источник: '; + + @override + String get uncompressed => 'Несжатый'; + + @override + String get dab_music_source_description => + 'Для аудиофилов. Предоставляет высококачественные/lossless аудиопотоки. Точное совпадение треков по ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_ta.dart b/lib/l10n/generated/app_localizations_ta.dart new file mode 100644 index 00000000..062a99dc --- /dev/null +++ b/lib/l10n/generated/app_localizations_ta.dart @@ -0,0 +1,1579 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Tamil (`ta`). +class AppLocalizationsTa extends AppLocalizations { + AppLocalizationsTa([String locale = 'ta']) : super(locale); + + @override + String get guest => 'விருந்தினர்'; + + @override + String get browse => 'உலாவு'; + + @override + String get search => 'தேடுக'; + + @override + String get library => 'நூலகம்'; + + @override + String get lyrics => 'பாடல் வரிகள்'; + + @override + String get settings => 'அமைப்புகள்'; + + @override + String get genre_categories_filter => 'வகைகள் அல்லது பாணிகளை வடிகட்டுக...'; + + @override + String get genre => 'பாணி'; + + @override + String get personalized => 'தனிப்பயனாக்கப்பட்ட'; + + @override + String get featured => 'சிறப்பிடம் பெற்ற'; + + @override + String get new_releases => 'புதிய வெளியீடுகள்'; + + @override + String get songs => 'பாடல்கள்'; + + @override + String playing_track(Object track) { + return '$track இயங்குகிறது'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'இது தற்போதைய வரிசையை அழிக்கும். $track_length பாடல்கள் நீக்கப்படும்\nதொடர விரும்புகிறீர்களா?'; + } + + @override + String get load_more => 'மேலும் ஏற்றுக'; + + @override + String get playlists => 'பாடல் பட்டியல்கள்'; + + @override + String get artists => 'கலைஞர்கள்'; + + @override + String get albums => 'ஆல்பங்கள்'; + + @override + String get tracks => 'பாடல்கள்'; + + @override + String get downloads => 'பதிவிறக்கங்கள்'; + + @override + String get filter_playlists => 'உங்கள் பாடல் பட்டியல்களை வடிகட்டுக...'; + + @override + String get liked_tracks => 'விரும்பிய பாடல்கள்'; + + @override + String get liked_tracks_description => 'உங்கள் விரும்பிய பாடல்கள் அனைத்தும்'; + + @override + String get playlist => 'பாடல் பட்டியல்'; + + @override + String get create_a_playlist => 'பாடல் பட்டியலை உருவாக்குக'; + + @override + String get update_playlist => 'பாடல் பட்டியலைப் புதுப்பிக்க'; + + @override + String get create => 'உருவாக்கு'; + + @override + String get cancel => 'ரத்து செய்'; + + @override + String get update => 'புதுப்பி'; + + @override + String get playlist_name => 'பாடல் பட்டியல் பெயர்'; + + @override + String get name_of_playlist => 'பாடல் பட்டியலின் பெயர்'; + + @override + String get description => 'விளக்கம்'; + + @override + String get public => 'பொது'; + + @override + String get collaborative => 'கூட்டு'; + + @override + String get search_local_tracks => 'உள்ளூர் பாடல்களைத் தேடுக...'; + + @override + String get play => 'இயக்கு'; + + @override + String get delete => 'அழி'; + + @override + String get none => 'எதுவுமில்லை'; + + @override + String get sort_a_z => 'A-Z வரிசைப்படுத்து'; + + @override + String get sort_z_a => 'Z-A வரிசைப்படுத்து'; + + @override + String get sort_artist => 'கலைஞர் மூலம் வரிசைப்படுத்து'; + + @override + String get sort_album => 'ஆல்பம் மூலம் வரிசைப்படுத்து'; + + @override + String get sort_duration => 'கால அளவு மூலம் வரிசைப்படுத்து'; + + @override + String get sort_tracks => 'பாடல்களை வரிசைப்படுத்து'; + + @override + String currently_downloading(Object tracks_length) { + return 'தற்போது பதிவிறக்குகிறது ($tracks_length)'; + } + + @override + String get cancel_all => 'அனைத்தையும் ரத்து செய்'; + + @override + String get filter_artist => 'கலைஞர்களை வடிகட்டுக...'; + + @override + String followers(Object followers) { + return '$followers பின்தொடர்பவர்கள்'; + } + + @override + String get add_artist_to_blacklist => 'கலைஞரை தடைப்பட்டியலில் சேர்க்க'; + + @override + String get top_tracks => 'சிறந்த பாடல்கள்'; + + @override + String get fans_also_like => 'ரசிகர்கள் விரும்புவது'; + + @override + String get loading => 'ஏற்றுகிறது...'; + + @override + String get artist => 'கலைஞர்'; + + @override + String get blacklisted => 'தடைப்பட்டியலில் உள்ளது'; + + @override + String get following => 'பின்தொடர்கிறது'; + + @override + String get follow => 'பின்தொடர்'; + + @override + String get artist_url_copied => + 'கலைஞர் URL கிளிப்போர்டுக்கு நகலெடுக்கப்பட்டது'; + + @override + String added_to_queue(Object tracks) { + return '$tracks பாடல்கள் வரிசையில் சேர்க்கப்பட்டன'; + } + + @override + String get filter_albums => 'ஆல்பங்களை வடிகட்டுக...'; + + @override + String get synced => 'ஒத்திசைக்கப்பட்டது'; + + @override + String get plain => 'சாதாரண'; + + @override + String get shuffle => 'கலக்கு'; + + @override + String get search_tracks => 'பாடல்களைத் தேடுக...'; + + @override + String get released => 'வெளியிடப்பட்டது'; + + @override + String error(Object error) { + return 'பிழை $error'; + } + + @override + String get title => 'தலைப்பு'; + + @override + String get time => 'நேரம்'; + + @override + String get more_actions => 'மேலும் செயல்கள்'; + + @override + String download_count(Object count) { + return 'பதிவிறக்கு ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return '($count) பாடல் பட்டியலில் சேர்'; + } + + @override + String add_count_to_queue(Object count) { + return '($count) வரிசையில் சேர்'; + } + + @override + String play_count_next(Object count) { + return '($count) அடுத்து இயக்கு'; + } + + @override + String get album => 'ஆல்பம்'; + + @override + String copied_to_clipboard(Object data) { + return '$data கிளிப்போர்டுக்கு நகலெடுக்கப்பட்டது'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track பின்வரும் பாடல் பட்டியல்களில் சேர்'; + } + + @override + String get add => 'சேர்'; + + @override + String added_track_to_queue(Object track) { + return '$track வரிசையில் சேர்க்கப்பட்டது'; + } + + @override + String get add_to_queue => 'வரிசையில் சேர்'; + + @override + String track_will_play_next(Object track) { + return '$track அடுத்து இயக்கப்படும்'; + } + + @override + String get play_next => 'அடுத்து இயக்கு'; + + @override + String removed_track_from_queue(Object track) { + return '$track வரிசையிலிருந்து நீக்கப்பட்டது'; + } + + @override + String get remove_from_queue => 'வரிசையிலிருந்து நீக்கு'; + + @override + String get remove_from_favorites => 'பிடித்தவையிலிருந்து நீக்கு'; + + @override + String get save_as_favorite => 'பிடித்தவையாக சேமி'; + + @override + String get add_to_playlist => 'பாடல் பட்டியலில் சேர்'; + + @override + String get remove_from_playlist => 'பாடல் பட்டியலிலிருந்து நீக்கு'; + + @override + String get add_to_blacklist => 'தடைப்பட்டியலில் சேர்'; + + @override + String get remove_from_blacklist => 'தடைப்பட்டியலிலிருந்து நீக்கு'; + + @override + String get share => 'பகிர்'; + + @override + String get mini_player => 'சிறிய இயக்கி'; + + @override + String get slide_to_seek => 'முன்னோக்கி அல்லது பின்னோக்கி செல்ல சறுக்கவும்'; + + @override + String get shuffle_playlist => 'பாடல் பட்டியலை கலக்கு'; + + @override + String get unshuffle_playlist => 'பாடல் பட்டியலை கலக்காதே'; + + @override + String get previous_track => 'முந்தைய பாடல்'; + + @override + String get next_track => 'அடுத்த பாடல்'; + + @override + String get pause_playback => 'இயக்கத்தை நிறுத்து'; + + @override + String get resume_playback => 'இயக்கத்தை தொடர்'; + + @override + String get loop_track => 'பாடலை சுழற்று'; + + @override + String get no_loop => 'சுழற்சி இல்லை'; + + @override + String get repeat_playlist => 'பாடல் பட்டியலை மீண்டும் இயக்கு'; + + @override + String get queue => 'வரிசை'; + + @override + String get alternative_track_sources => 'மாற்று பாடல் மூலங்கள்'; + + @override + String get download_track => 'பாடலைப் பதிவிறக்கு'; + + @override + String tracks_in_queue(Object tracks) { + return 'வரிசையில் $tracks பாடல்கள்'; + } + + @override + String get clear_all => 'அனைத்தையும் அழி'; + + @override + String get show_hide_ui_on_hover => 'மேலே வரும்போது UI ஐக் காட்டு/மறை'; + + @override + String get always_on_top => 'எப்போதும் மேலே'; + + @override + String get exit_mini_player => 'சிறிய இயக்கியிலிருந்து வெளியேறு'; + + @override + String get download_location => 'பதிவிறக்க இடம்'; + + @override + String get local_library => 'உள்ளூர் நூலகம்'; + + @override + String get add_library_location => 'நூலகத்தில் சேர்'; + + @override + String get remove_library_location => 'நூலகத்திலிருந்து நீக்கு'; + + @override + String get account => 'கணக்கு'; + + @override + String get logout => 'வெளியேறு'; + + @override + String get logout_of_this_account => 'இந்த கணக்கிலிருந்து வெளியேறு'; + + @override + String get language_region => 'மொழி & பிராந்தியம்'; + + @override + String get language => 'மொழி'; + + @override + String get system_default => 'கணினி இயல்புநிலை'; + + @override + String get market_place_region => 'சந்தை பிராந்தியம்'; + + @override + String get recommendation_country => 'பரிந்துரை நாடு'; + + @override + String get appearance => 'தோற்றம்'; + + @override + String get layout_mode => 'அமைப்பு முறை'; + + @override + String get override_layout_settings => 'தளவமைப்பு அமைப்புகளை மாற்றியமை'; + + @override + String get adaptive => 'தகவமைப்பு'; + + @override + String get compact => 'சுருக்கமான'; + + @override + String get extended => 'விரிவான'; + + @override + String get theme => 'தீம்'; + + @override + String get dark => 'இருள்'; + + @override + String get light => 'வெளிர்'; + + @override + String get system => 'கணினி வழி'; + + @override + String get accent_color => 'அழுத்த நிறம்'; + + @override + String get sync_album_color => 'ஆல்பம் நிறத்தை ஒத்திசை'; + + @override + String get sync_album_color_description => + 'ஆல்பம் படத்தின் முக்கிய நிறத்தை அழுத்த நிறமாகப் பயன்படுத்துகிறது'; + + @override + String get playback => 'பின்னணி'; + + @override + String get audio_quality => 'ஒலி தரம்'; + + @override + String get high => 'உயர்'; + + @override + String get low => 'குறைந்த'; + + @override + String get pre_download_play => 'முன்பதிவிறக்கம் மற்றும் இயக்கம்'; + + @override + String get pre_download_play_description => + 'ஒலியை ஸ்ட்ரீம் செய்வதற்குப் பதிலாக, பைட்டுகளைப் பதிவிறக்கி இயக்கவும் (அதிக பேண்ட்விட்த் பயனர்களுக்கு பரிந்துரைக்கப்படுகிறது)'; + + @override + String get skip_non_music => 'இசையல்லாத பகுதிகளைத் தவிர் (SponsorBlock)'; + + @override + String get blacklist_description => + 'தடைசெய்யப்பட்ட பாடல்கள் மற்றும் கலைஞர்கள்'; + + @override + String get wait_for_download_to_finish => + 'தற்போதைய பதிவிறக்கம் முடியும் வரை காத்திருக்கவும்'; + + @override + String get desktop => 'கணினி'; + + @override + String get close_behavior => 'மூடும் நடத்தை'; + + @override + String get close => 'மூடு'; + + @override + String get minimize_to_tray => 'ட்ரேயை குறைக்கவும்'; + + @override + String get show_tray_icon => 'ட்ரே ஐகானைக் காட்டு'; + + @override + String get about => 'பற்றி'; + + @override + String get u_love_spotube => + 'நீங்கள் Spotube ஐ நேசிக்கிறீர்கள் என்பது எங்களுக்குத் தெரியும்'; + + @override + String get check_for_updates => 'புதுப்பிப்புகளைச் சரிபார்'; + + @override + String get about_spotube => 'Spotube பற்றி'; + + @override + String get blacklist => 'தடைப்பட்டியல்'; + + @override + String get please_sponsor => 'தயவுசெய்து ஆதரவு/நன்கொடை அளியுங்கள்'; + + @override + String get spotube_description => + 'Spotube, ஒரு லேசான, பல தளங்களில் இயங்கும், அனைவருக்கும் இலவசமான spotify கிளையன்ட்'; + + @override + String get version => 'பதிப்பு'; + + @override + String get build_number => 'கட்டமைப்பு எண்'; + + @override + String get founder => 'நிறுவனர்'; + + @override + String get repository => 'களஞ்சியம்'; + + @override + String get bug_issues => 'பிழை_சிக்கல்கள்'; + + @override + String get made_with => 'வங்காளதேசத்திலிருந்து🇧🇩 ❤️ உருவாக்கப்பட்டது'; + + @override + String get kingkor_roy_tirtho => 'கிங்கர் ராய் திர்தோ'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year கிங்கர் ராய் திர்தோ'; + } + + @override + String get license => 'உரிமம்'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'கவலைப்பட வேண்டாம், உங்கள் சான்றுகள் எதுவும் சேகரிக்கப்படாது அல்லது யாருடனும் பகிரப்படாது'; + + @override + String get know_how_to_login => 'இதை எப்படி செய்வது என்று தெரியவில்லையா?'; + + @override + String get follow_step_by_step_guide => + 'படிப்படியான வழிகாட்டியைப் பின்பற்றவும்'; + + @override + String cookie_name_cookie(Object name) { + return '$name நட்புநிரல்'; + } + + @override + String get fill_in_all_fields => 'அனைத்து களங்களையும் நிரப்பவும்'; + + @override + String get submit => 'சமர்ப்பி'; + + @override + String get exit => 'வெளியேறு'; + + @override + String get previous => 'முந்தைய'; + + @override + String get next => 'அடுத்து'; + + @override + String get done => 'முடிந்தது'; + + @override + String get step_1 => 'முதல் படி'; + + @override + String get first_go_to => 'முதலில், செல்லவேண்டியது'; + + @override + String get something_went_wrong => 'ஏதோ தவறு நடந்துவிட்டது'; + + @override + String get piped_instance => 'Piped சேவையகம் நிகழ்வு'; + + @override + String get piped_description => + 'பாடல் பொருத்தத்திற்குப் பயன்படுத்த வேண்டிய Piped சேவையகம் நிகழ்வு'; + + @override + String get piped_warning => + 'அவற்றில் சில நன்றாக வேலை செய்யாமல் இருக்கலாம். எனவே உங்கள் சொந்த ஆபத்தில் பயன்படுத்தவும்'; + + @override + String get invidious_instance => 'Invidious சேவையக நிகழ்வு'; + + @override + String get invidious_description => + 'பாடல் பொருத்தத்திற்குப் பயன்படுத்த வேண்டிய Invidious சேவையக நிகழ்வு'; + + @override + String get invidious_warning => + 'அவற்றில் சில நன்றாக வேலை செய்யாமல் இருக்கலாம். எனவே உங்கள் சொந்த ஆபத்தில் பயன்படுத்தவும்'; + + @override + String get generate => 'உருவாக்கு'; + + @override + String track_exists(Object track) { + return 'பாடல் $track ஏற்கனவே உள்ளது'; + } + + @override + String get replace_downloaded_tracks => + 'பதிவிறக்கம் செய்யப்பட்ட அனைத்து பாடல்களையும் மாற்றவும்'; + + @override + String get skip_download_tracks => + 'பதிவிறக்கம் செய்யப்பட்ட அனைத்து பாடல்களையும் தவிர்க்கவும்'; + + @override + String get do_you_want_to_replace => + 'ஏற்கனவே உள்ள பாடலை மாற்ற விரும்புகிறீர்களா?'; + + @override + String get replace => 'மாற்று'; + + @override + String get skip => 'தவிர்'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '$count $type வரை தேர்ந்தெடுக்கவும்'; + } + + @override + String get select_genres => 'வகைகளைத் தேர்ந்தெடுக்கவும்'; + + @override + String get add_genres => 'வகைகளைச் சேர்க்கவும்'; + + @override + String get country => 'நாடு'; + + @override + String get number_of_tracks_generate => + 'உருவாக்க வேண்டிய பாடல்களின் எண்ணிக்கை'; + + @override + String get acousticness => 'அகவுஸ்டிக்னெஸ்'; + + @override + String get danceability => 'நடனத்தன்மை'; + + @override + String get energy => 'ஆற்றல்'; + + @override + String get instrumentalness => 'கருவித்தன்மை'; + + @override + String get liveness => 'உயிர்ப்புத்தன்மை'; + + @override + String get loudness => 'ஒலி அளவு'; + + @override + String get speechiness => 'பேச்சுத்தன்மை'; + + @override + String get valence => 'உணர்வு'; + + @override + String get popularity => 'பிரபலம்'; + + @override + String get key => 'இசை குறிப்பு'; + + @override + String get duration => 'கால அளவு (வினாடிகள்)'; + + @override + String get tempo => 'வேகம் (BPM)'; + + @override + String get mode => 'முறை'; + + @override + String get time_signature => 'நேர கையொப்பம்'; + + @override + String get short => 'குறுகிய'; + + @override + String get medium => 'நடுத்தர'; + + @override + String get long => 'நீண்ட'; + + @override + String get min => 'குறைந்தபட்சம்'; + + @override + String get max => 'அதிகபட்சம்'; + + @override + String get target => 'இலக்கு'; + + @override + String get moderate => 'மிதமான'; + + @override + String get deselect_all => 'அனைத்தையும் தேர்வுநீக்கு'; + + @override + String get select_all => 'அனைத்தையும் தேர்ந்தெடு'; + + @override + String get are_you_sure => 'உறுதியாக இருக்கிறீர்களா?'; + + @override + String get generating_playlist => + 'உங்கள் தனிப்பயன்பாட்டிற்கான பாடல் பட்டியலை உருவாக்குகிறது...'; + + @override + String selected_count_tracks(Object count) { + return '$count பாடல்கள் தேர்ந்தெடுக்கப்பட்டன'; + } + + @override + String get download_warning => + 'நீங்கள் அனைத்து பாடல்களையும் மொத்தமாக பதிவிறக்கினால், நீங்கள் தெளிவாக இசையைத் திருடுகிறீர்கள் மற்றும் இசையின் படைப்பாற்றல் சமூகத்திற்கு சேதம் விளைவிக்கிறீர்கள். நீங்கள் இதை அறிந்திருக்கிறீர்கள் என்று நம்புகிறேன். எப்போதும், கலைஞரின் கடின உழைப்பை மதித்து ஆதரிக்க முயற்சி செய்யுங்கள்'; + + @override + String get download_ip_ban_warning => + 'மேலும், அதிகப்படியான பதிவிறக்க கோரிக்கைகள் காரணமாக உங்கள் IP YouTube இல் தடைசெய்யப்படலாம். IP தடை என்பது குறைந்தது 2-3 மாதங்களுக்கு அந்த IP சாதனத்திலிருந்து YouTube ஐப் பயன்படுத்த முடியாது (நீங்கள் உள்நுழைந்திருந்தாலும் கூட). இது ஒருபோதும் நடந்தால் Spotube பொறுப்பேற்காது'; + + @override + String get by_clicking_accept_terms => + '\'ஏற்றுக்கொள்\' என்பதைக் கிளிக் செய்வதன் மூலம் பின்வரும் விதிமுறைகளுக்கு நீங்கள் ஒப்புக்கொள்கிறீர்கள்:'; + + @override + String get download_agreement_1 => + 'நான் இசையைத் திருடுகிறேன் என்பது எனக்குத் தெரியும். நான் கெட்டவன்'; + + @override + String get download_agreement_2 => + 'நான் கலைஞரை முடிந்தவரை ஆதரிப்பேன், அவர்களின் கலைக்கு பணம் செலுத்த எனக்கு பணம் இல்லாததால் மட்டுமே இதைச் செய்கிறேன்'; + + @override + String get download_agreement_3 => + 'என் IP YouTube இல் தடைசெய்யப்படலாம் என்பதை நான் முழுமையாக அறிவேன், மேலும் என் தற்போதைய செயலால் ஏற்படும் எந்த விபத்துகளுக்கும் Spotube அல்லது அதன் உரிமையாளர்கள்/பங்களிப்பாளர்களை பொறுப்பாக்க மாட்டேன்'; + + @override + String get decline => 'மறு'; + + @override + String get accept => 'ஏற்றுக்கொள்'; + + @override + String get details => 'விவரங்கள்'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'சேனல்'; + + @override + String get likes => 'விருப்பங்கள்'; + + @override + String get dislikes => 'விருப்பமில்லாதவை'; + + @override + String get views => 'பார்வைகள்'; + + @override + String get streamUrl => 'ஸ்ட்ரீம் URL'; + + @override + String get stop => 'நிறுத்து'; + + @override + String get sort_newest => 'புதிதாக சேர்க்கப்பட்டவற்றை வரிசைப்படுத்து'; + + @override + String get sort_oldest => 'பழமையானவற்றை வரிசைப்படுத்து'; + + @override + String get sleep_timer => 'உறக்க நேரம்'; + + @override + String mins(Object minutes) { + return '$minutes நிமிடங்கள்'; + } + + @override + String hours(Object hours) { + return '$hours மணிநேரங்கள்'; + } + + @override + String hour(Object hours) { + return '$hours மணிநேரம்'; + } + + @override + String get custom_hours => 'தனிப்பயன் மணிநேரங்கள்'; + + @override + String get logs => 'பதிவுகள்'; + + @override + String get developers => 'உருவாக்குநர்கள்'; + + @override + String get not_logged_in => 'நீங்கள் உள்நுழையவில்லை'; + + @override + String get search_mode => 'தேடல் முறை'; + + @override + String get audio_source => 'ஒலி மூலம்'; + + @override + String get ok => 'சரி'; + + @override + String get failed_to_encrypt => 'குறியாக்கம் தோல்வியடைந்தது'; + + @override + String get encryption_failed_warning => + 'Spotube உங்கள் தரவை பாதுகாப்பாக சேமிக்க குறியாக்கத்தைப் பயன்படுத்துகிறது. ஆனால் அவ்வாறு செய்ய முடியவில்லை. எனவே இது பாதுகாப்பற்ற சேமிப்பகத்திற்கு மாறும்\nநீங்கள் லினக்ஸ் பயன்படுத்துகிறீர்கள் என்றால், எந்த ரகசிய சேவையும் (gnome-keyring, kde-wallet, keepassxc போன்றவை) நிறுவப்பட்டுள்ளதா என்பதை உறுதிப்படுத்தவும்'; + + @override + String get querying_info => 'தகவலைக் கேட்கிறது...'; + + @override + String get piped_api_down => 'Piped API செயலிழந்துள்ளது'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Piped நிகழ்வு $pipedInstance தற்போது செயலிழந்துள்ளது\n\nநிகழ்வை மாற்றவும் அல்லது \'API வகை\'யை அதிகாரப்பூர்வ YouTube API க்கு மாற்றவும்\n\nமாற்றத்திற்குப் பிறகு பயன்பாட்டை மறுதொடக்கம் செய்வதை உறுதிப்படுத்தவும்'; + } + + @override + String get you_are_offline => 'நீங்கள் தற்போது ஆஃப்லைனில் உள்ளீர்கள்'; + + @override + String get connection_restored => 'உங்கள் இணைய இணைப்பு மீட்டெடுக்கப்பட்டது'; + + @override + String get use_system_title_bar => 'கணினி தலைப்புப் பட்டியைப் பயன்படுத்தவும்'; + + @override + String get crunching_results => 'முடிவுகளை செயலாக்குகிறது...'; + + @override + String get search_to_get_results => 'முடிவுகளைப் பெற தேடவும்'; + + @override + String get use_amoled_mode => 'கருமை நிற இருண்ட தீம்'; + + @override + String get pitch_dark_theme => 'AMOLED முறை'; + + @override + String get normalize_audio => 'ஒலியை சீரமை'; + + @override + String get change_cover => 'அட்டையை மாற்று'; + + @override + String get add_cover => 'அட்டையைச் சேர்'; + + @override + String get restore_defaults => 'இயல்புநிலைகளை மீட்டமை'; + + @override + String get download_music_format => 'இசை பதிவிறக்க வடிவம்'; + + @override + String get streaming_music_format => 'இசை ஸ்ட்ரீமிங் வடிவம்'; + + @override + String get download_music_quality => 'பதிவிறக்க தரம்'; + + @override + String get streaming_music_quality => 'ஸ்ட்ரீமிங் தரம்'; + + @override + String get login_with_lastfm => 'Last.fm உடன் உள்நுழைக'; + + @override + String get connect => 'இணை'; + + @override + String get disconnect_lastfm => 'Last.fm இலிருந்து துண்டி'; + + @override + String get disconnect => 'துண்டி'; + + @override + String get username => 'பயனர்பெயர்'; + + @override + String get password => 'கடவுச்சொல்'; + + @override + String get login => 'உள்நுழைக'; + + @override + String get login_with_your_lastfm => 'உங்கள் Last.fm கணக்குடன் உள்நுழைக'; + + @override + String get scrobble_to_lastfm => 'Last.fm க்கு ஸ்க்ரோபிள் செய்'; + + @override + String get go_to_album => 'ஆல்பத்திற்குச் செல்'; + + @override + String get discord_rich_presence => 'Discord செழுமையான தோற்றம்'; + + @override + String get browse_all => 'அனைத்தையும் உலாவு'; + + @override + String get genres => 'வகைகள்'; + + @override + String get explore_genres => 'வகைகளை ஆராயுங்கள்'; + + @override + String get friends => 'நண்பர்கள்'; + + @override + String get no_lyrics_available => + 'மன்னிக்கவும், இந்தப் பாடலுக்கான பாடல் வரிகளைக் கண்டுபிடிக்க முடியவில்லை'; + + @override + String get start_a_radio => 'வானொலியைத் தொடங்கு'; + + @override + String get how_to_start_radio => 'வானொலியை எவ்வாறு தொடங்க விரும்புகிறீர்கள்?'; + + @override + String get replace_queue_question => + 'தற்போதைய வரிசையை மாற்ற விரும்புகிறீர்களா அல்லது அதனுடன் சேர்க்க விரும்புகிறீர்களா?'; + + @override + String get endless_playback => 'முடிவற்ற இயக்கம்'; + + @override + String get delete_playlist => 'பாடல் பட்டியலை நீக்கு'; + + @override + String get delete_playlist_confirmation => + 'இந்த பாடல் பட்டியலை நீக்க விரும்புகிறீர்களா?'; + + @override + String get local_tracks => 'உள்ளூர் பாடல்கள்'; + + @override + String get local_tab => 'உள்ளூர்'; + + @override + String get song_link => 'பாடல் இணைப்பு'; + + @override + String get skip_this_nonsense => 'இந்த அர்த்தமற்றதைத் தவிர்'; + + @override + String get freedom_of_music => '\"இசையின் சுதந்திரம்\"'; + + @override + String get freedom_of_music_palm => '\"உங்கள் கைகளில் இசையின் சுதந்திரம்\"'; + + @override + String get get_started => 'தொடங்குவோம்'; + + @override + String get youtube_source_description => + 'பரிந்துரைக்கப்படுகிறது மற்றும் சிறப்பாக செயல்படுகிறது.'; + + @override + String get piped_source_description => + 'சுதந்திரமாக உணர்கிறீர்களா? YouTube போலவே ஆனால் மிகவும் சுதந்திரமானது.'; + + @override + String get jiosaavn_source_description => + 'தெற்காசியப் பிராந்தியத்திற்கு சிறந்தது.'; + + @override + String get invidious_source_description => + 'Piped ஐப் போன்றது ஆனால் அதிக கிடைக்கும் தன்மையுடன்.'; + + @override + String highest_quality(Object quality) { + return 'உயர்ந்த தரம்: $quality'; + } + + @override + String get select_audio_source => 'ஒலி மூலத்தைத் தேர்ந்தெடுக்கவும்'; + + @override + String get endless_playback_description => + 'வரிசையின் இறுதியில் புதிய பாடல்களை\nதானாகவே சேர்க்கவும்'; + + @override + String get choose_your_region => 'உங்கள் பிராந்தியத்தைத் தேர்ந்தெடுக்கவும்'; + + @override + String get choose_your_region_description => + 'இது உங்கள் இருப்பிடத்திற்கான சரியான உள்ளடக்கத்தை\nSpotube காட்ட உதவும்.'; + + @override + String get choose_your_language => 'உங்கள் மொழியைத் தேர்ந்தெடுக்கவும்'; + + @override + String get help_project_grow => 'இந்த திட்டம் வளர உதவுங்கள்'; + + @override + String get help_project_grow_description => + 'Spotube ஒரு திறந்த மூல திட்டம். திட்டத்திற்கு பங்களிப்பு செய்வதன் மூலம், பிழைகளைப் புகாரளிப்பதன் மூலம் அல்லது புதிய அம்சங்களைப் பரிந்துரைப்பதன் மூலம் இந்தத் திட்டம் வளர உதவலாம்.'; + + @override + String get contribute_on_github => 'GitHub இல் பங்களியுங்கள்'; + + @override + String get donate_on_open_collective => + 'Open Collective இல் நன்கொடை அளியுங்கள்'; + + @override + String get browse_anonymously => 'அநாமதேயமாக உலாவுக'; + + @override + String get enable_connect => 'இணைப்பை இயக்கு'; + + @override + String get enable_connect_description => + 'மற்ற சாதனங்களிலிருந்து Spotube ஐக் கட்டுப்படுத்தவும்'; + + @override + String get devices => 'சாதனங்கள்'; + + @override + String get select => 'தேர்ந்தெடு'; + + @override + String connect_client_alert(Object client) { + return 'நீங்கள் $client ஆல் கட்டுப்படுத்தப்படுகிறீர்கள்'; + } + + @override + String get this_device => 'இந்த சாதனம்'; + + @override + String get remote => 'தொலைநிலை'; + + @override + String get stats => 'புள்ளிவிவரங்கள்'; + + @override + String and_n_more(Object count) { + return 'மற்றும் $count கூடுதலாக'; + } + + @override + String get recently_played => 'சமீபத்தில் இயக்கியவை'; + + @override + String get browse_more => 'மேலும் உலாவு'; + + @override + String get no_title => 'தலைப்பு இல்லை'; + + @override + String get not_playing => 'இயக்கப்படவில்லை'; + + @override + String get epic_failure => 'மோசமான தோல்வி!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length பாடல்கள் வரிசையில் சேர்க்கப்பட்டன'; + } + + @override + String get spotube_has_an_update => 'Spotube க்கு ஒரு புதுப்பிப்பு உள்ளது'; + + @override + String get download_now => 'இப்போது பதிவிறக்கு'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum வெளியிடப்பட்டுள்ளது'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version வெளியிடப்பட்டுள்ளது'; + } + + @override + String get read_the_latest => 'சமீபத்திய '; + + @override + String get release_notes => 'வெளியீட்டு குறிப்புகளைப் படிக்கவும்'; + + @override + String get pick_color_scheme => 'வண்ணத் திட்டத்தைத் தேர்ந்தெடுக்கவும்'; + + @override + String get save => 'சேமி'; + + @override + String get choose_the_device => 'சாதனத்தைத் தேர்ந்தெடுக்கவும்:'; + + @override + String get multiple_device_connected => + 'பல சாதனங்கள் இணைக்கப்பட்டுள்ளன.\nஇந்த செயல் நடைபெற வேண்டிய சாதனத்தைத் தேர்ந்தெடுக்கவும்'; + + @override + String get nothing_found => 'எதுவும் கிடைக்கவில்லை'; + + @override + String get the_box_is_empty => 'பெட்டி காலியாக உள்ளது'; + + @override + String get top_artists => 'சிறந்த கலைஞர்கள்'; + + @override + String get top_albums => 'சிறந்த ஆல்பங்கள்'; + + @override + String get this_week => 'இந்த வாரம்'; + + @override + String get this_month => 'இந்த மாதம்'; + + @override + String get last_6_months => 'கடந்த 6 மாதங்கள்'; + + @override + String get this_year => 'இந்த ஆண்டு'; + + @override + String get last_2_years => 'கடந்த 2 ஆண்டுகள்'; + + @override + String get all_time => 'எல்லா நேரமும்'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName ஆல் இயக்கப்படுகிறது'; + } + + @override + String get email => 'மின்னஞ்சல்'; + + @override + String get profile_followers => 'பின்தொடர்பவர்கள்'; + + @override + String get birthday => 'பிறந்த நாள்'; + + @override + String get subscription => 'சந்தா'; + + @override + String get not_born => 'பிறக்கவில்லை'; + + @override + String get hacker => 'ஹேக்கர்'; + + @override + String get profile => 'சுயவிவரம்'; + + @override + String get no_name => 'பெயர் இல்லை'; + + @override + String get edit => 'திருத்து'; + + @override + String get user_profile => 'பயனர் சுயவிவரம்'; + + @override + String count_plays(Object count) { + return '$count முறை இசைக்கப்பட்டது'; + } + + @override + String get streaming_fees_hypothetical => 'ஸ்ட்ரீமிங் கட்டணங்கள் (கற்பனை)'; + + @override + String get minutes_listened => 'காலம் கேட்டது'; + + @override + String get streamed_songs => 'ஸ்ட்ரீமிங் செய்யப்பட்ட பாடல்கள்'; + + @override + String count_streams(Object count) { + return '$count ஸ்ட்ரீம்கள்'; + } + + @override + String get owned_by_you => 'உங்களால் கொண்டது'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'நகலெடுக்கப்பட்டது $shareUrl கிளிப்போர்டுக்காக'; + } + + @override + String get hipotetical_calculation => + '*இது சராசரி ஆன்லைன் இசை ஸ்ட்ரீமிங் தளத்தின் ஒரு ஸ்ட்ரீமிற்கான \$0.003 முதல் \$0.005 வரையிலான கட்டணத்தின் அடிப்படையில் கணக்கிடப்படுகிறது. இது ஒரு கற்பனையான கணக்கீடு ஆகும், இது பயனர்கள் வெவ்வேறு இசை ஸ்ட்ரீமிங் தளங்களில் தங்கள் பாடல்களைக் கேட்டால் கலைஞர்களுக்கு எவ்வளவு பணம் செலுத்தியிருப்பார்கள் என்பது குறித்த நுண்ணறிவை வழங்குகிறது.'; + + @override + String count_mins(Object minutes) { + return '$minutes நிமிடங்கள்'; + } + + @override + String get summary_minutes => 'நிமிடங்கள்'; + + @override + String get summary_listened_to_music => 'இசை கேட்டது'; + + @override + String get summary_songs => 'பாடல்கள்'; + + @override + String get summary_streamed_overall => 'மொத்தமாக ஸ்ட்ரீமிங்'; + + @override + String get summary_owed_to_artists => 'கலைஞர்களுக்கு\nஇந்த மாதம் சொந்தமானது'; + + @override + String get summary_artists => 'கலைஞர்கள்'; + + @override + String get summary_music_reached_you => 'இசை உங்களுக்கு வந்தது'; + + @override + String get summary_full_albums => 'முழு ஆல்பங்கள்'; + + @override + String get summary_got_your_love => 'உங்கள் அன்பை பெற்றுக்கொண்டேன்'; + + @override + String get summary_playlists => 'பாடல் பட்டியல்கள்'; + + @override + String get summary_were_on_repeat => 'மீண்டும் மீண்டும் இருந்தன'; + + @override + String total_money(Object money) { + return 'மொத்தம் $money'; + } + + @override + String get webview_not_found => 'வெப்வியூ கிடைக்கவில்லை'; + + @override + String get webview_not_found_description => + 'உங்கள் சாதனத்தில் எந்தவொரு வெப்வியூ இயக்கத்தை நிறுவவில்லை.\nஇது நிறுவப்பட்டிருந்தால், சுற்றுச்சூழல் பாதையில் PATH உள்ளது என்பதை உறுதிபடுத்தவும்\n\nநிறுவித்த பிறகு, செயலியை மறுதொடக்கம் செய்யவும்'; + + @override + String get unsupported_platform => 'அதிர்ஷ்டகாத உருப்படியை ஆதரிக்கவில்லை'; + + @override + String get cache_music => 'இசையை கேஷ் செய்'; + + @override + String get open => 'திறக்கவும்'; + + @override + String get cache_folder => 'கேஷ் அடைவு'; + + @override + String get export => 'ஏற்றுமதி'; + + @override + String get clear_cache => 'கேஷ் அழிக்கவும்'; + + @override + String get clear_cache_confirmation => 'கேஷைப் அழிக்க விரும்புகிறீர்களா?'; + + @override + String get export_cache_files => 'கேஷில் உள்ள கோப்புகளை ஏற்றுமதி செய்யவும்'; + + @override + String found_n_files(Object count) { + return '$count கோப்புகள் கிடைத்தன'; + } + + @override + String get export_cache_confirmation => + 'இந்த கோப்புகளை ஏற்றுமதி செய்ய விரும்புகிறீர்களா?'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported கோப்புகள் ஏற்றுமதி செய்யப்பட்டன, $files கோப்புகளில்'; + } + + @override + String get undo => 'செயல்தவிர்'; + + @override + String get download_all => 'அனைத்தையும் பதிவிறக்குக'; + + @override + String get add_all_to_playlist => 'அனைத்தையும் பாடல் பட்டியலில் சேர்க்கவும்'; + + @override + String get add_all_to_queue => 'அனைத்தையும் வரிசைப்படுத்து'; + + @override + String get play_all_next => 'அடுத்த உள்ள அனைத்தையும் இயக்கு'; + + @override + String get pause => 'நிறுத்து'; + + @override + String get view_all => 'அனைத்தையும் காண்க'; + + @override + String get no_tracks_added_yet => + 'உங்கள் பாடல்களை இன்னும் சேர்க்கவில்லை என்றால் தெரியாதே'; + + @override + String get no_tracks => 'இங்கு பாடல்கள் எதுவும் இல்லை'; + + @override + String get no_tracks_listened_yet => 'இன்னும் எதையும் கேள்வியில்லை'; + + @override + String get not_following_artists => 'நீங்கள் எந்த கலைஞரையும் பின்தொடரவில்லை'; + + @override + String get no_favorite_albums_yet => + 'நீங்கள் இன்னும் எந்த ஆல்பங்களையும் பிடித்தவையாகச் சேர்க்கவில்லை'; + + @override + String get no_logs_found => 'பதிவுகள் எதுவும் கிடைக்கவில்லை'; + + @override + String get youtube_engine => 'YouTube இயந்திரம்'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine நிறுவியதில்லை'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine உங்கள் கணினியில் நிறுவியதில்லை.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'PATH மாறியில் கிடைக்கிறதா என்பதை உறுதிப்படுத்தவும் அல்லது\n$engine செயல் செய்யக்கூடிய முறையை கீழே அமைக்கவும்'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/unix போல் OS இல், .zshrc/.bashrc/.bash_profile போன்றவை அமைப்பில் பாதையை PATH அமைப்பது இயலாது.\nநீங்கள்.shell configuration file இல் பாதையை அமைக்க வேண்டும்'; + + @override + String get download => 'பதிவிறக்கு'; + + @override + String get file_not_found => 'கோப்பு கிடைக்கவில்லை'; + + @override + String get custom => 'தனிப்பயன்'; + + @override + String get add_custom_url => 'தனிப்பயன் URL ஐச் சேர்க்கவும்'; + + @override + String get edit_port => 'போர்டு திருத்தவும்'; + + @override + String get port_helper_msg => + 'இயல்புநிலை -1 ஆகும், இது சீரற்ற எண்ணை குறிக்கிறது. நீங்கள் தீயணைப்பு அமைக்கப்பட்டிருந்தால், இதை அமைப்பது பரிந்துரைக்கப்படுகிறது.'; + + @override + String connect_request(Object client) { + return '$client க்கு இணைக்க அனுமதிக்கவா?'; + } + + @override + String get connection_request_denied => + 'இணைப்பு மறுக்கப்பட்டது. பயனர் அணுகலை மறுத்தார்.'; + + @override + String get an_error_occurred => 'ஒரு பிழை ஏற்பட்டது'; + + @override + String get copy_to_clipboard => 'கிளிப்போர்டுக்கு நகலெடுக்கவும்'; + + @override + String get view_logs => 'பதிவுகளைப் பார்க்கவும்'; + + @override + String get retry => 'மீண்டும் முயற்சிக்கவும்'; + + @override + String get no_default_metadata_provider_selected => + 'நீங்கள் எந்த இயல்புநிலை மெட்டாடேட்டா வழங்குநரையும் அமைக்கவில்லை'; + + @override + String get manage_metadata_providers => + 'மெட்டாடேட்டா வழங்குநர்களை நிர்வகிக்கவும்'; + + @override + String get open_link_in_browser => 'இணைப்பை உலாவியில் திறக்கவா?'; + + @override + String get do_you_want_to_open_the_following_link => + 'பின்வரும் இணைப்பை நீங்கள் திறக்க விரும்புகிறீர்களா'; + + @override + String get unsafe_url_warning => + 'நம்பத்தகாத மூலங்களிலிருந்து இணைப்புகளைத் திறப்பது பாதுகாப்பற்றதாக இருக்கலாம். எச்சரிக்கையாக இருங்கள்!\nநீங்கள் இணைப்பை உங்கள் கிளிப்போர்டுக்கு நகலெடுக்கலாம்.'; + + @override + String get copy_link => 'இணைப்பை நகலெடுக்கவும்'; + + @override + String get building_your_timeline => + 'உங்கள் கேட்டலின் அடிப்படையில் உங்கள் காலவரிசையை உருவாக்குகிறது...'; + + @override + String get official => 'அதிகாரபூர்வமானது'; + + @override + String author_name(Object author) { + return 'ஆசிரியர்: $author'; + } + + @override + String get third_party => 'மூன்றாம் தரப்பு'; + + @override + String get plugin_requires_authentication => + 'பிளகின் அங்கீகாரத்தைக் கோருகிறது'; + + @override + String get update_available => 'புதுப்பிப்பு உள்ளது'; + + @override + String get supports_scrobbling => 'ஸ்க்ரோப்ளிங்கை ஆதரிக்கிறது'; + + @override + String get plugin_scrobbling_info => + 'இந்த பிளகின் உங்கள் கேட்பதின் வரலாற்றை உருவாக்க உங்கள் இசையை ஸ்க்ரோப்ள் செய்கிறது.'; + + @override + String get default_metadata_source => 'இயல்புநிலை மெட்டாடேட்டா மூலம்'; + + @override + String get set_default_metadata_source => + 'இயல்புநிலை மெட்டாடேட்டா மூலத்தை அமை'; + + @override + String get default_audio_source => 'இயல்புநிலை ஆடியோ மூலம்'; + + @override + String get set_default_audio_source => 'இயல்புநிலை ஆடியோ மூலத்தை அமை'; + + @override + String get set_default => 'இயல்புநிலையாக அமைக்கவும்'; + + @override + String get support => 'ஆதரவு'; + + @override + String get support_plugin_development => 'பிளகின் வளர்ச்சிக்கு ஆதரவு'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API ஐ அணுக முடியும்'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'இந்த பிளகினை நீங்கள் நிறுவ விரும்புகிறீர்களா?'; + + @override + String get third_party_plugin_warning => + 'இந்த பிளகின் மூன்றாம் தரப்பு களஞ்சியத்திலிருந்து வருகிறது. நிறுவும் முன் மூலத்தை நீங்கள் நம்புகிறீர்கள் என்பதை உறுதிப்படுத்தவும்.'; + + @override + String get author => 'ஆசிரியர்'; + + @override + String get this_plugin_can_do_following => + 'இந்த பிளகின் பின்வருவனவற்றைச் செய்ய முடியும்'; + + @override + String get install => 'நிறுவவும்'; + + @override + String get install_a_metadata_provider => 'மெட்டாடேட்டா வழங்குநரை நிறுவவும்'; + + @override + String get no_tracks_playing => 'தற்போது எந்த பாடலும் இயங்கவில்லை'; + + @override + String get synced_lyrics_not_available => + 'இந்த பாடலுக்கு ஒத்திசைக்கப்பட்ட வரிகள் கிடைக்கவில்லை. தயவுசெய்து'; + + @override + String get plain_lyrics => 'சாதாரண வரிகள்'; + + @override + String get tab_instead => 'தாவலை அதற்கு பதிலாக பயன்படுத்தவும்.'; + + @override + String get disclaimer => 'துறப்பு'; + + @override + String get third_party_plugin_dmca_notice => + 'ஸ்பாட்யூப் குழு எந்த \"மூன்றாம் தரப்பு\" பிளகின்களுக்கும் எந்தப் பொறுப்பையும் (சட்டரீதியான உட்பட) ஏற்காது.\nதயவுசெய்து உங்கள் சொந்த ஆபத்தில் அவற்றைப் பயன்படுத்தவும். ஏதேனும் பிழைகள்/சிக்கல்களுக்கு, பிளகின் களஞ்சியத்தில் அவற்றைப் புகாரளிக்கவும்.\n\nஏதேனும் ஒரு \"மூன்றாம் தரப்பு\" பிளகின் ஒரு சேவை/சட்ட நிறுவனத்தின் ToS/DMCA ஐ மீறினால், தயவுசெய்து \"மூன்றாம் தரப்பு\" பிளகின் ஆசிரியரையோ அல்லது ஹோஸ்டிங் தளத்தையோ, எ.கா. GitHub/Codeberg, நடவடிக்கை எடுக்கக் கோரவும். மேலே பட்டியலிடப்பட்ட (\"மூன்றாம் தரப்பு\" என பெயரிடப்பட்ட) அனைத்து பொதுவான/சமூகத்தால் பராமரிக்கப்படும் பிளகின்கள். நாங்கள் அவற்றை க்யூரேட் செய்யவில்லை, எனவே அவற்றின் மீது எந்த நடவடிக்கையும் எடுக்க முடியாது.\n\n'; + + @override + String get input_does_not_match_format => + 'உள்ளீடு தேவையான வடிவத்துடன் பொருந்தவில்லை'; + + @override + String get plugins => 'செருகுநிரல்கள்'; + + @override + String get paste_plugin_download_url => + 'பதிவிறக்க url அல்லது GitHub/Codeberg repo url அல்லது .smplug கோப்பிற்கான நேரடி இணைப்பை ஒட்டவும்'; + + @override + String get download_and_install_plugin_from_url => + 'url இலிருந்து பிளகினைப் பதிவிறக்கி நிறுவவும்'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'பிளகினைச் சேர்க்கத் தவறிவிட்டது: $error'; + } + + @override + String get upload_plugin_from_file => 'கோப்பிலிருந்து பிளகினைப் பதிவேற்றவும்'; + + @override + String get installed => 'நிறுவப்பட்டது'; + + @override + String get available_plugins => 'கிடைக்கக்கூடிய பிளகின்கள்'; + + @override + String get configure_plugins => + 'உங்கள் சொந்த மெட்டாடேட்டா வழங்குநர் மற்றும் ஆடியோ மூல செருகுநிரல்களை அமைக்கவும்'; + + @override + String get audio_scrobblers => 'ஆடியோ ஸ்க்ரோப்ளர்கள்'; + + @override + String get scrobbling => 'ஸ்க்ரோப்ளிங்'; + + @override + String get source => 'மூலம்: '; + + @override + String get uncompressed => 'அழுத்தப்படாத'; + + @override + String get dab_music_source_description => + 'ஆடியோஃபைல்களுக்காக. உயர்தர/லாஸ்லெஸ் ஆடியோ ஸ்ட்ரீம்களை வழங்குகிறது. ISRC அடிப்படையில் துல்லியமான பாடல் பொருத்தம்.'; +} diff --git a/lib/l10n/generated/app_localizations_th.dart b/lib/l10n/generated/app_localizations_th.dart new file mode 100644 index 00000000..16584ab8 --- /dev/null +++ b/lib/l10n/generated/app_localizations_th.dart @@ -0,0 +1,1561 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Thai (`th`). +class AppLocalizationsTh extends AppLocalizations { + AppLocalizationsTh([String locale = 'th']) : super(locale); + + @override + String get guest => 'ผู้มาเยือน'; + + @override + String get browse => 'เรียกดู'; + + @override + String get search => 'ค้นหา'; + + @override + String get library => 'คลัง'; + + @override + String get lyrics => 'เนื้อเพลง'; + + @override + String get settings => 'ตั้งค่า'; + + @override + String get genre_categories_filter => 'กรองประเภทหรือแนวเพลง...'; + + @override + String get genre => 'ประเภท'; + + @override + String get personalized => 'ปรับแต่ง'; + + @override + String get featured => 'เด่น'; + + @override + String get new_releases => 'เพิ่งปล่อยใหม่'; + + @override + String get songs => 'เพลง'; + + @override + String playing_track(Object track) { + return 'กำลังเล่น $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'การดำเนินการนี้จะล้างคิวปัจจุบัน $track_length แทร็ก จะถูกลบออก\nคุณต้องการดำเนินการต่อหรือไม่?'; + } + + @override + String get load_more => 'โหลดเพิ่มเติม'; + + @override + String get playlists => 'เพลย์ลิสต์'; + + @override + String get artists => 'ศิลปิน'; + + @override + String get albums => 'อัลบั้ม'; + + @override + String get tracks => 'แทร็ก'; + + @override + String get downloads => 'ดาวน์โหลด'; + + @override + String get filter_playlists => 'กรองเพลย์ลิสต์...'; + + @override + String get liked_tracks => 'เพลงที่ชอบ'; + + @override + String get liked_tracks_description => 'เพลงที่คุณชื่นชอบทั้งหมด'; + + @override + String get playlist => 'เพลย์ลิสต์'; + + @override + String get create_a_playlist => 'สร้างเพลย์ลิสต์'; + + @override + String get update_playlist => 'อัพเดทเพลย์ลิสต์'; + + @override + String get create => 'สร้าง'; + + @override + String get cancel => 'ยกเลิก'; + + @override + String get update => 'อัพเดท'; + + @override + String get playlist_name => 'ชื่อเพลย์ลิสต์'; + + @override + String get name_of_playlist => 'ชื่อของเพลย์ลิสต์'; + + @override + String get description => 'คำอธิบาย'; + + @override + String get public => 'สาธารณะ'; + + @override + String get collaborative => 'ร่วมมือกัน'; + + @override + String get search_local_tracks => 'ค้นหาเพลงในเครื่อง...'; + + @override + String get play => 'เล่น'; + + @override + String get delete => 'ลบ'; + + @override + String get none => 'ไม่มี'; + + @override + String get sort_a_z => 'เรียงตาม A-Z'; + + @override + String get sort_z_a => 'เรียงตาม Z-A'; + + @override + String get sort_artist => 'เรียงตามศิลปิน'; + + @override + String get sort_album => 'เรียงตามอัลบั้ม'; + + @override + String get sort_duration => 'เรียงตามความยาว'; + + @override + String get sort_tracks => 'เรียงตามเพลง'; + + @override + String currently_downloading(Object tracks_length) { + return 'กำลังดาวน์โหลด ($tracks_length)'; + } + + @override + String get cancel_all => 'ยกเลิกทั้งหมด'; + + @override + String get filter_artist => 'กรองศิลปิน...'; + + @override + String followers(Object followers) { + return '$followers ผู้ติดตาม'; + } + + @override + String get add_artist_to_blacklist => 'เพิ่มศิลปินในบัญชีดำ'; + + @override + String get top_tracks => 'เพลงฮิต'; + + @override + String get fans_also_like => 'แฟนๆ ยังชอบ'; + + @override + String get loading => 'กำลังโหลด...'; + + @override + String get artist => 'ศิลปิน'; + + @override + String get blacklisted => 'อยู่ในบัญชีดำ'; + + @override + String get following => 'กำลังติดตาม'; + + @override + String get follow => 'ติดตาม'; + + @override + String get artist_url_copied => 'คัดลอก URL ศิลปินไปยังคลิปบอร์ด'; + + @override + String added_to_queue(Object tracks) { + return 'เพิ่ม $tracks เพลงลงในคิว'; + } + + @override + String get filter_albums => 'กรองอัลบั้ม...'; + + @override + String get synced => 'ซิงค์'; + + @override + String get plain => 'เรียบง่าย'; + + @override + String get shuffle => 'สุ่ม'; + + @override + String get search_tracks => 'ค้นหาเพลง...'; + + @override + String get released => 'เผยแพร่'; + + @override + String error(Object error) { + return 'ข้อผิดพลาด $error'; + } + + @override + String get title => 'ชื่อ'; + + @override + String get time => 'เวลา'; + + @override + String get more_actions => 'เพิ่มเติม'; + + @override + String download_count(Object count) { + return 'ดาวน์โหลด ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'เพิ่ม ($count) ลงในเพลย์ลิสต์'; + } + + @override + String add_count_to_queue(Object count) { + return 'เพิ่ม ($count) ลงในคิว'; + } + + @override + String play_count_next(Object count) { + return 'เล่น ($count) ต่อไป'; + } + + @override + String get album => 'อัลบั้ม'; + + @override + String copied_to_clipboard(Object data) { + return 'คัดลอก $data ไปยังคลิปบอร์ด'; + } + + @override + String add_to_following_playlists(Object track) { + return 'เพิ่ม $track ลงในเพลย์ลิสต์'; + } + + @override + String get add => 'เพิ่ม'; + + @override + String added_track_to_queue(Object track) { + return 'เพิ่ม $track ลงในคิว'; + } + + @override + String get add_to_queue => 'เพิ่มลงในคิว'; + + @override + String track_will_play_next(Object track) { + return '$track จะเล่นต่อไป'; + } + + @override + String get play_next => 'เล่นต่อไป'; + + @override + String removed_track_from_queue(Object track) { + return 'ลบ $track ออกจากคิว'; + } + + @override + String get remove_from_queue => 'ลบออกจากคิว'; + + @override + String get remove_from_favorites => 'ลบออกจากรายการโปรด'; + + @override + String get save_as_favorite => 'บันทึกเป็นรายการโปรด'; + + @override + String get add_to_playlist => 'เพิ่มลงในเพลย์ลิสต์'; + + @override + String get remove_from_playlist => 'ลบออกจากเพลย์ลิสต์'; + + @override + String get add_to_blacklist => 'เพิ่มลงในบัญชีดำ'; + + @override + String get remove_from_blacklist => 'ลบออกจากบัญชีดำ'; + + @override + String get share => 'แชร์'; + + @override + String get mini_player => 'มินิเพลเยอร์'; + + @override + String get slide_to_seek => 'เลื่อนเพื่อไปข้างหน้าหรือถอยหลัง'; + + @override + String get shuffle_playlist => 'สุ่มเพลย์ลิสต์'; + + @override + String get unshuffle_playlist => 'ยกเลิกการสุ่มเพลย์ลิสต์'; + + @override + String get previous_track => 'แทร็กก่อนหน้า'; + + @override + String get next_track => 'แทร็กถัดไป'; + + @override + String get pause_playback => 'หยุดการเล่น'; + + @override + String get resume_playback => 'เล่นต่อ'; + + @override + String get loop_track => 'วนเพลง'; + + @override + String get no_loop => 'ไม่มีการวนซ้ำ'; + + @override + String get repeat_playlist => 'ซ้ำเพลย์ลิสต์'; + + @override + String get queue => 'คิว'; + + @override + String get alternative_track_sources => 'แหล่งแทร็กอื่น'; + + @override + String get download_track => 'ดาวน์โหลดแทร็ก'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks แทร็กในคิว'; + } + + @override + String get clear_all => 'ล้างทั้งหมด'; + + @override + String get show_hide_ui_on_hover => 'แสดง/ซ่อน UI เมื่อโฮเวอร์'; + + @override + String get always_on_top => 'อยู่ด้านบนเสมอ'; + + @override + String get exit_mini_player => 'ออกจากมินิเพลย์เยอร์'; + + @override + String get download_location => 'ตำแหน่งดาวน์โหลด'; + + @override + String get local_library => 'ห้องสมุดท้องถิ่น'; + + @override + String get add_library_location => 'เพิ่มในห้องสมุด'; + + @override + String get remove_library_location => 'ลบออกจากห้องสมุด'; + + @override + String get account => 'บัญชี'; + + @override + String get logout => 'ออกจากระบบ'; + + @override + String get logout_of_this_account => 'ออกจากระบบบัญชีนี้'; + + @override + String get language_region => 'ภาษาและภูมิภาค'; + + @override + String get language => 'ภาษา'; + + @override + String get system_default => 'ค่าเริ่มต้นของระบบ'; + + @override + String get market_place_region => 'ภูมิภาค Marketplace'; + + @override + String get recommendation_country => 'ประเทศที่แนะนำ'; + + @override + String get appearance => 'ลักษณะที่ปรากฏ'; + + @override + String get layout_mode => 'โหมดเค้าโครง'; + + @override + String get override_layout_settings => + 'แทนที่การตั้งค่าโหมดเค้าโครงแบบตอบสนอง'; + + @override + String get adaptive => 'ปรับเปลี่ยน'; + + @override + String get compact => 'กระชับ'; + + @override + String get extended => 'ขยาย'; + + @override + String get theme => 'ธีม'; + + @override + String get dark => 'มืด'; + + @override + String get light => 'สว่าง'; + + @override + String get system => 'ระบบ'; + + @override + String get accent_color => 'สีเน้น'; + + @override + String get sync_album_color => 'ซิงค์สีอัลบั้ม'; + + @override + String get sync_album_color_description => + 'ใช้สีเด่นของอาร์ตอัลบั้มเป็นสีเน้น'; + + @override + String get playback => 'การเล่น'; + + @override + String get audio_quality => 'คุณภาพเสียง'; + + @override + String get high => 'สูง'; + + @override + String get low => 'ต่ำ'; + + @override + String get pre_download_play => 'ดาวน์โหลดล่วงหน้าและเล่น'; + + @override + String get pre_download_play_description => + 'แทนที่จะสตรีมเสียง ดาวน์โหลดข้อมูลและเล่นแทน (แนะนำสำหรับผู้ใช้แบนด์วิดธ์สูง)'; + + @override + String get skip_non_music => 'ข้ามส่วนที่ไม่ใช่เพลง (SponsorBlock)'; + + @override + String get blacklist_description => 'แทร็กและศิลปินที่บล็อก'; + + @override + String get wait_for_download_to_finish => + 'โปรดรอให้การดาวน์โหลดปัจจุบันเสร็จสิ้น'; + + @override + String get desktop => 'เดสก์ท็อป'; + + @override + String get close_behavior => 'ปิดพฤติกรรม'; + + @override + String get close => 'ปิด'; + + @override + String get minimize_to_tray => 'ลดขนาดลงถาด'; + + @override + String get show_tray_icon => 'แสดงไอคอนถาดระบบ'; + + @override + String get about => 'เกี่ยวกับ'; + + @override + String get u_love_spotube => 'เรารู้ว่าคุณรัก Spotube'; + + @override + String get check_for_updates => 'ตรวจสอบการปรับปรุง'; + + @override + String get about_spotube => 'เกี่ยวกับ Spotube'; + + @override + String get blacklist => 'แบล็กลิสต์'; + + @override + String get please_sponsor => 'กรุณาสนับสนุน/บริจาค'; + + @override + String get spotube_description => + 'Spotube โปรแกรมเล่น Spotify ฟรีสำหรับทุกคน น้ำหนักเบา รองรับหลายแพลตฟอร์ม'; + + @override + String get version => 'รุ่น'; + + @override + String get build_number => 'หมายเลขบิลด์'; + + @override + String get founder => 'ผู้ก่อตั้ง'; + + @override + String get repository => 'ที่เก็บ'; + + @override + String get bug_issues => 'ข้อผิดพลาด+ปัญหา'; + + @override + String get made_with => 'ทำด้วย❤️ใน บังคลาเทศ🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'ใบอนุญาต'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'ไม่ต้องกังวล ข้อมูลรับรองใดๆ ของคุณจะไม่ถูกเก็บรวบรวมหรือแชร์กับใคร'; + + @override + String get know_how_to_login => 'ไม่รู้จักวิธีดำเนินการนี้ใช่ไหม'; + + @override + String get follow_step_by_step_guide => 'ทำตามคู่มือทีละขั้น'; + + @override + String cookie_name_cookie(Object name) { + return 'คุกกี้ $name'; + } + + @override + String get fill_in_all_fields => 'กรุณากรอกข้อมูลทุกช่อง'; + + @override + String get submit => 'ยื่น'; + + @override + String get exit => 'ออก'; + + @override + String get previous => 'ย้อนกลับ'; + + @override + String get next => 'ถัดไป'; + + @override + String get done => 'เสร็จ'; + + @override + String get step_1 => 'ขั้นที่ 1'; + + @override + String get first_go_to => 'ก่อนอื่น ไปที่'; + + @override + String get something_went_wrong => 'มีอะไรผิดพลาด'; + + @override + String get piped_instance => 'อินสแตนซ์เซิร์ฟเวอร์แบบ Pipe'; + + @override + String get piped_description => + 'อินสแตนซ์เซิร์ฟเวอร์แบบ Pipe ที่ใช้สำหรับการจับคู่แทร็ก'; + + @override + String get piped_warning => + 'บางอย่างอาจใช้งานไม่ได้ผล คุณจึงต้องรับความเสี่ยงเอง'; + + @override + String get invidious_instance => 'อินสแตนซ์เซิร์ฟเวอร์ Invidious'; + + @override + String get invidious_description => + 'อินสแตนซ์เซิร์ฟเวอร์ Invidious ที่ใช้สำหรับการจับคู่เพลง'; + + @override + String get invidious_warning => + 'บางอันอาจใช้งานไม่ดี ใช้ด้วยความเสี่ยงของคุณเอง'; + + @override + String get generate => 'สร้าง'; + + @override + String track_exists(Object track) { + return 'แทร็ก $track มีอยู่แล้ว'; + } + + @override + String get replace_downloaded_tracks => 'แทนที่แทร็กที่ดาวน์โหลดทั้งหมด'; + + @override + String get skip_download_tracks => 'ข้ามการดาวน์โหลดแทร็กที่ดาวน์โหลดทั้งหมด'; + + @override + String get do_you_want_to_replace => 'คุณต้องการแทนที่แทร็กที่มีอยู่หรือไม่'; + + @override + String get replace => 'แทนที่'; + + @override + String get skip => 'ข้าม'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'เลือกสูงสุด $count $type'; + } + + @override + String get select_genres => 'เลือกประเภท'; + + @override + String get add_genres => 'เพิ่มประเภท'; + + @override + String get country => 'ประเทศ'; + + @override + String get number_of_tracks_generate => 'จำนวนแทร็กที่จะสร้าง'; + + @override + String get acousticness => 'อะคูสติก'; + + @override + String get danceability => 'ความสามารถในการเต้น'; + + @override + String get energy => 'พลัง'; + + @override + String get instrumentalness => 'บรรเลง'; + + @override + String get liveness => 'ความสด'; + + @override + String get loudness => 'ความดัง'; + + @override + String get speechiness => 'การพูด'; + + @override + String get valence => 'ความสุข'; + + @override + String get popularity => 'ความนิยม'; + + @override + String get key => 'คีย์'; + + @override + String get duration => 'ระยะเวลา (วินาที)'; + + @override + String get tempo => 'ความเร็ว (BPM)'; + + @override + String get mode => 'โหมด'; + + @override + String get time_signature => 'ลายเซ็นเวลา'; + + @override + String get short => 'สั้น'; + + @override + String get medium => 'กลาง'; + + @override + String get long => 'ยาว'; + + @override + String get min => 'ต่ำสุด'; + + @override + String get max => 'สูงสุด'; + + @override + String get target => 'เป้าหมาย'; + + @override + String get moderate => 'ปานกลาง'; + + @override + String get deselect_all => 'ยกเลิกการเลือกทั้งหมด'; + + @override + String get select_all => 'เลือกทั้งหมด'; + + @override + String get are_you_sure => 'คุณแน่ใจไหม?'; + + @override + String get generating_playlist => 'กำลังสร้างเพลย์ลิสต์ที่คุณกำหนดเอง...'; + + @override + String selected_count_tracks(Object count) { + return 'เลือก $count แทร็ก'; + } + + @override + String get download_warning => + 'ถ้าคุณดาวน์โหลดเพลงทั้งหมดเป็นจำนวนมาก คุณกำลังละเมิดลิขสิทธิ์เพลงและสร้างความเสียหายให้กับสังคมดนตรี สร้างสรรค์ หวังว่าคุณจะรับรู้เรื่องนี้ เสมอ พยายามเคารพและสนับสนุนผลงานหนักของศิลปิน'; + + @override + String get download_ip_ban_warning => + 'นอกเหนือจากนั้น IP ของคุณอาจถูกบล็อกบน YouTube เนื่องจากคำขอดาวน์โหลดมากเกินกว่าปกติ การบล็อก IP หมายความว่าคุณไม่สามารถใช้ YouTube (แม้ว่าคุณจะล็อกอินอยู่) เป็นเวลาอย่างน้อย 2-3 เดือนจากอุปกรณ์ IP นั้น และ Spotube จะไม่รับผิดชอบใด ๆ หากสิ่งนี้เกิดขึ้น'; + + @override + String get by_clicking_accept_terms => + 'คลิก \'ยอมรับ\' คุณยินยอมตามเงื่อนไขต่อไปนี้:'; + + @override + String get download_agreement_1 => + 'ฉันรู้ว่าฉันกำลังละเมิดลิขสิทธิ์เพลง ฉันเลว'; + + @override + String get download_agreement_2 => + 'ฉันจะสนับสนุนศิลปินทุกที่ที่ฉันทำได้และฉันทำสิ่งนี้เพียงเพราะฉันไม่มีเงินซื้อผลงานศิลปะของพวกเขา'; + + @override + String get download_agreement_3 => + 'ฉันรับทราบอย่างสมบูรณ์ว่า IP ของฉันอาจถูกบล็อกบน YouTube และฉันจะไม่ถือ Spotube หรือเจ้าของ/ผู้มีส่วนร่วมใด ๆ รับผิดชอบต่ออุบัติเหตุใด ๆ ที่เกิดจากการกระทำปัจจุบันของฉัน'; + + @override + String get decline => 'ปฏิเสธ'; + + @override + String get accept => 'ยอมรับ'; + + @override + String get details => 'รายละเอียด'; + + @override + String get youtube => 'youtube'; + + @override + String get channel => 'ช่อง'; + + @override + String get likes => 'ถูกใจ'; + + @override + String get dislikes => 'ไม่ชอบ'; + + @override + String get views => 'วิว'; + + @override + String get streamUrl => 'สตรีม URL'; + + @override + String get stop => 'หยุด'; + + @override + String get sort_newest => 'เรียงตามการเพิ่มใหม่ล่าสุด'; + + @override + String get sort_oldest => 'เรียงตามการเพิ่มเก่าสุด'; + + @override + String get sleep_timer => 'ตั้งเวลาปิด'; + + @override + String mins(Object minutes) { + return '$minutes นาที'; + } + + @override + String hours(Object hours) { + return '$hours ชั่วโมง'; + } + + @override + String hour(Object hours) { + return '$hours ชั่วโมง'; + } + + @override + String get custom_hours => 'ชั่วโมงที่กำหนดเอง'; + + @override + String get logs => 'บันทึก'; + + @override + String get developers => 'นักพัฒนา'; + + @override + String get not_logged_in => 'คุณไม่ได้เข้าสู่ระบบ'; + + @override + String get search_mode => 'โหมดการค้นหา'; + + @override + String get audio_source => 'แหล่งที่มาของเสียง'; + + @override + String get ok => 'ตกลง'; + + @override + String get failed_to_encrypt => 'เข้ารหัสล้มเหลว'; + + @override + String get encryption_failed_warning => + 'Spotube ใช้การเข้ารหัสเพื่อเก็บข้อมูลของคุณอย่างปลอดภัย แต่ไม่สามารถทำได้ ดังนั้นจะเปลี่ยนเป็นการจัดเก็บที่ไม่ปลอดภัย\nหากคุณใช้ Linux โปรดตรวจสอบว่าคุณได้ติดตั้งบริการลับ (gnome-keyring, kde-wallet, keepassxc เป็นต้น)'; + + @override + String get querying_info => 'กำลังดึงข้อมูล...'; + + @override + String get piped_api_down => 'Piped API ไม่ทำงาน'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Piped instance $pipedInstance ไม่ทำงานขณะนี้\n\nเปลี่ยนอินสแตนซ์หรือเปลี่ยน \'ประเภท API\' เป็น YouTube API อย่างเป็นทางการ\n\nอย่าลืมรีสตาร์ทแอปหลังจากเปลี่ยน'; + } + + @override + String get you_are_offline => 'คุณออฟไลน์อยู่'; + + @override + String get connection_restored => + 'การเชื่อมต่ออินเทอร์เน็ตของคุณได้รับการกู้คืน'; + + @override + String get use_system_title_bar => 'ใช้แถบชื่อระบบ'; + + @override + String get crunching_results => 'กำลังประมวลผล...'; + + @override + String get search_to_get_results => 'ค้นหาเพื่อดูผลลัพธ์'; + + @override + String get use_amoled_mode => 'ธีมมืดสนิท'; + + @override + String get pitch_dark_theme => 'โหมด AMOLED'; + + @override + String get normalize_audio => 'ปรับระดับเสียง'; + + @override + String get change_cover => 'เปลี่ยนปก'; + + @override + String get add_cover => 'เพิ่มปก'; + + @override + String get restore_defaults => 'คืนค่าเริ่มต้น'; + + @override + String get download_music_format => 'รูปแบบการดาวน์โหลดเพลง'; + + @override + String get streaming_music_format => 'รูปแบบการสตรีมเพลง'; + + @override + String get download_music_quality => 'คุณภาพการดาวน์โหลด'; + + @override + String get streaming_music_quality => 'คุณภาพการสตรีม'; + + @override + String get login_with_lastfm => 'เข้าสู่ระบบด้วย Last.fm'; + + @override + String get connect => 'เชื่อมต่อ'; + + @override + String get disconnect_lastfm => 'ตัดการเชื่อมต่อ Last.fm'; + + @override + String get disconnect => 'ตัดการเชื่อมต่อ'; + + @override + String get username => 'ชื่อผู้ใช้'; + + @override + String get password => 'รหัสผ่าน'; + + @override + String get login => 'เข้าสู่ระบบ'; + + @override + String get login_with_your_lastfm => 'เข้าสู่ระบบด้วย Last.fm'; + + @override + String get scrobble_to_lastfm => 'Scrobble ไปเป็น Last.fm'; + + @override + String get go_to_album => 'ไปที่อัลบั้ม'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'เรียกดูทั้งหมด'; + + @override + String get genres => 'ประเภท'; + + @override + String get explore_genres => 'สำรวจประเภท'; + + @override + String get friends => 'เพื่อน'; + + @override + String get no_lyrics_available => 'ขออภัย ไม่พบเนื้อเพลงสำหรับเพลงนี้'; + + @override + String get start_a_radio => 'เปิดวิทยุ'; + + @override + String get how_to_start_radio => 'หากต้องการเปิดวิทยุฟังยังไง?'; + + @override + String get replace_queue_question => + 'คุณต้องการแทนที่คิวปัจจุบันหรือเพิ่มเข้าไปหรือไม่'; + + @override + String get endless_playback => 'เล่นซ้ำ'; + + @override + String get delete_playlist => 'ลบเพลย์ลิสต์'; + + @override + String get delete_playlist_confirmation => + 'คุณแน่ใจที่จะลบเพลย์ลิสต์นี้หรือไม่'; + + @override + String get local_tracks => 'เพลงในเครื่อง'; + + @override + String get local_tab => 'ท้องถิ่น'; + + @override + String get song_link => 'ลิงค์เพลง'; + + @override + String get skip_this_nonsense => 'ข้ามสิ่งไร้สาระนี้'; + + @override + String get freedom_of_music => '“เสรีภาพแห่งเสียงเพลง”'; + + @override + String get freedom_of_music_palm => '“เสรีภาพแห่งเสียงเพลง ในมือของคุณ”'; + + @override + String get get_started => 'เริ่มต้น'; + + @override + String get youtube_source_description => 'แนะนำและใช้งานได้ดีที่สุด'; + + @override + String get piped_source_description => + 'รู้สึกอิสระ? เหมือน YouTube แต่ฟรีกว่าเยอะ'; + + @override + String get jiosaavn_source_description => 'ดีที่สุดสำหรับภูมิภาคเอเชียใต้'; + + @override + String get invidious_source_description => + 'คล้ายกับ Piped แต่มีความพร้อมใช้งานสูงกว่า'; + + @override + String highest_quality(Object quality) { + return 'คุณภาพสูงสุด: $quality'; + } + + @override + String get select_audio_source => 'เลือกแหล่งเสียง'; + + @override + String get endless_playback_description => 'เพิ่มเพลงใหม่ลงในคิวโดยอัตโนมัติ'; + + @override + String get choose_your_region => 'เลือกภูมิภาคของคุณ'; + + @override + String get choose_your_region_description => + 'สิ่งนี้จะช่วยให้ Spotube แสดงเนื้อหาที่เหมาะสมสำหรับคุณ'; + + @override + String get choose_your_language => 'เลือกภาษาของคุณ'; + + @override + String get help_project_grow => 'ช่วยให้โครงการนี้เติบโต'; + + @override + String get help_project_grow_description => + 'Spotube เป็นโครงการโอเพนซอร์ส คุณสามารถช่วยให้โครงการนี้เติบโตได้โดยการมีส่วนร่วมในโครงการ รายงานข้อบกพร่อง หรือเสนอคุณสมบัติใหม่'; + + @override + String get contribute_on_github => 'มีส่วนร่วมบน GitHub'; + + @override + String get donate_on_open_collective => 'บริจาคบน Open Collective'; + + @override + String get browse_anonymously => 'เรียกดูแบบไม่ระบุตัวตน'; + + @override + String get enable_connect => 'เปิดใช้งานการเชื่อมต่อ'; + + @override + String get enable_connect_description => 'ควบคุม Spotube จากอุปกรณ์อื่น'; + + @override + String get devices => 'อุปกรณ์'; + + @override + String get select => 'เลือก'; + + @override + String connect_client_alert(Object client) { + return 'คุณกำลังถูกควบคุมโดย $client'; + } + + @override + String get this_device => 'อุปกรณ์นี้'; + + @override + String get remote => 'ระยะไกล'; + + @override + String get stats => 'สถิติ'; + + @override + String and_n_more(Object count) { + return 'และ $count อีก'; + } + + @override + String get recently_played => 'เพลงที่เพิ่งเล่น'; + + @override + String get browse_more => 'ดูเพิ่มเติม'; + + @override + String get no_title => 'ไม่มีชื่อ'; + + @override + String get not_playing => 'ไม่เล่น'; + + @override + String get epic_failure => 'ล้มเหลวอย่างยิ่ง!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'เพิ่ม $tracks_length เพลงในคิว'; + } + + @override + String get spotube_has_an_update => 'Spotube มีการอัปเดต'; + + @override + String get download_now => 'ดาวน์โหลดตอนนี้'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum ได้รับการปล่อยออกมา'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version ได้รับการปล่อยออกมา'; + } + + @override + String get read_the_latest => 'อ่านข่าวสารล่าสุด '; + + @override + String get release_notes => 'บันทึกการปล่อย'; + + @override + String get pick_color_scheme => 'เลือกธีมสี'; + + @override + String get save => 'บันทึก'; + + @override + String get choose_the_device => 'เลือกอุปกรณ์:'; + + @override + String get multiple_device_connected => + 'มีอุปกรณ์เชื่อมต่อหลายเครื่อง\nเลือกอุปกรณ์ที่คุณต้องการให้การดำเนินการนี้เกิดขึ้น'; + + @override + String get nothing_found => 'ไม่พบข้อมูล'; + + @override + String get the_box_is_empty => 'กล่องว่างเปล่า'; + + @override + String get top_artists => 'ศิลปินยอดนิยม'; + + @override + String get top_albums => 'อัลบั้มยอดนิยม'; + + @override + String get this_week => 'สัปดาห์นี้'; + + @override + String get this_month => 'เดือนนี้'; + + @override + String get last_6_months => '6 เดือนที่ผ่านมา'; + + @override + String get this_year => 'ปีนี้'; + + @override + String get last_2_years => '2 ปีที่ผ่านมา'; + + @override + String get all_time => 'ตลอดกาล'; + + @override + String powered_by_provider(Object providerName) { + return 'ขับเคลื่อนโดย $providerName'; + } + + @override + String get email => 'อีเมล'; + + @override + String get profile_followers => 'ผู้ติดตาม'; + + @override + String get birthday => 'วันเกิด'; + + @override + String get subscription => 'การสมัครสมาชิก'; + + @override + String get not_born => 'ยังไม่เกิด'; + + @override + String get hacker => 'แฮ็กเกอร์'; + + @override + String get profile => 'โปรไฟล์'; + + @override + String get no_name => 'ไม่มีชื่อ'; + + @override + String get edit => 'แก้ไข'; + + @override + String get user_profile => 'โปรไฟล์ผู้ใช้'; + + @override + String count_plays(Object count) { + return '$count การเล่น'; + } + + @override + String get streaming_fees_hypothetical => + '*คำนวณจากการจ่ายเงินต่อการสตรีมของ Spotify\nระหว่าง \$0.003 ถึง \$0.005 นี่เป็นการคำนวณสมมุติ\nเพื่อให้ข้อมูลแก่ผู้ใช้เกี่ยวกับจำนวนเงินที่พวกเขา\nอาจจะจ่ายให้กับศิลปินหากพวกเขาฟังเพลงของพวกเขาใน Spotify'; + + @override + String get minutes_listened => 'เวลาที่ฟัง'; + + @override + String get streamed_songs => 'เพลงที่สตรีม'; + + @override + String count_streams(Object count) { + return '$count สตรีม'; + } + + @override + String get owned_by_you => 'เป็นเจ้าของโดยคุณ'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl คัดลอกไปที่คลิปบอร์ดแล้ว'; + } + + @override + String get hipotetical_calculation => + '*การคำนวณนี้อิงจากค่าเฉลี่ยการจ่ายเงินต่อสตรีมของแพลตฟอร์มสตรีมมิ่งเพลงออนไลน์ที่ \$0.003 ถึง \$0.005 นี่เป็นการคำนวณสมมติฐานเพื่อให้ผู้ใช้เข้าใจว่าพวกเขาจะต้องจ่ายเงินให้ศิลปินเท่าไหร่หากพวกเขาฟังเพลงบนแพลตฟอร์มสตรีมมิ่งเพลงที่แตกต่างกัน'; + + @override + String count_mins(Object minutes) { + return '$minutes นาที'; + } + + @override + String get summary_minutes => 'นาที'; + + @override + String get summary_listened_to_music => 'ฟังเพลง'; + + @override + String get summary_songs => 'เพลง'; + + @override + String get summary_streamed_overall => 'สตรีมทั้งหมด'; + + @override + String get summary_owed_to_artists => 'ค้างชำระให้ศิลปิน\nในเดือนนี้'; + + @override + String get summary_artists => 'ศิลปิน'; + + @override + String get summary_music_reached_you => 'เพลงมาถึงคุณ'; + + @override + String get summary_full_albums => 'อัลบั้มเต็ม'; + + @override + String get summary_got_your_love => 'ได้รับความรักของคุณ'; + + @override + String get summary_playlists => 'เพลย์ลิสต์'; + + @override + String get summary_were_on_repeat => 'อยู่ในโหมดซ้ำ'; + + @override + String total_money(Object money) { + return 'รวม $money'; + } + + @override + String get webview_not_found => 'ไม่พบ Webview'; + + @override + String get webview_not_found_description => + 'ไม่พบ runtime ของ Webview บนอุปกรณ์ของคุณ\nหากติดตั้งแล้วตรวจสอบให้แน่ใจว่าอยู่ใน environment PATH\n\nหลังจากติดตั้งแล้ว ให้รีสตาร์ทแอป'; + + @override + String get unsupported_platform => 'แพลตฟอร์มไม่รองรับ'; + + @override + String get cache_music => 'แคชเพลง'; + + @override + String get open => 'เปิด'; + + @override + String get cache_folder => 'โฟลเดอร์แคช'; + + @override + String get export => 'ส่งออก'; + + @override + String get clear_cache => 'ล้างแคช'; + + @override + String get clear_cache_confirmation => 'คุณต้องการล้างแคชหรือไม่?'; + + @override + String get export_cache_files => 'ส่งออกไฟล์แคช'; + + @override + String found_n_files(Object count) { + return 'พบ $count ไฟล์'; + } + + @override + String get export_cache_confirmation => 'คุณต้องการส่งออกไฟล์เหล่านี้ไปยัง'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'ส่งออก $filesExported จาก $files ไฟล์'; + } + + @override + String get undo => 'ย้อนกลับ'; + + @override + String get download_all => 'ดาวน์โหลดทั้งหมด'; + + @override + String get add_all_to_playlist => 'เพิ่มทั้งหมดในเพลย์ลิสต์'; + + @override + String get add_all_to_queue => 'เพิ่มทั้งหมดในคิว'; + + @override + String get play_all_next => 'เล่นทั้งหมดถัดไป'; + + @override + String get pause => 'หยุดชั่วคราว'; + + @override + String get view_all => 'ดูทั้งหมด'; + + @override + String get no_tracks_added_yet => 'ดูเหมือนคุณยังไม่ได้เพิ่มเพลงใด ๆ'; + + @override + String get no_tracks => 'ดูเหมือนจะไม่มีเพลงที่นี่'; + + @override + String get no_tracks_listened_yet => 'ดูเหมือนคุณยังไม่ได้ฟังอะไรเลย'; + + @override + String get not_following_artists => 'คุณไม่ได้ติดตามศิลปินใด ๆ'; + + @override + String get no_favorite_albums_yet => + 'ดูเหมือนคุณยังไม่ได้เพิ่มอัลบัมใด ๆ ในรายการโปรด'; + + @override + String get no_logs_found => 'ไม่พบบันทึก'; + + @override + String get youtube_engine => 'เครื่องมือ YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine ยังไม่ได้ติดตั้ง'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine ยังไม่ได้ติดตั้งในระบบของคุณ'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'ตรวจสอบให้แน่ใจว่ามันมีอยู่ในตัวแปร PATH หรือ\nตั้งค่าพาธที่แท้จริงของไฟล์ที่สามารถทำงานได้ $engine ด้านล่าง'; + } + + @override + String get youtube_engine_unix_issue_message => + 'ใน macOS/Linux/Unix อย่าง OS การตั้งค่าพาธใน .zshrc/.bashrc/.bash_profile เป็นต้น จะไม่ทำงาน\nคุณต้องตั้งค่าพาธในไฟล์การกำหนดค่า shell'; + + @override + String get download => 'ดาวน์โหลด'; + + @override + String get file_not_found => 'ไม่พบไฟล์'; + + @override + String get custom => 'กำหนดเอง'; + + @override + String get add_custom_url => 'เพิ่ม URL แบบกำหนดเอง'; + + @override + String get edit_port => 'แก้ไขพอร์ต'; + + @override + String get port_helper_msg => + 'ค่าเริ่มต้นคือ -1 ซึ่งหมายถึงหมายเลขสุ่ม หากคุณได้กำหนดค่าไฟร์วอลล์แล้ว แนะนำให้ตั้งค่านี้'; + + @override + String connect_request(Object client) { + return 'อนุญาตให้ $client เชื่อมต่อหรือไม่?'; + } + + @override + String get connection_request_denied => + 'การเชื่อมต่อล้มเหลว ผู้ใช้ปฏิเสธการเข้าถึง'; + + @override + String get an_error_occurred => 'เกิดข้อผิดพลาด'; + + @override + String get copy_to_clipboard => 'คัดลอกไปยังคลิปบอร์ด'; + + @override + String get view_logs => 'ดูบันทึก'; + + @override + String get retry => 'ลองใหม่'; + + @override + String get no_default_metadata_provider_selected => + 'คุณไม่ได้ตั้งค่าผู้ให้บริการเมตาดาต้าเริ่มต้น'; + + @override + String get manage_metadata_providers => 'จัดการผู้ให้บริการเมตาดาต้า'; + + @override + String get open_link_in_browser => 'เปิดลิงก์ในเบราว์เซอร์หรือไม่?'; + + @override + String get do_you_want_to_open_the_following_link => + 'คุณต้องการเปิดลิงก์ต่อไปนี้หรือไม่'; + + @override + String get unsafe_url_warning => + 'การเปิดลิงก์จากแหล่งที่ไม่น่าเชื่อถืออาจไม่ปลอดภัย โปรดระมัดระวัง!\nคุณยังสามารถคัดลอกลิงก์ไปยังคลิปบอร์ดของคุณได้'; + + @override + String get copy_link => 'คัดลอกลิงก์'; + + @override + String get building_your_timeline => + 'กำลังสร้างไทม์ไลน์ของคุณตามการฟังของคุณ...'; + + @override + String get official => 'อย่างเป็นทางการ'; + + @override + String author_name(Object author) { + return 'ผู้เขียน: $author'; + } + + @override + String get third_party => 'บุคคลที่สาม'; + + @override + String get plugin_requires_authentication => + 'ปลั๊กอินต้องมีการรับรองความถูกต้อง'; + + @override + String get update_available => 'มีการอัปเดต'; + + @override + String get supports_scrobbling => 'รองรับการ scrobbling'; + + @override + String get plugin_scrobbling_info => + 'ปลั๊กอินนี้จะ scrobble เพลงของคุณเพื่อสร้างประวัติการฟังของคุณ'; + + @override + String get default_metadata_source => 'แหล่งเมตาดาต้าพื้นฐาน'; + + @override + String get set_default_metadata_source => 'ตั้งค่าแหล่งเมตาดาต้าพื้นฐาน'; + + @override + String get default_audio_source => 'แหล่งเสียงพื้นฐาน'; + + @override + String get set_default_audio_source => 'ตั้งค่าแหล่งเสียงพื้นฐาน'; + + @override + String get set_default => 'ตั้งค่าเริ่มต้น'; + + @override + String get support => 'สนับสนุน'; + + @override + String get support_plugin_development => 'สนับสนุนการพัฒนาปลั๊กอิน'; + + @override + String can_access_name_api(Object name) { + return '- สามารถเข้าถึง API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'คุณต้องการติดตั้งปลั๊กอินนี้หรือไม่?'; + + @override + String get third_party_plugin_warning => + 'ปลั๊กอินนี้มาจากที่เก็บของบุคคลที่สาม โปรดตรวจสอบให้แน่ใจว่าคุณเชื่อถือแหล่งที่มาก่อนทำการติดตั้ง'; + + @override + String get author => 'ผู้เขียน'; + + @override + String get this_plugin_can_do_following => 'ปลั๊กอินนี้สามารถทำสิ่งต่อไปนี้'; + + @override + String get install => 'ติดตั้ง'; + + @override + String get install_a_metadata_provider => 'ติดตั้งผู้ให้บริการเมตาดาต้า'; + + @override + String get no_tracks_playing => 'ขณะนี้ไม่มีเพลงที่กำลังเล่นอยู่'; + + @override + String get synced_lyrics_not_available => + 'ไม่มีเนื้อเพลงที่ซิงค์สำหรับเพลงนี้ กรุณาใช้แท็บ'; + + @override + String get plain_lyrics => 'เนื้อเพลงธรรมดา'; + + @override + String get tab_instead => 'แทน'; + + @override + String get disclaimer => 'ข้อสงวนสิทธิ์'; + + @override + String get third_party_plugin_dmca_notice => + 'ทีม Spotube ไม่รับผิดชอบใดๆ (รวมถึงทางกฎหมาย) สำหรับปลั๊กอิน \"บุคคลที่สาม\" ใดๆ\nโปรดใช้งานด้วยความเสี่ยงของคุณเอง สำหรับข้อบกพร่อง/ปัญหาใดๆ โปรดรายงานไปยังที่เก็บปลั๊กอิน\n\nหากปลั๊กอิน \"บุคคลที่สาม\" ใดๆ ละเมิด ToS/DMCA ของบริการ/นิติบุคคลใดๆ โปรดขอให้ผู้เขียนปลั๊กอิน \"บุคคลที่สาม\" หรือแพลตฟอร์มโฮสติ้ง เช่น GitHub/Codeberg ดำเนินการ ที่ระบุไว้ข้างต้น (ที่ติดป้าย \"บุคคลที่สาม\") เป็นปลั๊กอินสาธารณะ/ที่ดูแลโดยชุมชนทั้งหมด เราไม่ได้จัดการดูแล ดังนั้นเราจึงไม่สามารถดำเนินการใดๆ กับพวกเขาได้\n\n'; + + @override + String get input_does_not_match_format => 'อินพุตไม่ตรงกับรูปแบบที่ต้องการ'; + + @override + String get plugins => 'ปลั๊กอิน'; + + @override + String get paste_plugin_download_url => + 'วาง url ดาวน์โหลดหรือ url ที่เก็บ GitHub/Codeberg หรือลิงก์โดยตรงไปยังไฟล์ .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'ดาวน์โหลดและติดตั้งปลั๊กอินจาก url'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'ไม่สามารถเพิ่มปลั๊กอินได้: $error'; + } + + @override + String get upload_plugin_from_file => 'อัปโหลดปลั๊กอินจากไฟล์'; + + @override + String get installed => 'ติดตั้งแล้ว'; + + @override + String get available_plugins => 'ปลั๊กอินที่มีอยู่'; + + @override + String get configure_plugins => + 'กำหนดค่าปลั๊กอินผู้ให้บริการเมตาดาต้าและแหล่งเสียงของคุณเอง'; + + @override + String get audio_scrobblers => 'เครื่อง scrobbler เสียง'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'แหล่งที่มา: '; + + @override + String get uncompressed => 'ไม่บีบอัด'; + + @override + String get dab_music_source_description => + 'สำหรับคนรักเสียงเพลง ให้สตรีมเสียงคุณภาพสูง/ไร้การสูญเสียการบีบอัด การจับคู่แทร็กแม่นยำตาม ISRC'; +} diff --git a/lib/l10n/generated/app_localizations_tl.dart b/lib/l10n/generated/app_localizations_tl.dart new file mode 100644 index 00000000..5febc92d --- /dev/null +++ b/lib/l10n/generated/app_localizations_tl.dart @@ -0,0 +1,1581 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Tagalog (`tl`). +class AppLocalizationsTl extends AppLocalizations { + AppLocalizationsTl([String locale = 'tl']) : super(locale); + + @override + String get guest => 'Bisita'; + + @override + String get browse => 'Mag-browse'; + + @override + String get search => 'Maghanap'; + + @override + String get library => 'Silid-aklatan'; + + @override + String get lyrics => 'Mga Liriko'; + + @override + String get settings => 'Mga Setting'; + + @override + String get genre_categories_filter => 'I-filter ang mga kategorya o genre...'; + + @override + String get genre => 'Genre'; + + @override + String get personalized => 'Naka-personalize'; + + @override + String get featured => 'Tampok'; + + @override + String get new_releases => 'Mga Bagong Paglabas'; + + @override + String get songs => 'Mga Kanta'; + + @override + String playing_track(Object track) { + return 'Tumutugtog ang $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Ito ay magbubura ng kasalukuyang pila. $track_length na mga track ang tatanggalin\nGusto mo bang magpatuloy?'; + } + + @override + String get load_more => 'Mag-load pa'; + + @override + String get playlists => 'Mga Playlist'; + + @override + String get artists => 'Mga Artista'; + + @override + String get albums => 'Mga Album'; + + @override + String get tracks => 'Mga Track'; + + @override + String get downloads => 'Mga Download'; + + @override + String get filter_playlists => 'I-filter ang iyong mga playlist...'; + + @override + String get liked_tracks => 'Mga Nagustuhang Track'; + + @override + String get liked_tracks_description => + 'Lahat ng mga track na iyong nagustuhan'; + + @override + String get playlist => 'Playlist'; + + @override + String get create_a_playlist => 'Gumawa ng playlist'; + + @override + String get update_playlist => 'I-update ang playlist'; + + @override + String get create => 'Lumikha'; + + @override + String get cancel => 'Ikansela'; + + @override + String get update => 'I-update'; + + @override + String get playlist_name => 'Pangalan ng Playlist'; + + @override + String get name_of_playlist => 'Pangalan ng playlist'; + + @override + String get description => 'Paglalarawan'; + + @override + String get public => 'Pampubliko'; + + @override + String get collaborative => 'Pakikipagtulungan'; + + @override + String get search_local_tracks => 'Maghanap ng mga lokal na track...'; + + @override + String get play => 'I-play'; + + @override + String get delete => 'Burahin'; + + @override + String get none => 'Wala'; + + @override + String get sort_a_z => 'Ayusin ayon sa A-Z'; + + @override + String get sort_z_a => 'Ayusin ayon sa Z-A'; + + @override + String get sort_artist => 'Ayusin ayon sa Artista'; + + @override + String get sort_album => 'Ayusin ayon sa Album'; + + @override + String get sort_duration => 'Ayusin ayon sa Tagal'; + + @override + String get sort_tracks => 'Ayusin ang mga Track'; + + @override + String currently_downloading(Object tracks_length) { + return 'Kasalukuyang Nagda-download ($tracks_length)'; + } + + @override + String get cancel_all => 'Kanselahin Lahat'; + + @override + String get filter_artist => 'I-filter ang mga artista...'; + + @override + String followers(Object followers) { + return '$followers na mga Tagasunod'; + } + + @override + String get add_artist_to_blacklist => 'Idagdag ang artista sa blacklist'; + + @override + String get top_tracks => 'Mga Nangungunang Track'; + + @override + String get fans_also_like => 'Gusto rin ng mga tagahanga'; + + @override + String get loading => 'Naglo-load...'; + + @override + String get artist => 'Artista'; + + @override + String get blacklisted => 'Naka-blacklist'; + + @override + String get following => 'Sinusundan'; + + @override + String get follow => 'Sundan'; + + @override + String get artist_url_copied => 'Na-copy sa clipboard ang URL ng artista'; + + @override + String added_to_queue(Object tracks) { + return 'Idinagdag ang $tracks na mga track sa pila'; + } + + @override + String get filter_albums => 'I-filter ang mga album...'; + + @override + String get synced => 'Naka-sync'; + + @override + String get plain => 'Simpleng'; + + @override + String get shuffle => 'I-shuffle'; + + @override + String get search_tracks => 'Maghanap ng mga track...'; + + @override + String get released => 'Inilabas'; + + @override + String error(Object error) { + return 'Error $error'; + } + + @override + String get title => 'Pamagat'; + + @override + String get time => 'Oras'; + + @override + String get more_actions => 'Higit pang mga aksyon'; + + @override + String download_count(Object count) { + return 'I-download ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Idagdag ($count) sa Playlist'; + } + + @override + String add_count_to_queue(Object count) { + return 'Idagdag ($count) sa Pila'; + } + + @override + String play_count_next(Object count) { + return 'I-play ($count) kasunod'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return 'Na-copy ang $data sa clipboard'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Idagdag ang $track sa mga sumusunod na Playlist'; + } + + @override + String get add => 'Idagdag'; + + @override + String added_track_to_queue(Object track) { + return 'Idinagdag ang $track sa pila'; + } + + @override + String get add_to_queue => 'Idagdag sa pila'; + + @override + String track_will_play_next(Object track) { + return 'Ang $track ay tutugtog susunod'; + } + + @override + String get play_next => 'I-play susunod'; + + @override + String removed_track_from_queue(Object track) { + return 'Tinanggal ang $track mula sa pila'; + } + + @override + String get remove_from_queue => 'Alisin mula sa pila'; + + @override + String get remove_from_favorites => 'Alisin mula sa mga paborito'; + + @override + String get save_as_favorite => 'I-save bilang paborito'; + + @override + String get add_to_playlist => 'Idagdag sa playlist'; + + @override + String get remove_from_playlist => 'Alisin mula sa playlist'; + + @override + String get add_to_blacklist => 'Idagdag sa blacklist'; + + @override + String get remove_from_blacklist => 'Alisin mula sa blacklist'; + + @override + String get share => 'Ibahagi'; + + @override + String get mini_player => 'Mini Player'; + + @override + String get slide_to_seek => 'I-slide para mag-seek pasulong o pabalik'; + + @override + String get shuffle_playlist => 'I-shuffle ang playlist'; + + @override + String get unshuffle_playlist => 'I-unshuffle ang playlist'; + + @override + String get previous_track => 'Nakaraang track'; + + @override + String get next_track => 'Susunod na track'; + + @override + String get pause_playback => 'I-pause ang Playback'; + + @override + String get resume_playback => 'Ipagpatuloy ang Playback'; + + @override + String get loop_track => 'I-loop ang track'; + + @override + String get no_loop => 'Walang loop'; + + @override + String get repeat_playlist => 'Ulitin ang playlist'; + + @override + String get queue => 'Pila'; + + @override + String get alternative_track_sources => + 'Alternatibong mga pinagmulan ng track'; + + @override + String get download_track => 'I-download ang track'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks na mga track sa pila'; + } + + @override + String get clear_all => 'Burahin lahat'; + + @override + String get show_hide_ui_on_hover => 'Ipakita/Itago ang UI sa hover'; + + @override + String get always_on_top => 'Palaging nasa ibabaw'; + + @override + String get exit_mini_player => 'Lumabas sa Mini player'; + + @override + String get download_location => 'Lokasyon ng pag-download'; + + @override + String get local_library => 'Lokal na silid-aklatan'; + + @override + String get add_library_location => 'Idagdag sa silid-aklatan'; + + @override + String get remove_library_location => 'Alisin mula sa silid-aklatan'; + + @override + String get account => 'Account'; + + @override + String get logout => 'Mag-logout'; + + @override + String get logout_of_this_account => 'Mag-logout sa account na ito'; + + @override + String get language_region => 'Wika at Rehiyon'; + + @override + String get language => 'Wika'; + + @override + String get system_default => 'Default ng Sistema'; + + @override + String get market_place_region => 'Rehiyon ng Marketplace'; + + @override + String get recommendation_country => 'Bansang Inirerekomenda'; + + @override + String get appearance => 'Hitsura'; + + @override + String get layout_mode => 'Mode ng Layout'; + + @override + String get override_layout_settings => + 'I-override ang mga setting ng responsive layout mode'; + + @override + String get adaptive => 'Umaangkop'; + + @override + String get compact => 'Kompakto'; + + @override + String get extended => 'Pinalawig'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Madilim'; + + @override + String get light => 'Maliwanag'; + + @override + String get system => 'Sistema'; + + @override + String get accent_color => 'Kulay ng Accent'; + + @override + String get sync_album_color => 'I-sync ang kulay ng album'; + + @override + String get sync_album_color_description => + 'Ginagamit ang pangunahing kulay ng album art bilang kulay ng accent'; + + @override + String get playback => 'Playback'; + + @override + String get audio_quality => 'Kalidad ng Audio'; + + @override + String get high => 'Mataas'; + + @override + String get low => 'Mababa'; + + @override + String get pre_download_play => 'Mag-pre-download at i-play'; + + @override + String get pre_download_play_description => + 'Sa halip na mag-stream ng audio, mag-download ng bytes at i-play sa halip (Inirerekomenda para sa mga gumagamit ng mataas na bandwidth)'; + + @override + String get skip_non_music => + 'Laktawan ang mga segment na hindi musika (SponsorBlock)'; + + @override + String get blacklist_description => 'Mga track at artista na nasa blacklist'; + + @override + String get wait_for_download_to_finish => + 'Mangyaring maghintay para matapos ang kasalukuyang pag-download'; + + @override + String get desktop => 'Desktop'; + + @override + String get close_behavior => 'Pag-uugali ng Pagsara'; + + @override + String get close => 'Isara'; + + @override + String get minimize_to_tray => 'I-minimize sa tray'; + + @override + String get show_tray_icon => 'Ipakita ang icon ng System tray'; + + @override + String get about => 'Tungkol sa'; + + @override + String get u_love_spotube => 'Alam naming gusto mo ang Spotube'; + + @override + String get check_for_updates => 'Maghanap ng mga update'; + + @override + String get about_spotube => 'Tungkol sa Spotube'; + + @override + String get blacklist => 'Blacklist'; + + @override + String get please_sponsor => 'Mangyaring Mag-sponsor/Mag-donate'; + + @override + String get spotube_description => + 'Spotube, isang magaan, cross-platform, libreng-para-sa-lahat na spotify client'; + + @override + String get version => 'Bersyon'; + + @override + String get build_number => 'Build Number'; + + @override + String get founder => 'Nagtatag'; + + @override + String get repository => 'Repository'; + + @override + String get bug_issues => 'Bug+Mga Isyu'; + + @override + String get made_with => 'Ginawa nang may ❤️ sa Bangladesh🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Lisensya'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Huwag mag-alala, ang alinman sa iyong mga kredensyal ay hindi kokolektahin o ibabahagi sa sinuman'; + + @override + String get know_how_to_login => 'Hindi mo alam kung paano gawin ito?'; + + @override + String get follow_step_by_step_guide => 'Sundin ang Hakbang-hakbang na gabay'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => 'Mangyaring punan ang lahat ng field'; + + @override + String get submit => 'Isumite'; + + @override + String get exit => 'Lumabas'; + + @override + String get previous => 'Nakaraan'; + + @override + String get next => 'Susunod'; + + @override + String get done => 'Tapos na'; + + @override + String get step_1 => 'Hakbang 1'; + + @override + String get first_go_to => 'Una, Pumunta sa'; + + @override + String get something_went_wrong => 'May nangyaring mali'; + + @override + String get piped_instance => 'Instance ng Piped Server'; + + @override + String get piped_description => + 'Ang instance ng Piped server na gagamitin para sa pagtutugma ng track'; + + @override + String get piped_warning => + 'Maaaring hindi gumagana nang mabuti ang ilan sa mga ito. Kaya gamitin sa sarili mong peligro'; + + @override + String get invidious_instance => 'Instance ng Invidious Server'; + + @override + String get invidious_description => + 'Ang instance ng Invidious server na gagamitin para sa pagtutugma ng track'; + + @override + String get invidious_warning => + 'Maaaring hindi gumagana nang mabuti ang ilan sa mga ito. Kaya gamitin sa sarili mong peligro'; + + @override + String get generate => 'Gumawa'; + + @override + String track_exists(Object track) { + return 'Ang Track na $track ay umiiral na'; + } + + @override + String get replace_downloaded_tracks => + 'Palitan ang lahat ng na-download na mga track'; + + @override + String get skip_download_tracks => + 'Laktawan ang pag-download ng lahat ng na-download na mga track'; + + @override + String get do_you_want_to_replace => + 'Gusto mo bang palitan ang umiiral na track??'; + + @override + String get replace => 'Palitan'; + + @override + String get skip => 'Laktawan'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Pumili ng hanggang $count $type'; + } + + @override + String get select_genres => 'Pumili ng mga Genre'; + + @override + String get add_genres => 'Magdagdag ng mga Genre'; + + @override + String get country => 'Bansa'; + + @override + String get number_of_tracks_generate => 'Bilang ng mga track na gagawin'; + + @override + String get acousticness => 'Acoustic-ness'; + + @override + String get danceability => 'Kakayahang Sayawin'; + + @override + String get energy => 'Enerhiya'; + + @override + String get instrumentalness => 'Instrumental-ness'; + + @override + String get liveness => 'Liveness'; + + @override + String get loudness => 'Lakas'; + + @override + String get speechiness => 'Pagsasalita'; + + @override + String get valence => 'Valence'; + + @override + String get popularity => 'Popularidad'; + + @override + String get key => 'Key'; + + @override + String get duration => 'Tagal (s)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mode'; + + @override + String get time_signature => 'Time Signature'; + + @override + String get short => 'Maikli'; + + @override + String get medium => 'Katamtaman'; + + @override + String get long => 'Mahaba'; + + @override + String get min => 'Min'; + + @override + String get max => 'Max'; + + @override + String get target => 'Target'; + + @override + String get moderate => 'Katamtaman'; + + @override + String get deselect_all => 'Alisin ang Pagkakapili sa Lahat'; + + @override + String get select_all => 'Piliin Lahat'; + + @override + String get are_you_sure => 'Sigurado ka ba?'; + + @override + String get generating_playlist => 'Gumagawa ng iyong custom na playlist...'; + + @override + String selected_count_tracks(Object count) { + return 'Napili ang $count na mga track'; + } + + @override + String get download_warning => + 'Kung nag-download ka ng lahat ng Track sa maramihan, malinaw na nagpa-pirate ka ng Musika at nagsasanhi ng pinsala sa creative society ng Musika. Sana ay alam mo ito. Palaging, subukang igalang at suportahan ang masipag na paggawa ng Artist'; + + @override + String get download_ip_ban_warning => + 'Sa nga pala, ang iyong IP ay maaaring ma-block sa YouTube dahil sa sobrang mga kahilingan sa pag-download kaysa sa karaniwan. Ang IP block ay nangangahulugang hindi mo magagamit ang YouTube (kahit na naka-log in ka) sa loob ng hindi bababa sa 2-3 buwan mula sa device na may IP na iyon. At hindi pinanghahawakan ng Spotube ang anumang responsibilidad kung mangyayari ito'; + + @override + String get by_clicking_accept_terms => + 'Sa pamamagitan ng pag-click sa \'tanggapin\', sumasang-ayon ka sa mga sumusunod na tuntunin:'; + + @override + String get download_agreement_1 => + 'Alam kong nagpa-pirate ako ng Musika. Masama ako'; + + @override + String get download_agreement_2 => + 'Susuportahan ko ang Artist saan man ako maaari at ginagawa ko lang ito dahil wala akong pera para bumili ng kanilang sining'; + + @override + String get download_agreement_3 => + 'Lubos kong nauunawaan na ang aking IP ay maaaring ma-block sa YouTube at hindi ko pinanghahawakan ang Spotube o ang kanyang mga may-ari/nag-ambag na responsable para sa anumang aksidente na sanhi ng aking kasalukuyang aksyon'; + + @override + String get decline => 'Tanggihan'; + + @override + String get accept => 'Tanggapin'; + + @override + String get details => 'Mga Detalye'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Channel'; + + @override + String get likes => 'Mga Like'; + + @override + String get dislikes => 'Mga Dislike'; + + @override + String get views => 'Mga View'; + + @override + String get streamUrl => 'Stream URL'; + + @override + String get stop => 'Ihinto'; + + @override + String get sort_newest => 'Ayusin ayon sa pinakabagong idinagdag'; + + @override + String get sort_oldest => 'Ayusin ayon sa pinakalumang idinagdag'; + + @override + String get sleep_timer => 'Sleep Timer'; + + @override + String mins(Object minutes) { + return '$minutes Minuto'; + } + + @override + String hours(Object hours) { + return '$hours Oras'; + } + + @override + String hour(Object hours) { + return '$hours Oras'; + } + + @override + String get custom_hours => 'Custom na Oras'; + + @override + String get logs => 'Mga Log'; + + @override + String get developers => 'Mga Developer'; + + @override + String get not_logged_in => 'Hindi ka naka-log in'; + + @override + String get search_mode => 'Mode ng Paghahanap'; + + @override + String get audio_source => 'Pinagmulan ng Audio'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Nabigong i-encrypt'; + + @override + String get encryption_failed_warning => + 'Gumagamit ng encryption ang Spotube para ligtas na i-store ang iyong data. Ngunit nabigo. Kaya babalik ito sa hindi secure na storage\nKung gumagamit ka ng linux, mangyaring tiyakin na mayroon kang anumang secret-service na naka-install (gnome-keyring, kde-wallet, keepassxc atbp)'; + + @override + String get querying_info => 'Kinukuha ang impormasyon...'; + + @override + String get piped_api_down => 'Ang Piped API ay hindi gumagana'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Ang instance ng Piped na $pipedInstance ay kasalukuyang hindi gumagana\n\nMaaari mong baguhin ang instance o baguhin ang \'Uri ng API\' sa opisyal na YouTube API\n\nSiguraduhing i-restart ang app pagkatapos ng pagbabago'; + } + + @override + String get you_are_offline => 'Kasalukuyan kang offline'; + + @override + String get connection_restored => + 'Naibalik na ang iyong koneksyon sa internet'; + + @override + String get use_system_title_bar => 'Gamitin ang title bar ng system'; + + @override + String get crunching_results => 'Pinaproseso ang mga resulta...'; + + @override + String get search_to_get_results => 'Maghanap para makakuha ng mga resulta'; + + @override + String get use_amoled_mode => 'Matingkad na itim na madilim na tema'; + + @override + String get pitch_dark_theme => 'AMOLED Mode'; + + @override + String get normalize_audio => 'I-normalize ang audio'; + + @override + String get change_cover => 'Baguhin ang cover'; + + @override + String get add_cover => 'Magdagdag ng cover'; + + @override + String get restore_defaults => 'Ibalik ang mga default'; + + @override + String get download_music_format => 'I-download na format ng musika'; + + @override + String get streaming_music_format => 'Format ng streaming ng musika'; + + @override + String get download_music_quality => 'Kalidad ng i-download na musika'; + + @override + String get streaming_music_quality => 'Kalidad ng streaming ng musika'; + + @override + String get login_with_lastfm => 'Mag-login gamit ang Last.fm'; + + @override + String get connect => 'Kumonekta'; + + @override + String get disconnect_lastfm => 'Idiskonekta ang Last.fm'; + + @override + String get disconnect => 'Idiskonekta'; + + @override + String get username => 'Username'; + + @override + String get password => 'Password'; + + @override + String get login => 'Mag-login'; + + @override + String get login_with_your_lastfm => + 'Mag-login gamit ang iyong Last.fm account'; + + @override + String get scrobble_to_lastfm => 'I-scrobble sa Last.fm'; + + @override + String get go_to_album => 'Pumunta sa Album'; + + @override + String get discord_rich_presence => 'Discord Rich Presence'; + + @override + String get browse_all => 'I-browse Lahat'; + + @override + String get genres => 'Mga Genre'; + + @override + String get explore_genres => 'Tuklasin ang mga Genre'; + + @override + String get friends => 'Mga Kaibigan'; + + @override + String get no_lyrics_available => + 'Paumanhin, hindi mahanap ang lyrics para sa track na ito'; + + @override + String get start_a_radio => 'Magsimula ng Radio'; + + @override + String get how_to_start_radio => 'Paano mo gustong simulan ang radio?'; + + @override + String get replace_queue_question => + 'Gusto mo bang palitan ang kasalukuyang pila o idagdag dito?'; + + @override + String get endless_playback => 'Walang Hanggang Playback'; + + @override + String get delete_playlist => 'Burahin ang Playlist'; + + @override + String get delete_playlist_confirmation => + 'Sigurado ka bang gusto mong burahin ang playlist na ito?'; + + @override + String get local_tracks => 'Mga Lokal na Track'; + + @override + String get local_tab => 'Lokal'; + + @override + String get song_link => 'Link ng Kanta'; + + @override + String get skip_this_nonsense => 'Laktawan ang kalokohan na ito'; + + @override + String get freedom_of_music => '\"Kalayaan ng Musika\"'; + + @override + String get freedom_of_music_palm => '\"Kalayaan ng Musika sa iyong palad\"'; + + @override + String get get_started => 'Magsimula na tayo'; + + @override + String get youtube_source_description => + 'Inirerekomenda at pinakamahusay na gumagana.'; + + @override + String get piped_source_description => + 'Gusto ng kalayaan? Kapareho ng YouTube ngunit mas malaya.'; + + @override + String get jiosaavn_source_description => + 'Pinakamahusay para sa rehiyon ng South Asia.'; + + @override + String get invidious_source_description => + 'Katulad ng Piped ngunit may mas mataas na availability.'; + + @override + String highest_quality(Object quality) { + return 'Pinakamataas na Kalidad: $quality'; + } + + @override + String get select_audio_source => 'Pumili ng Pinagmulan ng Audio'; + + @override + String get endless_playback_description => + 'Awtomatikong magdagdag ng mga bagong kanta\nsa dulo ng pila'; + + @override + String get choose_your_region => 'Piliin ang iyong rehiyon'; + + @override + String get choose_your_region_description => + 'Ito ay tutulong sa Spotube na ipakita sa iyo ang tamang content\npara sa iyong lokasyon.'; + + @override + String get choose_your_language => 'Piliin ang iyong wika'; + + @override + String get help_project_grow => 'Tulungan ang proyektong ito na lumago'; + + @override + String get help_project_grow_description => + 'Ang Spotube ay isang open-source na proyekto. Maaari mong tulungan ang proyektong ito na lumago sa pamamagitan ng pag-contribute sa proyekto, pag-ulat ng mga bug, o pagmungkahi ng mga bagong feature.'; + + @override + String get contribute_on_github => 'Mag-contribute sa GitHub'; + + @override + String get donate_on_open_collective => 'Mag-donate sa Open Collective'; + + @override + String get browse_anonymously => 'Mag-browse nang Anonymous'; + + @override + String get enable_connect => 'I-enable ang Connect'; + + @override + String get enable_connect_description => + 'Kontrolin ang Spotube mula sa ibang mga device'; + + @override + String get devices => 'Mga Device'; + + @override + String get select => 'Pumili'; + + @override + String connect_client_alert(Object client) { + return 'Ikaw ay kontrolado ng $client'; + } + + @override + String get this_device => 'Ang Device na ito'; + + @override + String get remote => 'Remote'; + + @override + String get stats => 'Mga Stat'; + + @override + String and_n_more(Object count) { + return 'at $count pa'; + } + + @override + String get recently_played => 'Kamakailan Lang na Ni-play'; + + @override + String get browse_more => 'Mag-browse pa'; + + @override + String get no_title => 'Walang Pamagat'; + + @override + String get not_playing => 'Hindi tumutugtog'; + + @override + String get epic_failure => 'Epic na pagkabigo!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Nagdagdag ng $tracks_length na mga track sa pila'; + } + + @override + String get spotube_has_an_update => 'Ang Spotube ay may update'; + + @override + String get download_now => 'I-download Ngayon'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Ang Spotube Nightly $nightlyBuildNum ay inilabas na'; + } + + @override + String release_version(Object version) { + return 'Ang Spotube v$version ay inilabas na'; + } + + @override + String get read_the_latest => 'Basahin ang pinakabagong '; + + @override + String get release_notes => 'release notes'; + + @override + String get pick_color_scheme => 'Pumili ng color scheme'; + + @override + String get save => 'I-save'; + + @override + String get choose_the_device => 'Piliin ang device:'; + + @override + String get multiple_device_connected => + 'Mayroong maraming device na nakakonekta.\nPiliin ang device kung saan mo gustong maganap ang aksyon na ito'; + + @override + String get nothing_found => 'Walang nahanap'; + + @override + String get the_box_is_empty => 'Ang kahon ay walang laman'; + + @override + String get top_artists => 'Nangungunang mga Artista'; + + @override + String get top_albums => 'Nangungunang mga Album'; + + @override + String get this_week => 'Ngayong linggo'; + + @override + String get this_month => 'Ngayong buwan'; + + @override + String get last_6_months => 'Nakaraang 6 na buwan'; + + @override + String get this_year => 'Ngayong taon'; + + @override + String get last_2_years => 'Nakaraang 2 taon'; + + @override + String get all_time => 'Lahat ng panahon'; + + @override + String powered_by_provider(Object providerName) { + return 'Pinapagana ng $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Mga Tagasunod'; + + @override + String get birthday => 'Kaarawan'; + + @override + String get subscription => 'Subscription'; + + @override + String get not_born => 'Hindi pa ipinanganak'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profile'; + + @override + String get no_name => 'Walang Pangalan'; + + @override + String get edit => 'I-edit'; + + @override + String get user_profile => 'Profile ng User'; + + @override + String count_plays(Object count) { + return '$count na mga play'; + } + + @override + String get streaming_fees_hypothetical => + 'Mga bayarin sa streaming (hypothetical)'; + + @override + String get minutes_listened => 'Mga minutong pinapakinggan'; + + @override + String get streamed_songs => 'Mga na-stream na kanta'; + + @override + String count_streams(Object count) { + return '$count na mga stream'; + } + + @override + String get owned_by_you => 'Pag-aari mo'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return 'Na-kopya ang $shareUrl sa clipboard'; + } + + @override + String get hipotetical_calculation => + '*Ito ay kinakalkula batay sa average na payout ng online music streaming platform na \$0.003 hanggang \$0.005 kada stream. Ito ay isang hypothetical na kalkulasyon upang bigyan ang user ng insight kung magkano ang babayaran nila sa mga artist kung sakaling makinig sila ng kanilang kanta sa iba\'t ibang music streaming platform.'; + + @override + String count_mins(Object minutes) { + return '$minutes minuto'; + } + + @override + String get summary_minutes => 'minuto'; + + @override + String get summary_listened_to_music => 'Nakinig sa musika'; + + @override + String get summary_songs => 'mga kanta'; + + @override + String get summary_streamed_overall => 'Na-stream sa kabuuan'; + + @override + String get summary_owed_to_artists => 'Utang sa mga artista\nngayong buwan'; + + @override + String get summary_artists => 'artista'; + + @override + String get summary_music_reached_you => 'Umabot sa iyo ang musika'; + + @override + String get summary_full_albums => 'buong album'; + + @override + String get summary_got_your_love => 'Nakuha ang iyong pagmamahal'; + + @override + String get summary_playlists => 'mga playlist'; + + @override + String get summary_were_on_repeat => 'Pinu-playlst muli'; + + @override + String total_money(Object money) { + return 'Kabuuang $money'; + } + + @override + String get webview_not_found => 'Hindi nahanap ang Webview'; + + @override + String get webview_not_found_description => + 'Walang webview runtime na naka-install sa iyong device.\nKung naka-install ito, siguraduhing nasa Environment PATH\n\nPagkatapos mag-install, i-restart ang app'; + + @override + String get unsupported_platform => 'Hindi suportadong platform'; + + @override + String get cache_music => 'I-cache ang musika'; + + @override + String get open => 'Buksan'; + + @override + String get cache_folder => 'Folder ng cache'; + + @override + String get export => 'I-export'; + + @override + String get clear_cache => 'Burahin ang cache'; + + @override + String get clear_cache_confirmation => 'Gusto mo bang burahin ang cache?'; + + @override + String get export_cache_files => 'I-export ang mga Naka-cache na File'; + + @override + String found_n_files(Object count) { + return 'Nahanap ang $count na mga file'; + } + + @override + String get export_cache_confirmation => + 'Gusto mo bang i-export ang mga file na ito sa'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Na-export ang $filesExported mula sa $files na mga file'; + } + + @override + String get undo => 'I-undo'; + + @override + String get download_all => 'I-download lahat'; + + @override + String get add_all_to_playlist => 'Idagdag lahat sa playlist'; + + @override + String get add_all_to_queue => 'Idagdag lahat sa pila'; + + @override + String get play_all_next => 'I-play lahat susunod'; + + @override + String get pause => 'Pause'; + + @override + String get view_all => 'Tingnan lahat'; + + @override + String get no_tracks_added_yet => + 'Mukhang wala ka pang idinaragdag na mga track'; + + @override + String get no_tracks => 'Mukhang walang mga track dito'; + + @override + String get no_tracks_listened_yet => 'Mukhang wala ka pang pinakikinggan'; + + @override + String get not_following_artists => + 'Hindi ka sumusunod sa anumang mga artista'; + + @override + String get no_favorite_albums_yet => + 'Mukhang wala ka pang idinagdag na anumang mga album sa iyong mga paborito'; + + @override + String get no_logs_found => 'Walang nahanap na mga log'; + + @override + String get youtube_engine => 'YouTube Engine'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return 'Hindi naka-install ang $engine'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return 'Hindi naka-install ang $engine sa iyong sistema.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Siguraduhing available ito sa PATH variable o\ni-set ang absolute path sa $engine executable sa ibaba'; + } + + @override + String get youtube_engine_unix_issue_message => + 'Sa macOS/Linux/unix tulad ng OS, ang pag-set ng path sa .zshrc/.bashrc/.bash_profile atbp. ay hindi gagana.\nKailangan mong i-set ang path sa configuration file ng shell'; + + @override + String get download => 'I-download'; + + @override + String get file_not_found => 'Hindi nahanap ang file'; + + @override + String get custom => 'Custom'; + + @override + String get add_custom_url => 'Magdagdag ng custom URL'; + + @override + String get edit_port => 'I-edit ang port'; + + @override + String get port_helper_msg => + 'Ang default ay -1 na nagpapahiwatig ng random na numero. Kung na-configure mo ang firewall, inirerekomenda na itakda ito.'; + + @override + String connect_request(Object client) { + return 'Payagan ang $client na kumonekta?'; + } + + @override + String get connection_request_denied => + 'Tanggihan ang koneksyon. Tinanggihan ng gumagamit ang pag-access.'; + + @override + String get an_error_occurred => 'May naganap na error'; + + @override + String get copy_to_clipboard => 'Kopyahin sa clipboard'; + + @override + String get view_logs => 'Tingnan ang mga log'; + + @override + String get retry => 'Subukang muli'; + + @override + String get no_default_metadata_provider_selected => + 'Wala kang nakatakdang default na metadata provider'; + + @override + String get manage_metadata_providers => + 'Pamahalaan ang mga metadata provider'; + + @override + String get open_link_in_browser => 'Buksan ang Link sa Browser?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Gusto mo bang buksan ang sumusunod na link'; + + @override + String get unsafe_url_warning => + 'Maaaring hindi ligtas ang pagbukas ng mga link mula sa hindi pinagkakatiwalaang pinagmulan. Mag-ingat!\nMaaari mo ring kopyahin ang link sa iyong clipboard.'; + + @override + String get copy_link => 'Kopyahin ang Link'; + + @override + String get building_your_timeline => + 'Binubuo ang iyong timeline batay sa iyong mga pinakinggan...'; + + @override + String get official => 'Opisyal'; + + @override + String author_name(Object author) { + return 'May-akda: $author'; + } + + @override + String get third_party => 'Third-party'; + + @override + String get plugin_requires_authentication => + 'Nangangailangan ng authentication ang plugin'; + + @override + String get update_available => 'May available na update'; + + @override + String get supports_scrobbling => 'Sinusuportahan ang scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Sinis-scrobble ng plugin na ito ang iyong musika upang mabuo ang iyong kasaysayan ng pakikinig.'; + + @override + String get default_metadata_source => 'Default na pinagmulan ng metadata'; + + @override + String get set_default_metadata_source => + 'Itakda ang default na pinagmulan ng metadata'; + + @override + String get default_audio_source => 'Default na pinagmulan ng audio'; + + @override + String get set_default_audio_source => + 'Itakda ang default na pinagmulan ng audio'; + + @override + String get set_default => 'Itakda bilang default'; + + @override + String get support => 'Suporta'; + + @override + String get support_plugin_development => 'Suportahan ang pagbuo ng plugin'; + + @override + String can_access_name_api(Object name) { + return '- Maaaring i-access ang **$name** API'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Gusto mo bang i-install ang plugin na ito?'; + + @override + String get third_party_plugin_warning => + 'Ang plugin na ito ay mula sa third-party na repository. Mangyaring tiyakin na pinagkakatiwalaan mo ang pinagmulan bago mag-install.'; + + @override + String get author => 'May-akda'; + + @override + String get this_plugin_can_do_following => + 'Maaaring gawin ng plugin na ito ang sumusunod'; + + @override + String get install => 'I-install'; + + @override + String get install_a_metadata_provider => 'Mag-install ng Metadata Provider'; + + @override + String get no_tracks_playing => 'Walang Track na kasalukuyang tumutugtog'; + + @override + String get synced_lyrics_not_available => + 'Hindi available ang mga naka-sync na lyrics para sa kantang ito. Mangyaring gamitin ang'; + + @override + String get plain_lyrics => 'Simpleng Lyrics'; + + @override + String get tab_instead => 'na tab sa halip.'; + + @override + String get disclaimer => 'Disclaimer'; + + @override + String get third_party_plugin_dmca_notice => + 'Ang Spotube team ay walang hawak na anumang responsibilidad (kabilang ang legal) para sa anumang \"Third-party\" plugins.\nMangyaring gamitin ang mga ito sa iyong sariling peligro. Para sa anumang mga bug/isyu, mangyaring iulat ang mga ito sa repository ng plugin.\n\nKung ang anumang \"Third-party\" plugin ay lumalabag sa ToS/DMCA ng anumang serbisyo/legal na entity, mangyaring hilingin sa \"Third-party\" plugin author o sa hosting platform e.g. GitHub/Codeberg na gumawa ng aksyon. Ang nakalista sa itaas (\"Third-party\" na may label) ay lahat ng pampubliko/komunidad na pinananatiling mga plugin. Hindi namin sila kinukurusado, kaya hindi kami makakagawa ng anumang aksyon sa kanila.\n\n'; + + @override + String get input_does_not_match_format => + 'Ang input ay hindi tumutugma sa kinakailangang format'; + + @override + String get plugins => 'Mga plugin'; + + @override + String get paste_plugin_download_url => + 'I-paste ang download url o GitHub/Codeberg repo url o direktang link sa .smplug file'; + + @override + String get download_and_install_plugin_from_url => + 'I-download at i-install ang plugin mula sa url'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Nabigo ang pagdagdag ng plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'I-upload ang plugin mula sa file'; + + @override + String get installed => 'Naka-install'; + + @override + String get available_plugins => 'Mga available na plugin'; + + @override + String get configure_plugins => + 'I-configure ang sarili mong metadata provider at mga audio source plugin'; + + @override + String get audio_scrobblers => 'Mga Audio Scrobbler'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Pinagmulan: '; + + @override + String get uncompressed => 'Hindi naka-compress'; + + @override + String get dab_music_source_description => + 'Para sa mga audiophile. Nagbibigay ng de-kalidad/walang loss na audio streams. Tumpak na pagtutugma ng track batay sa ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_tr.dart b/lib/l10n/generated/app_localizations_tr.dart new file mode 100644 index 00000000..c2280f47 --- /dev/null +++ b/lib/l10n/generated/app_localizations_tr.dart @@ -0,0 +1,1573 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Turkish (`tr`). +class AppLocalizationsTr extends AppLocalizations { + AppLocalizationsTr([String locale = 'tr']) : super(locale); + + @override + String get guest => 'Misafir'; + + @override + String get browse => 'Göz at'; + + @override + String get search => 'Ara'; + + @override + String get library => 'Kütüphane'; + + @override + String get lyrics => 'Şarkı sözleri'; + + @override + String get settings => 'Ayarlar'; + + @override + String get genre_categories_filter => + 'Kategorileri veya türleri filtreleyin...'; + + @override + String get genre => 'Tür'; + + @override + String get personalized => 'Kişiselleştirilmiş'; + + @override + String get featured => 'Öne çıkanlar'; + + @override + String get new_releases => 'Yeni çıkanlar'; + + @override + String get songs => 'Şarkılar'; + + @override + String playing_track(Object track) { + return '$track oynatılıyor'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Bu, mevcut kuyruğu temizleyecektir. $track_length parça kaldırılacak\nDevam etmek istiyor musunuz?'; + } + + @override + String get load_more => 'Daha fazlasını yükle'; + + @override + String get playlists => 'Oynatma listeleri'; + + @override + String get artists => 'Sanatçılar'; + + @override + String get albums => 'Albümler'; + + @override + String get tracks => 'Parçalar'; + + @override + String get downloads => 'İndirilenler'; + + @override + String get filter_playlists => 'Oynatma listelerinizi filtreleyin...'; + + @override + String get liked_tracks => 'Beğenilen parçalar'; + + @override + String get liked_tracks_description => 'Beğendiğiniz tüm parçalar'; + + @override + String get playlist => 'Çalma Listesi'; + + @override + String get create_a_playlist => 'Bir oynatma listesi oluştur'; + + @override + String get update_playlist => 'Oynatma listesini güncelle'; + + @override + String get create => 'Oluştur'; + + @override + String get cancel => 'İptal'; + + @override + String get update => 'Güncelle'; + + @override + String get playlist_name => 'Oynatma listesi adı'; + + @override + String get name_of_playlist => 'Oynatma listesinin adı'; + + @override + String get description => 'Açıklama'; + + @override + String get public => 'Halka açık'; + + @override + String get collaborative => 'İşbirliği'; + + @override + String get search_local_tracks => 'Yerel parçaları ara...'; + + @override + String get play => 'Oynat'; + + @override + String get delete => 'Sil'; + + @override + String get none => 'Yok'; + + @override + String get sort_a_z => 'A - Z\'ye göre sırala'; + + @override + String get sort_z_a => 'Z - A\'ya göre sırala'; + + @override + String get sort_artist => 'Sanatçıya göre sırala'; + + @override + String get sort_album => 'Albüme göre sırala'; + + @override + String get sort_duration => 'Süreye göre sırala'; + + @override + String get sort_tracks => 'Parçaları sırala'; + + @override + String currently_downloading(Object tracks_length) { + return 'Şu anda indirilenler ($tracks_length)'; + } + + @override + String get cancel_all => 'Tümünü iptal et'; + + @override + String get filter_artist => 'Sanatçıları filtreleyin...'; + + @override + String followers(Object followers) { + return '$followers Takipçiler'; + } + + @override + String get add_artist_to_blacklist => 'Sanatçıyı kara listeye ekle'; + + @override + String get top_tracks => 'En iyi parçalar'; + + @override + String get fans_also_like => 'Hayranlar ayrıca şunları da beğendi'; + + @override + String get loading => 'Yükleniyor...'; + + @override + String get artist => 'Sanatçı'; + + @override + String get blacklisted => 'Kara listeye alındı'; + + @override + String get following => 'Takip ediliyor'; + + @override + String get follow => 'Takip et'; + + @override + String get artist_url_copied => 'Sanatçı bağlantısı panoya kopyalandı'; + + @override + String added_to_queue(Object tracks) { + return 'Kuyruğa $tracks parçası eklendi'; + } + + @override + String get filter_albums => 'Albümleri filtreleyin...'; + + @override + String get synced => 'Senkronize edildi'; + + @override + String get plain => 'Sade'; + + @override + String get shuffle => 'Karıştır'; + + @override + String get search_tracks => 'Parça ara...'; + + @override + String get released => 'Yayınlandı'; + + @override + String error(Object error) { + return 'Hata $error'; + } + + @override + String get title => 'Başlık'; + + @override + String get time => 'Zaman'; + + @override + String get more_actions => 'Daha fazla eylem'; + + @override + String download_count(Object count) { + return 'İndir ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Oynatma Listesine ekle ($count)'; + } + + @override + String add_count_to_queue(Object count) { + return 'Kuyruğa ekle ($count)'; + } + + @override + String play_count_next(Object count) { + return 'Sonrakini oynat ($count)'; + } + + @override + String get album => 'Albüm'; + + @override + String copied_to_clipboard(Object data) { + return '$data panoya kopyalandı'; + } + + @override + String add_to_following_playlists(Object track) { + return '$track parçasını aşağıdaki oynatma listelerine ekle'; + } + + @override + String get add => 'Ekle'; + + @override + String added_track_to_queue(Object track) { + return '$track kuyruğa eklendi'; + } + + @override + String get add_to_queue => 'Kuyruğa ekle'; + + @override + String track_will_play_next(Object track) { + return '$track bir sonraki çalacak'; + } + + @override + String get play_next => 'Sonrakini oynat'; + + @override + String removed_track_from_queue(Object track) { + return '$track kuyruktan kaldırıldı'; + } + + @override + String get remove_from_queue => 'Kuyruktan kaldır'; + + @override + String get remove_from_favorites => 'Favorilerden kaldır'; + + @override + String get save_as_favorite => 'Favori olarak kaydet'; + + @override + String get add_to_playlist => 'Oynatma listesine ekle'; + + @override + String get remove_from_playlist => 'Oynatma listesinden kaldır'; + + @override + String get add_to_blacklist => 'Kara listeye ekle'; + + @override + String get remove_from_blacklist => 'Kara listeden kaldır'; + + @override + String get share => 'Paylaş'; + + @override + String get mini_player => 'Mini oynatıcı'; + + @override + String get slide_to_seek => 'İleri veya geri arama yapmak için kaydırın'; + + @override + String get shuffle_playlist => 'Oynatma listesini karıştır'; + + @override + String get unshuffle_playlist => 'Oynatma listesinin karışıklığını kaldır'; + + @override + String get previous_track => 'Önceki parça'; + + @override + String get next_track => 'Sonraki parça'; + + @override + String get pause_playback => 'Oynatmayı duraklat'; + + @override + String get resume_playback => 'Oynatmayı sürdür'; + + @override + String get loop_track => 'Döngü parçası'; + + @override + String get no_loop => 'Dönüş Yok'; + + @override + String get repeat_playlist => 'Oynatma listesini tekrarla'; + + @override + String get queue => 'Kuyruk'; + + @override + String get alternative_track_sources => 'Alternatif parça kaynakları'; + + @override + String get download_track => 'Parçayı indir'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks parça kuyrukta'; + } + + @override + String get clear_all => 'Tümünü temizle'; + + @override + String get show_hide_ui_on_hover => + 'Fareyle üzerine gelindiğinde kullanıcı arayüzünü göster/gizle'; + + @override + String get always_on_top => 'Her zaman üstte'; + + @override + String get exit_mini_player => 'Mini oynatıcıdan çık'; + + @override + String get download_location => 'İndirme konumu'; + + @override + String get local_library => 'Yerel kütüphane'; + + @override + String get add_library_location => 'Kütüphaneye ekle'; + + @override + String get remove_library_location => 'Kütüphaneden çıkar'; + + @override + String get account => 'Hesap'; + + @override + String get logout => 'Çıkış yap'; + + @override + String get logout_of_this_account => 'Hesaptan çıkış yap'; + + @override + String get language_region => 'Dil ve bölge'; + + @override + String get language => 'Tercih edilen dil'; + + @override + String get system_default => 'Sistem varsayılanı'; + + @override + String get market_place_region => 'Tercih edilen bölge'; + + @override + String get recommendation_country => 'Tavsiye edilen ülke'; + + @override + String get appearance => 'Görünüm'; + + @override + String get layout_mode => 'Düzen modu'; + + @override + String get override_layout_settings => + 'Duyarlı düzen modu ayarlarını geçersiz kıl'; + + @override + String get adaptive => 'Uyarlanabilir'; + + @override + String get compact => 'Sıkıştırılmış'; + + @override + String get extended => 'Genişletilmiş'; + + @override + String get theme => 'Tema'; + + @override + String get dark => 'Koyu'; + + @override + String get light => 'Açık'; + + @override + String get system => 'Sistem'; + + @override + String get accent_color => 'Vurgu rengi'; + + @override + String get sync_album_color => 'Albüm rengini senkronize et'; + + @override + String get sync_album_color_description => + 'Vurgu rengi olarak albüm resminin baskın rengini kullanır'; + + @override + String get playback => 'Oynatma'; + + @override + String get audio_quality => 'Ses kalitesi'; + + @override + String get high => 'Yüksek'; + + @override + String get low => 'Düşük'; + + @override + String get pre_download_play => 'Önceden indir ve oynat'; + + @override + String get pre_download_play_description => + 'Ses akışı yerine baytları indir ve oynat (Daha yüksek bant genişliğine sahip kullanıcılar için önerilir)'; + + @override + String get skip_non_music => 'Müzik olmayan bölümleri atlat (SponsorBlock)'; + + @override + String get blacklist_description => + 'Kara listeye alınan parçalar ve sanatçılar'; + + @override + String get wait_for_download_to_finish => + 'Lütfen mevcut indirme işleminin tamamlanmasını bekleyin'; + + @override + String get desktop => 'Masaüstü'; + + @override + String get close_behavior => 'Kapatma davranışı'; + + @override + String get close => 'Kapat'; + + @override + String get minimize_to_tray => 'Tepsiye küçült'; + + @override + String get show_tray_icon => 'Sistem tepsisi simgesini göster'; + + @override + String get about => 'Hakkında'; + + @override + String get u_love_spotube => 'Spotube\'u sevdiğinizi biliyoruz'; + + @override + String get check_for_updates => 'Güncellemeleri kontrol et'; + + @override + String get about_spotube => 'Spotube hakkında'; + + @override + String get blacklist => 'Kara liste'; + + @override + String get please_sponsor => 'Sponsor Ol/Bağış Yap'; + + @override + String get spotube_description => + 'Spotube, hafif, platformlar arası uyumlu ve herkes için ücretsiz bir Spotify istemcisidir.'; + + @override + String get version => 'Sürüm'; + + @override + String get build_number => 'Derleme numarası'; + + @override + String get founder => 'Geliştirici'; + + @override + String get repository => 'Depo'; + + @override + String get bug_issues => 'Hata + Sorunlar'; + + @override + String get made_with => '❤️ ile Bangladeş\'te yapıldı'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Lisans'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Endişelenmeyin, kimlik bilgilerinizden hiçbiri toplanmayacak veya kimseyle paylaşılmayacak'; + + @override + String get know_how_to_login => 'Bunu nasıl yapacağınızı bilmiyor musunuz?'; + + @override + String get follow_step_by_step_guide => 'Adım adım kılavuzu takip edin'; + + @override + String cookie_name_cookie(Object name) { + return '$name çerezi'; + } + + @override + String get fill_in_all_fields => 'Lütfen tüm alanları doldurun'; + + @override + String get submit => 'Başvur'; + + @override + String get exit => 'Çık'; + + @override + String get previous => 'Önceki'; + + @override + String get next => 'Sonraki'; + + @override + String get done => 'Bitti'; + + @override + String get step_1 => '1. Adım'; + + @override + String get first_go_to => 'İlk olarak şuraya gidin:'; + + @override + String get something_went_wrong => 'Bir hata oluştu'; + + @override + String get piped_instance => 'Piped sunucu örneği'; + + @override + String get piped_description => + 'Parça eşleştirme için kullanılacak Piped sunucu örneği'; + + @override + String get piped_warning => + 'Bazıları iyi çalışmayabilir. Yani riski size ait olmak üzere kullanın'; + + @override + String get invidious_instance => 'Invidious Sunucu Örneği'; + + @override + String get invidious_description => + 'Parça eşleştirmesi için kullanılacak Invidious sunucu örneği'; + + @override + String get invidious_warning => + 'Bazıları iyi çalışmayabilir. Kendi riskinizde kullanın'; + + @override + String get generate => 'Oluştur'; + + @override + String track_exists(Object track) { + return '$track parçası zaten var'; + } + + @override + String get replace_downloaded_tracks => 'İndirilen tüm parçaları değiştir'; + + @override + String get skip_download_tracks => 'İndirilen tüm parçaları indirmeyi atla'; + + @override + String get do_you_want_to_replace => + 'Mevcut parçayı değiştirmek istiyor musunuz?'; + + @override + String get replace => 'Değiştir'; + + @override + String get skip => 'Atla'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'En fazla $count $type seçin'; + } + + @override + String get select_genres => 'Türleri seç'; + + @override + String get add_genres => 'Tür ekle'; + + @override + String get country => 'Ülke'; + + @override + String get number_of_tracks_generate => 'Oluşturulacak parça sayısı'; + + @override + String get acousticness => 'Akustiklik'; + + @override + String get danceability => 'Dans Edilebilirlik'; + + @override + String get energy => 'Enerji'; + + @override + String get instrumentalness => 'Araçsallık'; + + @override + String get liveness => 'Canlılık'; + + @override + String get loudness => 'Ses yüksekliği'; + + @override + String get speechiness => 'Konuşkanlık'; + + @override + String get valence => 'Değerlik'; + + @override + String get popularity => 'Popülerlik'; + + @override + String get key => 'Anahtar'; + + @override + String get duration => 'Süre (sn)'; + + @override + String get tempo => 'Tempo (BPM)'; + + @override + String get mode => 'Mod'; + + @override + String get time_signature => 'Zaman imzası'; + + @override + String get short => 'Kısa'; + + @override + String get medium => 'Orta'; + + @override + String get long => 'Uzun'; + + @override + String get min => 'Min'; + + @override + String get max => 'Maks'; + + @override + String get target => 'Hedef'; + + @override + String get moderate => 'Orta'; + + @override + String get deselect_all => 'Tüm seçimleri kaldır'; + + @override + String get select_all => 'Tümünü seç'; + + @override + String get are_you_sure => 'Emin misiniz?'; + + @override + String get generating_playlist => 'Özel oynatma listeniz oluşturuluyor...'; + + @override + String selected_count_tracks(Object count) { + return '$count parça seçildi'; + } + + @override + String get download_warning => + 'Tüm şarkıları toplu olarak indiriyorsanız, açıkça müzik korsanlığı yapıyorsunuz ve müzik dünyasının yaratıcı topluluğuna zarar veriyorsunuz demektir. Umuyorum bunun farkındasınızdır. Her zaman, sanatçıların emeğine saygı göstermeyi ve desteklemeyi deneyin.'; + + @override + String get download_ip_ban_warning => + 'Ayrıca, normalden fazla indirme istekleri nedeniyle YouTube\'da IP\'niz engellenebilir. IP engeli, en az 2-3 ay boyunca YouTube\'u (hatta oturum açmış olsanız bile) o IP cihazından kullanamayacağınız anlamına gelir. Ve eğer böyle bir durum yaşanırsa, Spotube bundan hiçbir sorumluluk kabul etmez.'; + + @override + String get by_clicking_accept_terms => + '\"Kabul et\" e tıklayarak aşağıdaki şartları kabul etmiş olursunuz:'; + + @override + String get download_agreement_1 => + 'Müzik korsanlığı yaptığımı biliyorum. Ben fakir biriyim.'; + + @override + String get download_agreement_2 => + 'Sanatçıyı elimden geldiğince destekleyeceğim ve bunu sadece sanatını satın alacak param olmadığı için yapıyorum'; + + @override + String get download_agreement_3 => + 'YouTube\'da IP\'min engellenebileceğinin tamamen farkındayım ve mevcut eylemlerimden kaynaklanan herhangi bir kaza için Spotube\'u veya sahiplerini/katkıda bulunanları sorumlu tutmuyorum.'; + + @override + String get decline => 'Reddet'; + + @override + String get accept => 'Kabul et'; + + @override + String get details => 'Detaylar'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kanal'; + + @override + String get likes => 'Beğenenler'; + + @override + String get dislikes => 'Beğenmeyenler'; + + @override + String get views => 'İzlenmeler'; + + @override + String get streamUrl => 'Akış bağlantısı'; + + @override + String get stop => 'Durdur'; + + @override + String get sort_newest => 'En yeni eklenene göre sırala.'; + + @override + String get sort_oldest => 'En eski eklenene göre sırala'; + + @override + String get sleep_timer => 'Uyku Zamanlayıcısı'; + + @override + String mins(Object minutes) { + return '$minutes Dakika'; + } + + @override + String hours(Object hours) { + return '$hours Saatler'; + } + + @override + String hour(Object hours) { + return '$hours Saat'; + } + + @override + String get custom_hours => 'Özel Saatler'; + + @override + String get logs => 'Günlükler'; + + @override + String get developers => 'Geliştiriciler'; + + @override + String get not_logged_in => 'Giriş yapmadınız'; + + @override + String get search_mode => 'Arama modu'; + + @override + String get audio_source => 'Ses kaynağı'; + + @override + String get ok => 'Tamam'; + + @override + String get failed_to_encrypt => 'Şifreleme başarısız oldu'; + + @override + String get encryption_failed_warning => + 'Spotube, verilerinizi güvenli bir şekilde depolamak için şifreleme kullanır. Ancak bunu başaramadı. Bu nedenle, güvensiz depolamaya geri dönecektir\nLinux kullanıyorsanız, lütfen gnome-keyring, kde-wallet, keepassxc vb. herhangi bir gizli servisin yüklü olduğundan emin olun.'; + + @override + String get querying_info => 'Bilgi sorgulanıyor...'; + + @override + String get piped_api_down => 'Piped API kapalı'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Piped örneği $pipedInstance şu anda kapalı\n\nÖrneği değiştirin veya \'API türünü\' resmi YouTube API\'si olarak değiştirin\n\nDeğişiklikten sonra uygulamayı yeniden başlattığınızdan emin olun'; + } + + @override + String get you_are_offline => 'Şu anda çevrimdışısınız'; + + @override + String get connection_restored => 'İnternet bağlantınız geri yüklendi'; + + @override + String get use_system_title_bar => 'Sistem başlık çubuğunu kullan'; + + @override + String get crunching_results => 'Sonuçlar...'; + + @override + String get search_to_get_results => 'Sonuç almak için arayın'; + + @override + String get use_amoled_mode => 'AMOLED modu kullan'; + + @override + String get pitch_dark_theme => 'Zifiri karanlık koyu tema'; + + @override + String get normalize_audio => 'Sesi normalleştir'; + + @override + String get change_cover => 'Kapağı değiştir'; + + @override + String get add_cover => 'Kapak ekle'; + + @override + String get restore_defaults => 'Varsayılanları geri yükle'; + + @override + String get download_music_format => 'Müzik indirme formatı'; + + @override + String get streaming_music_format => 'Müzik akış formatı'; + + @override + String get download_music_quality => 'İndirilen müzik kalitesi'; + + @override + String get streaming_music_quality => 'Yayınlanan müzik kalitesi'; + + @override + String get login_with_lastfm => 'Last.fm ile giriş yap'; + + @override + String get connect => 'Bağlan'; + + @override + String get disconnect_lastfm => 'Last.fm bağlantısını kes'; + + @override + String get disconnect => 'Bağlantıyı kes'; + + @override + String get username => 'Kullanıcı adı'; + + @override + String get password => 'Şifre'; + + @override + String get login => 'Giriş yap'; + + @override + String get login_with_your_lastfm => 'Last.fm hesabınızla giriş yapın'; + + @override + String get scrobble_to_lastfm => 'Last.fm için Scrobble'; + + @override + String get go_to_album => 'Albüme git'; + + @override + String get discord_rich_presence => 'Discord zengin varlığı'; + + @override + String get browse_all => 'Tümüne göz at'; + + @override + String get genres => 'Müzik türleri'; + + @override + String get explore_genres => 'Türleri keşfet'; + + @override + String get friends => 'Arkadaşlar'; + + @override + String get no_lyrics_available => 'Üzgünüz, bu parçanın sözleri bulunamıyor'; + + @override + String get start_a_radio => 'Radyo başlat'; + + @override + String get how_to_start_radio => 'Radyoyu nasıl başlatmak istersiniz?'; + + @override + String get replace_queue_question => + 'Mevcut kuyruğu değiştirmek mi yoksa eklemek mi istersiniz?'; + + @override + String get endless_playback => 'Sonsuz olarak oynat'; + + @override + String get delete_playlist => 'Oynatma listesini sil'; + + @override + String get delete_playlist_confirmation => + 'Bu oynatma listesini silmek istediğinizden emin misiniz?'; + + @override + String get local_tracks => 'Yerel parçalar'; + + @override + String get local_tab => 'Yerel'; + + @override + String get song_link => 'Şarkı bağlantısı'; + + @override + String get skip_this_nonsense => 'Bu saçmalığı atla'; + + @override + String get freedom_of_music => '“Müzik özgürlüğü”'; + + @override + String get freedom_of_music_palm => '“Müzik özgürlüğü avucunuzun içinde”'; + + @override + String get get_started => 'Haydi başlayalım'; + + @override + String get youtube_source_description => + 'Tavsiye edilir ve en iyi şekilde çalışır.'; + + @override + String get piped_source_description => + 'Özgür hissediyor musunuz? YouTube ile aynı, ama çok daha özgür.'; + + @override + String get jiosaavn_source_description => 'Güney Asya bölgesi için en iyisi.'; + + @override + String get invidious_source_description => + 'Piped\'a benzer, ancak daha yüksek kullanılabilirliğe sahip.'; + + @override + String highest_quality(Object quality) { + return 'En yüksek kalite: $quality'; + } + + @override + String get select_audio_source => 'Ses kaynağını seçin'; + + @override + String get endless_playback_description => + 'Yeni şarkıları otomatik olarak\nkuyruğun sonuna ekle'; + + @override + String get choose_your_region => 'Bölgenizi seçin'; + + @override + String get choose_your_region_description => + 'Bu, Spotube\'un konumunuza uygun içerikleri göstermesine yardımcı olacaktır.'; + + @override + String get choose_your_language => 'Dilinizi seçin'; + + @override + String get help_project_grow => 'Bu projenin büyümesine yardımcı olun'; + + @override + String get help_project_grow_description => + 'Spotube açık kaynaklı bir projedir. Projeye katkıda bulunarak, hataları bildirerek veya yeni özellikler önererek bu projenin büyümesine yardımcı olabilirsiniz.'; + + @override + String get contribute_on_github => 'GitHub\'da katkıda bulun'; + + @override + String get donate_on_open_collective => 'Open Collective\'de bağış yap'; + + @override + String get browse_anonymously => 'Anonim olarak giriş yap'; + + @override + String get enable_connect => 'Bağlanmayı etkinleştir'; + + @override + String get enable_connect_description => + 'Spotube\'u diğer cihazlardan kontrol edin'; + + @override + String get devices => 'Cihazlar'; + + @override + String get select => 'Seç'; + + @override + String connect_client_alert(Object client) { + return '$client tarafından kontrol ediliyorsun.'; + } + + @override + String get this_device => 'Bu cihaz'; + + @override + String get remote => 'Yönet'; + + @override + String get stats => 'İstatistikler'; + + @override + String and_n_more(Object count) { + return 've $count daha'; + } + + @override + String get recently_played => 'Son Çalınanlar'; + + @override + String get browse_more => 'Daha Fazla Göz At'; + + @override + String get no_title => 'Başlık Yok'; + + @override + String get not_playing => 'Çalmıyor'; + + @override + String get epic_failure => 'Efsanevi başarısızlık!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '$tracks_length şarkı sıraya eklendi'; + } + + @override + String get spotube_has_an_update => 'Spotube bir güncelleme aldı'; + + @override + String get download_now => 'Şimdi İndir'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum yayımlandı'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version yayımlandı'; + } + + @override + String get read_the_latest => 'Son haberleri oku'; + + @override + String get release_notes => 'sürüm notları'; + + @override + String get pick_color_scheme => 'Renk şeması seç'; + + @override + String get save => 'Kaydet'; + + @override + String get choose_the_device => 'Cihazı seçin:'; + + @override + String get multiple_device_connected => + 'Birden fazla cihaz bağlı.\nBu işlemi gerçekleştirmek istediğiniz cihazı seçin'; + + @override + String get nothing_found => 'Hiçbir şey bulunamadı'; + + @override + String get the_box_is_empty => 'Kutu boş'; + + @override + String get top_artists => 'En İyi Sanatçılar'; + + @override + String get top_albums => 'En İyi Albümler'; + + @override + String get this_week => 'Bu hafta'; + + @override + String get this_month => 'Bu ay'; + + @override + String get last_6_months => 'Son 6 ay'; + + @override + String get this_year => 'Bu yıl'; + + @override + String get last_2_years => 'Son 2 yıl'; + + @override + String get all_time => 'Tüm zamanlar'; + + @override + String powered_by_provider(Object providerName) { + return '$providerName tarafından desteklenmektedir'; + } + + @override + String get email => 'E-posta'; + + @override + String get profile_followers => 'Takipçiler'; + + @override + String get birthday => 'Doğum Günü'; + + @override + String get subscription => 'Abonelik'; + + @override + String get not_born => 'Henüz doğmadı'; + + @override + String get hacker => 'Hacker'; + + @override + String get profile => 'Profil'; + + @override + String get no_name => 'İsim Yok'; + + @override + String get edit => 'Düzenle'; + + @override + String get user_profile => 'Kullanıcı Profili'; + + @override + String count_plays(Object count) { + return '$count çalma'; + } + + @override + String get streaming_fees_hypothetical => + '*Spotify\'ın akış başına ödeme miktarına\n\$0.003 ile \$0.005 arasında hesaplanmıştır. Bu, kullanıcıya\nSpotify\'da şarkılarını dinlerse sanatçılara ne kadar ödeme\nyapmış olabileceğini göstermek için hipotetik bir hesaplamadır.'; + + @override + String get minutes_listened => 'Dinlenilen Dakikalar'; + + @override + String get streamed_songs => 'Yayınlanan Şarkılar'; + + @override + String count_streams(Object count) { + return '$count yayın'; + } + + @override + String get owned_by_you => 'Sahip olduğunuz'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl panoya kopyalandı'; + } + + @override + String get hipotetical_calculation => + '*Bu, çevrimiçi müzik akışı platformlarının ortalama akış başına \$0,003 ile \$0,005 arasındaki ödemesine göre hesaplanmıştır. Bu, kullanıcının farklı müzik akışı platformlarında şarkılarını dinleselerdi sanatçılara ne kadar ödeme yapacaklarına dair fikir vermek için yapılan varsayımsal bir hesaplamadır.'; + + @override + String count_mins(Object minutes) { + return '$minutes dk'; + } + + @override + String get summary_minutes => 'dakika'; + + @override + String get summary_listened_to_music => 'Dinlenen müzik'; + + @override + String get summary_songs => 'şarkılar'; + + @override + String get summary_streamed_overall => 'Genel olarak akış'; + + @override + String get summary_owed_to_artists => 'Sanatçılara borç\nbu ay'; + + @override + String get summary_artists => 'sanatçının'; + + @override + String get summary_music_reached_you => 'Müzik sana ulaştı'; + + @override + String get summary_full_albums => 'tam albümler'; + + @override + String get summary_got_your_love => 'Sevgini aldı'; + + @override + String get summary_playlists => 'çalma listeleri'; + + @override + String get summary_were_on_repeat => 'Tekrarda vardı'; + + @override + String total_money(Object money) { + return 'Toplam $money'; + } + + @override + String get webview_not_found => 'Webview bulunamadı'; + + @override + String get webview_not_found_description => + 'Cihazınızda herhangi bir Webview çalışma zamanı yüklü değil.\nEğer kuruluysa, ortam YOLUNDA olduğundan emin olun\n\nKurulumdan sonra uygulamayı yeniden başlatın'; + + @override + String get unsupported_platform => 'Desteklenmeyen platform'; + + @override + String get cache_music => 'Müziği önbellekle'; + + @override + String get open => 'Aç'; + + @override + String get cache_folder => 'Önbellek klasörü'; + + @override + String get export => 'Dışa aktar'; + + @override + String get clear_cache => 'Önbelleği temizle'; + + @override + String get clear_cache_confirmation => + 'Önbelleği temizlemek istiyor musunuz?'; + + @override + String get export_cache_files => 'Önbelleğe Alınmış Dosyaları Dışa Aktar'; + + @override + String found_n_files(Object count) { + return '$count dosya bulundu'; + } + + @override + String get export_cache_confirmation => + 'Bu dosyaları dışa aktarmak istiyor musunuz'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '$filesExported / $files dosya dışa aktarıldı'; + } + + @override + String get undo => 'Geri Al'; + + @override + String get download_all => 'Tümünü İndir'; + + @override + String get add_all_to_playlist => 'Hepsini çalma listesine ekle'; + + @override + String get add_all_to_queue => 'Hepsini kuyruğa ekle'; + + @override + String get play_all_next => 'Hepsini bir sonraki çal'; + + @override + String get pause => 'Duraklat'; + + @override + String get view_all => 'Tümünü Gör'; + + @override + String get no_tracks_added_yet => + 'Henüz hiçbir şarkı eklemediniz gibi görünüyor'; + + @override + String get no_tracks => 'Burada hiç şarkı yok gibi görünüyor'; + + @override + String get no_tracks_listened_yet => + 'Henüz hiçbir şey dinlemediniz gibi görünüyor'; + + @override + String get not_following_artists => 'Hiçbir sanatçıyı takip etmiyorsunuz'; + + @override + String get no_favorite_albums_yet => + 'Henüz favorilerinize herhangi bir albüm eklemediniz gibi görünüyor'; + + @override + String get no_logs_found => 'Log bulunamadı'; + + @override + String get youtube_engine => 'YouTube Motoru'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine Yüklü değil'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine sisteminizde yüklü değil.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'PATH değişkeninde kullanılabilir olduğundan emin olun veya\n$engine çalıştırılabilir dosyasının mutlak yolunu aşağıda ayarlayın'; + } + + @override + String get youtube_engine_unix_issue_message => + 'macOS/Linux/Unix benzeri işletim sistemlerinde, .zshrc/.bashrc/.bash_profile gibi dosyalarda yol ayarlamak işe yaramaz.\nYolunuzu kabuk yapılandırma dosyasına ayarlamanız gerekir'; + + @override + String get download => 'İndir'; + + @override + String get file_not_found => 'Dosya bulunamadı'; + + @override + String get custom => 'Özel'; + + @override + String get add_custom_url => 'Özel URL ekle'; + + @override + String get edit_port => 'Portu düzenle'; + + @override + String get port_helper_msg => + 'Varsayılan -1\'dir, bu da rastgele bir sayıyı gösterir. Bir güvenlik duvarınız varsa, bunu ayarlamanız önerilir.'; + + @override + String connect_request(Object client) { + return '$client bağlantısına izin verilsin mi?'; + } + + @override + String get connection_request_denied => + 'Bağlantı reddedildi. Kullanıcı erişimi reddetti.'; + + @override + String get an_error_occurred => 'Bir hata oluştu'; + + @override + String get copy_to_clipboard => 'Panoya kopyala'; + + @override + String get view_logs => 'Günlükleri görüntüle'; + + @override + String get retry => 'Tekrar dene'; + + @override + String get no_default_metadata_provider_selected => + 'Varsayılan bir meta veri sağlayıcısı ayarlanmadı'; + + @override + String get manage_metadata_providers => 'Meta veri sağlayıcılarını yönet'; + + @override + String get open_link_in_browser => 'Bağlantıyı Tarayıcıda Aç?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Aşağıdaki bağlantıyı açmak istiyor musunuz'; + + @override + String get unsafe_url_warning => + 'Güvenilmeyen kaynaklardan bağlantı açmak güvensiz olabilir. Dikkatli olun!\nBağlantıyı panonuza da kopyalayabilirsiniz.'; + + @override + String get copy_link => 'Bağlantıyı Kopyala'; + + @override + String get building_your_timeline => + 'Dinlemelerinize göre zaman çizelgeniz oluşturuluyor...'; + + @override + String get official => 'Resmi'; + + @override + String author_name(Object author) { + return 'Yazar: $author'; + } + + @override + String get third_party => 'Üçüncü taraf'; + + @override + String get plugin_requires_authentication => + 'Eklenti kimlik doğrulama gerektirir'; + + @override + String get update_available => 'Güncelleme mevcut'; + + @override + String get supports_scrobbling => 'Scrobbling\'i destekler'; + + @override + String get plugin_scrobbling_info => + 'Bu eklenti, dinleme geçmişinizi oluşturmak için müziğinizi scrobble eder.'; + + @override + String get default_metadata_source => 'Varsayılan meta veri kaynağı'; + + @override + String get set_default_metadata_source => + 'Varsayılan meta veri kaynağını ayarla'; + + @override + String get default_audio_source => 'Varsayılan ses kaynağı'; + + @override + String get set_default_audio_source => 'Varsayılan ses kaynağını ayarla'; + + @override + String get set_default => 'Varsayılan olarak ayarla'; + + @override + String get support => 'Destek'; + + @override + String get support_plugin_development => 'Eklenti geliştirmeyi destekle'; + + @override + String can_access_name_api(Object name) { + return '- **$name** API\'ye erişebilir'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Bu eklentiyi yüklemek istiyor musunuz?'; + + @override + String get third_party_plugin_warning => + 'Bu eklenti üçüncü taraf bir depodan gelmektedir. Lütfen yüklemeden önce kaynağa güvendiğinizden emin olun.'; + + @override + String get author => 'Yazar'; + + @override + String get this_plugin_can_do_following => + 'Bu eklenti aşağıdakileri yapabilir'; + + @override + String get install => 'Yükle'; + + @override + String get install_a_metadata_provider => 'Bir Meta Veri Sağlayıcısı Yükle'; + + @override + String get no_tracks_playing => 'Şu anda çalınan bir Parça yok'; + + @override + String get synced_lyrics_not_available => + 'Bu şarkı için senkronize şarkı sözleri mevcut değil. Lütfen'; + + @override + String get plain_lyrics => 'Düz Şarkı Sözleri'; + + @override + String get tab_instead => 'sekmesini kullanın.'; + + @override + String get disclaimer => 'Sorumluluk Reddi'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube ekibi, herhangi bir \"Üçüncü taraf\" eklentisi için herhangi bir sorumluluk (yasal olanlar dahil) kabul etmez.\nLütfen bunları kendi riskinizde kullanın. Herhangi bir hata/sorun için lütfen bunları eklenti deposuna bildirin.\n\nHerhangi bir \"Üçüncü taraf\" eklentisi bir hizmetin/yasal varlığın ToS/DMCA\'sını ihlal ediyorsa, lütfen \"Üçüncü taraf\" eklenti yazarından veya barındırma platformundan, örneğin GitHub/Codeberg\'den harekete geçmesini isteyin. Yukarıda listelenen (\"Üçüncü taraf\" olarak etiketlenen) eklentilerin tümü genel/topluluk tarafından sürdürülen eklentilerdir. Biz bunları küratörlüğünü yapmıyoruz, bu yüzden onlar üzerinde herhangi bir işlem yapamayız.\n\n'; + + @override + String get input_does_not_match_format => 'Girdi, gerekli biçimle eşleşmiyor'; + + @override + String get plugins => 'Eklentiler'; + + @override + String get paste_plugin_download_url => + 'İndirme url\'sini veya GitHub/Codeberg repo url\'sini veya .smplug dosyasına doğrudan bağlantıyı yapıştırın'; + + @override + String get download_and_install_plugin_from_url => + 'url\'den eklentiyi indir ve yükle'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Eklenti eklenemedi: $error'; + } + + @override + String get upload_plugin_from_file => 'Dosyadan eklenti yükle'; + + @override + String get installed => 'Yüklü'; + + @override + String get available_plugins => 'Mevcut eklentiler'; + + @override + String get configure_plugins => + 'Kendi meta veri sağlayıcı ve ses kaynağı eklentilerinizi yapılandırın'; + + @override + String get audio_scrobblers => 'Ses Scrobbler\'lar'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Kaynak: '; + + @override + String get uncompressed => 'Sıkıştırılmamış'; + + @override + String get dab_music_source_description => + 'Audiophile\'ler için. Yüksek kaliteli/kayıpsız ses akışları sağlar. Doğru ISRC tabanlı parça eşleştirme.'; +} diff --git a/lib/l10n/generated/app_localizations_uk.dart b/lib/l10n/generated/app_localizations_uk.dart new file mode 100644 index 00000000..c2bed426 --- /dev/null +++ b/lib/l10n/generated/app_localizations_uk.dart @@ -0,0 +1,1570 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Ukrainian (`uk`). +class AppLocalizationsUk extends AppLocalizations { + AppLocalizationsUk([String locale = 'uk']) : super(locale); + + @override + String get guest => 'Гість'; + + @override + String get browse => 'Огляд'; + + @override + String get search => 'Пошук'; + + @override + String get library => 'Медіатека'; + + @override + String get lyrics => 'Тексти пісень'; + + @override + String get settings => 'Налаштування'; + + @override + String get genre_categories_filter => 'Фільтрувати категорії або жанри...'; + + @override + String get genre => 'Жанр'; + + @override + String get personalized => 'Персоналізовані'; + + @override + String get featured => 'Рекомендовані'; + + @override + String get new_releases => 'Нові релізи'; + + @override + String get songs => 'Пісні'; + + @override + String playing_track(Object track) { + return 'Відтворюється $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Це очистить поточну чергу. Буде видалено $track_length треків\nПродовжити?'; + } + + @override + String get load_more => 'Завантажити більше'; + + @override + String get playlists => 'Плейлисти'; + + @override + String get artists => 'Виконавці'; + + @override + String get albums => 'Альбоми'; + + @override + String get tracks => 'Треки'; + + @override + String get downloads => 'Завантаження'; + + @override + String get filter_playlists => 'Фільтрувати плейлисти...'; + + @override + String get liked_tracks => 'Сподобалися треки'; + + @override + String get liked_tracks_description => 'Усі ваші сподобалися треки'; + + @override + String get playlist => 'Плейлист'; + + @override + String get create_a_playlist => 'Створити плейлист'; + + @override + String get update_playlist => 'Оновити плейлист'; + + @override + String get create => 'Створити'; + + @override + String get cancel => 'Скасувати'; + + @override + String get update => 'Оновити'; + + @override + String get playlist_name => 'Назва плейлиста'; + + @override + String get name_of_playlist => 'Назва плейлиста'; + + @override + String get description => 'Опис'; + + @override + String get public => 'Публічний'; + + @override + String get collaborative => 'Спільний'; + + @override + String get search_local_tracks => 'Пошук локальних треків...'; + + @override + String get play => 'Відтворити'; + + @override + String get delete => 'Видалити'; + + @override + String get none => 'Немає'; + + @override + String get sort_a_z => 'Сортувати за алфавітом A-Я'; + + @override + String get sort_z_a => 'Сортувати за алфавітом Я-А'; + + @override + String get sort_artist => 'Сортувати за виконавцем'; + + @override + String get sort_album => 'Сортувати за альбомом'; + + @override + String get sort_duration => 'Сортувати за тривалістю'; + + @override + String get sort_tracks => 'Сортувати треки'; + + @override + String currently_downloading(Object tracks_length) { + return 'Завантажується ($tracks_length)'; + } + + @override + String get cancel_all => 'Скасувати все'; + + @override + String get filter_artist => 'Фільтрувати виконавців...'; + + @override + String followers(Object followers) { + return '$followers підписників'; + } + + @override + String get add_artist_to_blacklist => 'Додати виконавця до чорного списку'; + + @override + String get top_tracks => 'Топ треки'; + + @override + String get fans_also_like => 'Шанувальникам також подобається'; + + @override + String get loading => 'Завантаження...'; + + @override + String get artist => 'Виконавець'; + + @override + String get blacklisted => 'У чорному списку'; + + @override + String get following => 'Стежу'; + + @override + String get follow => 'Стежити'; + + @override + String get artist_url_copied => 'URL виконавця скопійовано до буфера обміну'; + + @override + String added_to_queue(Object tracks) { + return 'Додано $tracks треків до черги'; + } + + @override + String get filter_albums => 'Фільтрувати альбоми...'; + + @override + String get synced => 'Синхронізовано'; + + @override + String get plain => 'Звичайний'; + + @override + String get shuffle => 'Випадковий порядок'; + + @override + String get search_tracks => 'Пошук треків...'; + + @override + String get released => 'Випущено'; + + @override + String error(Object error) { + return 'Помилка $error'; + } + + @override + String get title => 'Назва'; + + @override + String get time => 'Час'; + + @override + String get more_actions => 'Більше дій'; + + @override + String download_count(Object count) { + return 'Завантажено ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Додати ($count) до плейлиста'; + } + + @override + String add_count_to_queue(Object count) { + return 'Додати ($count) до черги'; + } + + @override + String play_count_next(Object count) { + return 'Відтворити ($count) наступними'; + } + + @override + String get album => 'Альбом'; + + @override + String copied_to_clipboard(Object data) { + return 'Скопійовано $data до буфера обміну'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Додати $track до наступних плейлистів'; + } + + @override + String get add => 'Додати'; + + @override + String added_track_to_queue(Object track) { + return 'Додано $track до черги'; + } + + @override + String get add_to_queue => 'Додати до черги'; + + @override + String track_will_play_next(Object track) { + return '$track буде відтворено наступним'; + } + + @override + String get play_next => 'Відтворити наступним'; + + @override + String removed_track_from_queue(Object track) { + return 'Видалено $track з черги'; + } + + @override + String get remove_from_queue => 'Видалити з черги'; + + @override + String get remove_from_favorites => 'Видалити з обраних'; + + @override + String get save_as_favorite => 'Зберегти як обране'; + + @override + String get add_to_playlist => 'Додати до плейлиста'; + + @override + String get remove_from_playlist => 'Видалити з плейлиста'; + + @override + String get add_to_blacklist => 'Додати до чорного списку'; + + @override + String get remove_from_blacklist => 'Видалити з чорного списку'; + + @override + String get share => 'Поділитися'; + + @override + String get mini_player => 'Міні-плеєр'; + + @override + String get slide_to_seek => + 'Проведіть пальцем, щоб перемотати вперед або назад'; + + @override + String get shuffle_playlist => 'Випадковий порядок відтворення плейлиста'; + + @override + String get unshuffle_playlist => + 'Відключити випадковий порядок відтворення плейлиста'; + + @override + String get previous_track => 'Попередній трек'; + + @override + String get next_track => 'Наступний трек'; + + @override + String get pause_playback => 'Призупинити відтворення'; + + @override + String get resume_playback => 'Відновити відтворення'; + + @override + String get loop_track => 'Повторювати трек'; + + @override + String get no_loop => 'Без повтору'; + + @override + String get repeat_playlist => 'Повторювати плейлист'; + + @override + String get queue => 'Черга'; + + @override + String get alternative_track_sources => 'Альтернативні джерела треків'; + + @override + String get download_track => 'Завантажити трек'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks треків у черзі'; + } + + @override + String get clear_all => 'Очистити все'; + + @override + String get show_hide_ui_on_hover => + 'Показувати/приховувати інтерфейс при наведенні курсору'; + + @override + String get always_on_top => 'Завжди зверху'; + + @override + String get exit_mini_player => 'Вийти з міні-плеєра'; + + @override + String get download_location => 'Шлях завантаження'; + + @override + String get local_library => 'Місцева бібліотека'; + + @override + String get add_library_location => 'Додати до бібліотеки'; + + @override + String get remove_library_location => 'Видалити з бібліотеки'; + + @override + String get account => 'Обліковий запис'; + + @override + String get logout => 'Вийти'; + + @override + String get logout_of_this_account => 'Вийти з цього облікового запису'; + + @override + String get language_region => 'Мова та регіон'; + + @override + String get language => 'Мова'; + + @override + String get system_default => 'Системна мова'; + + @override + String get market_place_region => 'Регіон маркетплейсу'; + + @override + String get recommendation_country => 'Країна рекомендацій'; + + @override + String get appearance => 'Зовнішній вигляд'; + + @override + String get layout_mode => 'Режим макета'; + + @override + String get override_layout_settings => + 'Перезаписати налаштування адаптивного режиму макета'; + + @override + String get adaptive => 'Адаптивний'; + + @override + String get compact => 'Компактний'; + + @override + String get extended => 'Розширений'; + + @override + String get theme => 'Тема'; + + @override + String get dark => 'Темна'; + + @override + String get light => 'Світла'; + + @override + String get system => 'Системна'; + + @override + String get accent_color => 'Колір акценту'; + + @override + String get sync_album_color => 'Синхронізувати колір альбому'; + + @override + String get sync_album_color_description => + 'Використовує домінуючий колір обкладинки альбому як колір акценту'; + + @override + String get playback => 'Відтворення'; + + @override + String get audio_quality => 'Якість аудіо'; + + @override + String get high => 'Висока'; + + @override + String get low => 'Низька'; + + @override + String get pre_download_play => 'Попереднє завантаження та відтворення'; + + @override + String get pre_download_play_description => + 'Замість потокового відтворення аудіо завантажте байти та відтворіть їх (рекомендовано для користувачів з високою пропускною здатністю)'; + + @override + String get skip_non_music => 'Пропустити не музичні сегменти'; + + @override + String get blacklist_description => 'Треки та виконавці в чорному списку'; + + @override + String get wait_for_download_to_finish => + 'Зачекайте, поки завершиться поточна загрузка'; + + @override + String get desktop => 'Робочий стіл'; + + @override + String get close_behavior => 'Поведінка при закритті'; + + @override + String get close => 'Закрити'; + + @override + String get minimize_to_tray => 'Згорнути в трей'; + + @override + String get show_tray_icon => 'Показувати значок у системному треї'; + + @override + String get about => 'Про'; + + @override + String get u_love_spotube => 'Ми знаємо, що ви любите Spotube'; + + @override + String get check_for_updates => 'Перевірити наявність оновлень'; + + @override + String get about_spotube => 'Про Spotube'; + + @override + String get blacklist => 'Чорний список'; + + @override + String get please_sponsor => 'Будь ласка, станьте спонсором/зробіть пожертву'; + + @override + String get spotube_description => + 'Spotube, легкий, кросплатформовий, безкоштовний клієнт Spotify'; + + @override + String get version => 'Версія'; + + @override + String get build_number => 'Номер збірки'; + + @override + String get founder => 'Засновник'; + + @override + String get repository => 'Репозиторій'; + + @override + String get bug_issues => 'Помилки та проблеми'; + + @override + String get made_with => 'Зроблено з ❤️ в Бангладеш 🇧🇩'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Ліцензія'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Не хвилюйтеся, жодні ваші облікові дані не будуть зібрані або передані кому-небудь'; + + @override + String get know_how_to_login => 'Не знаєте, як це зробити?'; + + @override + String get follow_step_by_step_guide => 'Дотримуйтесь покрокової інструкції'; + + @override + String cookie_name_cookie(Object name) { + return 'Кукі-файл $name'; + } + + @override + String get fill_in_all_fields => 'Будь ласка, заповніть усі поля'; + + @override + String get submit => 'Надіслати'; + + @override + String get exit => 'Вийти'; + + @override + String get previous => 'Попередній'; + + @override + String get next => 'Наступний'; + + @override + String get done => 'Готово'; + + @override + String get step_1 => 'Крок 1'; + + @override + String get first_go_to => 'Спочатку перейдіть на'; + + @override + String get something_went_wrong => 'Щось пішло не так'; + + @override + String get piped_instance => 'Примірник сервера Piped'; + + @override + String get piped_description => + 'Примірник сервера Piped, який використовуватиметься для зіставлення треків'; + + @override + String get piped_warning => + 'Деякі з них можуть працювати неправильно. Тому використовуйте на свій страх і ризик'; + + @override + String get invidious_instance => 'Екземпляр сервера Invidious'; + + @override + String get invidious_description => + 'Екземпляр сервера Invidious для зіставлення треків'; + + @override + String get invidious_warning => + 'Деякі можуть працювати не дуже добре. Використовуйте на власний ризик'; + + @override + String get generate => 'Генерувати'; + + @override + String track_exists(Object track) { + return 'Трек $track вже існує'; + } + + @override + String get replace_downloaded_tracks => 'Замінити всі завантажені треки'; + + @override + String get skip_download_tracks => + 'Пропустити завантаження всіх завантажених треків'; + + @override + String get do_you_want_to_replace => 'Ви хочете замінити існуючий трек?'; + + @override + String get replace => 'Замінити'; + + @override + String get skip => 'Пропустити'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Виберіть до $count $type'; + } + + @override + String get select_genres => 'Виберіть жанри'; + + @override + String get add_genres => 'Додати жанри'; + + @override + String get country => 'Країна'; + + @override + String get number_of_tracks_generate => 'Кількість треків для створення'; + + @override + String get acousticness => 'Акустичність'; + + @override + String get danceability => 'Танцювальність'; + + @override + String get energy => 'Енергія'; + + @override + String get instrumentalness => 'Інструментальність'; + + @override + String get liveness => 'Живість'; + + @override + String get loudness => 'Гучність'; + + @override + String get speechiness => 'Розмовність'; + + @override + String get valence => 'Валентність'; + + @override + String get popularity => 'Популярність'; + + @override + String get key => 'Тональність'; + + @override + String get duration => 'Тривалість (с)'; + + @override + String get tempo => 'Темп (BPM)'; + + @override + String get mode => 'Режим'; + + @override + String get time_signature => 'Розмір'; + + @override + String get short => 'Короткий'; + + @override + String get medium => 'Середній'; + + @override + String get long => 'Довгий'; + + @override + String get min => 'Мін'; + + @override + String get max => 'Макс'; + + @override + String get target => 'Цільовий'; + + @override + String get moderate => 'Помірний'; + + @override + String get deselect_all => 'Зняти вибір з усіх'; + + @override + String get select_all => 'Вибрати всі'; + + @override + String get are_you_sure => 'Ви впевнені?'; + + @override + String get generating_playlist => + 'Створення вашого персонального плейлиста...'; + + @override + String selected_count_tracks(Object count) { + return 'Вибрано $count треків'; + } + + @override + String get download_warning => + 'Якщо ви завантажуєте всі треки масово, ви явно піратствуєте і завдаєте шкоди музичному творчому співтовариству. Сподіваюся, ви усвідомлюєте це. Завжди намагайтеся поважати і підтримувати важку працю артиста'; + + @override + String get download_ip_ban_warning => + 'До речі, ваш IP може бути заблокований на YouTube через надмірну кількість запитів на завантаження, ніж зазвичай. Блокування IP-адреси означає, що ви не зможете користуватися YouTube (навіть якщо ви увійшли в систему) протягом щонайменше 2-3 місяців з цього пристрою. І Spotube не несе жодної відповідальності, якщо це станеться'; + + @override + String get by_clicking_accept_terms => + 'Натискаючи \'прийняти\', ви погоджуєтеся з наступними умовами:'; + + @override + String get download_agreement_1 => 'Я знаю, що краду музику. Я поганий.'; + + @override + String get download_agreement_2 => + 'Я підтримаю автора, де тільки зможу, і роблю це лише тому, що не маю грошей, щоб купити його роботи.'; + + @override + String get download_agreement_3 => + 'Я повністю усвідомлюю, що мій IP може бути заблокований на YouTube, і я не покладаю на Spotube або його власників/контрибуторів відповідальність за будь-які нещасні випадки, спричинені моїми діями.'; + + @override + String get decline => 'Відхилити'; + + @override + String get accept => 'Прийняти'; + + @override + String get details => 'Деталі'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Канал'; + + @override + String get likes => 'Подобається'; + + @override + String get dislikes => 'Не подобається'; + + @override + String get views => 'Переглядів'; + + @override + String get streamUrl => 'Посилання на стрімінг'; + + @override + String get stop => 'Зупинити'; + + @override + String get sort_newest => 'Сортувати за датою додавання (новіші першими)'; + + @override + String get sort_oldest => 'Сортувати за датою додавання (старіші першими)'; + + @override + String get sleep_timer => 'Таймер сну'; + + @override + String mins(Object minutes) { + return '$minutes хвилин'; + } + + @override + String hours(Object hours) { + return '$hours годин'; + } + + @override + String hour(Object hours) { + return '$hours година'; + } + + @override + String get custom_hours => 'Кількість годин на замовлення'; + + @override + String get logs => 'Логи'; + + @override + String get developers => 'Розробники'; + + @override + String get not_logged_in => 'Ви не ввійшли в обліковий запис'; + + @override + String get search_mode => 'Режим пошуку'; + + @override + String get audio_source => 'Джерело аудіо'; + + @override + String get ok => 'Гаразд'; + + @override + String get failed_to_encrypt => 'Не вдалося зашифрувати'; + + @override + String get encryption_failed_warning => + 'Spotube використовує шифрування для безпечного зберігання ваших даних. Але не вдалося цього зробити. Тому він перейде до небезпечного зберігання\nЯкщо ви використовуєте Linux, переконайтеся, що у вас встановлено будь-який секретний сервіс (gnome-keyring, kde-wallet, keepassxc тощо)'; + + @override + String get querying_info => 'Запит інформації...'; + + @override + String get piped_api_down => 'API Piped не працює'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Поточний екземпляр Piped $pipedInstance не працює\n\nЗмініть екземпляр або змініть \'Тип API\' на офіційний YouTube API\n\nОбов\'язково перезапустіть програму після зміни'; + } + + @override + String get you_are_offline => 'Ви зараз не в мережі'; + + @override + String get connection_restored => 'Ваше інтернет-з\'єднання відновлено'; + + @override + String get use_system_title_bar => 'Використовувати системний заголовок'; + + @override + String get crunching_results => 'Опрацювання результатів...'; + + @override + String get search_to_get_results => 'Почніть пошук, щоб отримати результати'; + + @override + String get use_amoled_mode => 'Режим AMOLED'; + + @override + String get pitch_dark_theme => 'Темна тема'; + + @override + String get normalize_audio => 'Нормалізувати звук'; + + @override + String get change_cover => 'Змінити обкладинку'; + + @override + String get add_cover => 'Додати обкладинку'; + + @override + String get restore_defaults => 'Відновити налаштування за замовчуванням'; + + @override + String get download_music_format => 'Формат завантаження музики'; + + @override + String get streaming_music_format => 'Формат потокової музики'; + + @override + String get download_music_quality => 'Якість завантаженої музики'; + + @override + String get streaming_music_quality => 'Якість потокової музики'; + + @override + String get login_with_lastfm => 'Увійти з Last.fm'; + + @override + String get connect => 'Підключити'; + + @override + String get disconnect_lastfm => 'Відключитися від Last.fm'; + + @override + String get disconnect => 'Відключити'; + + @override + String get username => 'Ім\'я користувача'; + + @override + String get password => 'Пароль'; + + @override + String get login => 'Увійти'; + + @override + String get login_with_your_lastfm => 'Увійти в свій обліковий запис Last.fm'; + + @override + String get scrobble_to_lastfm => 'Скробблінг на Last.fm'; + + @override + String get go_to_album => 'Перейти до альбому'; + + @override + String get discord_rich_presence => 'Багата присутність у Discord'; + + @override + String get browse_all => 'Переглянути все'; + + @override + String get genres => 'Жанри'; + + @override + String get explore_genres => 'Досліджувати жанри'; + + @override + String get friends => 'Друзі'; + + @override + String get no_lyrics_available => + 'Вибачте, не вдалося знайти текст для цього треку'; + + @override + String get start_a_radio => 'Запустити радіо'; + + @override + String get how_to_start_radio => 'Як ви хочете запустити радіо?'; + + @override + String get replace_queue_question => + 'Ви хочете замінити поточну чергу чи додати до неї?'; + + @override + String get endless_playback => 'Безкінечне відтворення'; + + @override + String get delete_playlist => 'Видалити плейлист'; + + @override + String get delete_playlist_confirmation => + 'Ви впевнені, що хочете видалити цей плейлист?'; + + @override + String get local_tracks => 'Місцеві треки'; + + @override + String get local_tab => 'Місцевий'; + + @override + String get song_link => 'Посилання на пісню'; + + @override + String get skip_this_nonsense => 'Пропустити цей бред'; + + @override + String get freedom_of_music => '“Свобода музики”'; + + @override + String get freedom_of_music_palm => '“Свобода музики у вашій долоні”'; + + @override + String get get_started => 'Давайте почнемо'; + + @override + String get youtube_source_description => + 'Рекомендовано та працює краще за все.'; + + @override + String get piped_source_description => + 'Чи почуваєте себе вільно? Те саме, що і на YouTube, але набагато безкоштовно.'; + + @override + String get jiosaavn_source_description => + 'Найкраще для регіону Південної Азії.'; + + @override + String get invidious_source_description => + 'Подібний до Piped, але з вищою доступністю.'; + + @override + String highest_quality(Object quality) { + return 'Найвища якість: $quality'; + } + + @override + String get select_audio_source => 'Виберіть джерело аудіо'; + + @override + String get endless_playback_description => + 'Автоматично додавати нові пісні\nв кінець черги'; + + @override + String get choose_your_region => 'Виберіть ваш регіон'; + + @override + String get choose_your_region_description => + 'Це допоможе Spotube показати вам правильний контент\nдля вашого місцезнаходження.'; + + @override + String get choose_your_language => 'Виберіть свою мову'; + + @override + String get help_project_grow => 'Допоможіть цьому проекту рости'; + + @override + String get help_project_grow_description => + 'Spotube - це проект з відкритим кодом. Ви можете допомогти цьому проекту зростати, вносячи свій внесок у проект, повідомляючи про помилки або пропонуючи нові функції.'; + + @override + String get contribute_on_github => 'Долучайтесь на GitHub'; + + @override + String get donate_on_open_collective => 'Пожертвуйте на Open Collective'; + + @override + String get browse_anonymously => 'Анонімно переглядати'; + + @override + String get enable_connect => 'Увімкнути підключення'; + + @override + String get enable_connect_description => 'Керуйте Spotube з інших пристроїв'; + + @override + String get devices => 'Пристрої'; + + @override + String get select => 'Вибрати'; + + @override + String connect_client_alert(Object client) { + return 'Вас керує $client'; + } + + @override + String get this_device => 'Цей пристрій'; + + @override + String get remote => 'Віддалений'; + + @override + String get stats => 'Статистика'; + + @override + String and_n_more(Object count) { + return 'і $count більше'; + } + + @override + String get recently_played => 'Нещодавно Відтворене'; + + @override + String get browse_more => 'Переглянути Більше'; + + @override + String get no_title => 'Без Назви'; + + @override + String get not_playing => 'Не Відтворюється'; + + @override + String get epic_failure => 'Епічний провал!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Додано $tracks_length треків до черги'; + } + + @override + String get spotube_has_an_update => 'Spotube має оновлення'; + + @override + String get download_now => 'Завантажити Зараз'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum було випущено'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version було випущено'; + } + + @override + String get read_the_latest => 'Читати останні новини'; + + @override + String get release_notes => 'ноти про випуск'; + + @override + String get pick_color_scheme => 'Оберіть кольорову схему'; + + @override + String get save => 'Зберегти'; + + @override + String get choose_the_device => 'Виберіть пристрій:'; + + @override + String get multiple_device_connected => + 'Підключено кілька пристроїв.\nВиберіть пристрій, на якому ви хочете виконати цю дію'; + + @override + String get nothing_found => 'Нічого не знайдено'; + + @override + String get the_box_is_empty => 'Коробка порожня'; + + @override + String get top_artists => 'Топ Артисти'; + + @override + String get top_albums => 'Топ Альбоми'; + + @override + String get this_week => 'Цього тижня'; + + @override + String get this_month => 'Цього місяця'; + + @override + String get last_6_months => 'Останні 6 місяців'; + + @override + String get this_year => 'Цього року'; + + @override + String get last_2_years => 'Останні 2 роки'; + + @override + String get all_time => 'Усі часи'; + + @override + String powered_by_provider(Object providerName) { + return 'Забезпечено $providerName'; + } + + @override + String get email => 'Електронна пошта'; + + @override + String get profile_followers => 'Підписники'; + + @override + String get birthday => 'День народження'; + + @override + String get subscription => 'Підписка'; + + @override + String get not_born => 'Ще не народжений'; + + @override + String get hacker => 'Хакер'; + + @override + String get profile => 'Профіль'; + + @override + String get no_name => 'Без імені'; + + @override + String get edit => 'Редагувати'; + + @override + String get user_profile => 'Профіль користувача'; + + @override + String count_plays(Object count) { + return '$count відтворень'; + } + + @override + String get streaming_fees_hypothetical => + '*Розраховано на основі виплат Spotify за стримінг\nвід \$0.003 до \$0.005. Це гіпотетичний\nрозрахунок, щоб дати уявлення користувачу про те, скільки б він\nзаплатив артистам, якби слухав їхні пісні на Spotify.'; + + @override + String get minutes_listened => 'Хвилини прослуховування'; + + @override + String get streamed_songs => 'Стримлені пісні'; + + @override + String count_streams(Object count) { + return '$count стримів'; + } + + @override + String get owned_by_you => 'Ваша власність'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl скопійовано в буфер обміну'; + } + + @override + String get hipotetical_calculation => + '*Це розраховано на основі середньої виплати за стрім онлайн-платформ для потокового відтворення музики, що становить від \$0,003 до \$0,005. Це гіпотетичний розрахунок, щоб дати користувачеві уявлення про те, скільки б вони заплатили артистам, якщо б слухали їхні пісні на різних музичних стрімінгових платформах.'; + + @override + String count_mins(Object minutes) { + return '$minutes хв'; + } + + @override + String get summary_minutes => 'хвилини'; + + @override + String get summary_listened_to_music => 'Прослухана музика'; + + @override + String get summary_songs => 'пісні'; + + @override + String get summary_streamed_overall => 'Загалом стримів'; + + @override + String get summary_owed_to_artists => 'Заборгованість артистам\nцього місяця'; + + @override + String get summary_artists => 'артистів'; + + @override + String get summary_music_reached_you => 'Музика досягла вас'; + + @override + String get summary_full_albums => 'повні альбоми'; + + @override + String get summary_got_your_love => 'Отримав вашу любов'; + + @override + String get summary_playlists => 'плейлисти'; + + @override + String get summary_were_on_repeat => 'Були на повторі'; + + @override + String total_money(Object money) { + return 'Загалом $money'; + } + + @override + String get webview_not_found => 'Webview не знайдено'; + + @override + String get webview_not_found_description => + 'На вашому пристрої не встановлено виконуване середовище Webview.\nЯкщо воно встановлено, переконайтеся, що воно знаходиться в environment PATH\n\nПісля встановлення перезапустіть програму'; + + @override + String get unsupported_platform => 'Непідтримувана платформа'; + + @override + String get cache_music => 'Кешувати музику'; + + @override + String get open => 'Відкрити'; + + @override + String get cache_folder => 'Тека кешу'; + + @override + String get export => 'Експорт'; + + @override + String get clear_cache => 'Очистити кеш'; + + @override + String get clear_cache_confirmation => 'Ви хочете очистити кеш?'; + + @override + String get export_cache_files => 'Експортувати кешовані файли'; + + @override + String found_n_files(Object count) { + return 'Знайдено $count файлів'; + } + + @override + String get export_cache_confirmation => 'Ви хочете експортувати ці файли до'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Експортовано $filesExported з $files файлів'; + } + + @override + String get undo => 'Скасувати'; + + @override + String get download_all => 'Завантажити все'; + + @override + String get add_all_to_playlist => 'Додати все до плейлиста'; + + @override + String get add_all_to_queue => 'Додати все в чергу'; + + @override + String get play_all_next => 'Відтворити все наступне'; + + @override + String get pause => 'Пауза'; + + @override + String get view_all => 'Переглянути все'; + + @override + String get no_tracks_added_yet => 'Здається, ви ще не додали жодної пісні'; + + @override + String get no_tracks => 'Здається, тут немає пісень'; + + @override + String get no_tracks_listened_yet => 'Здається, ви ще нічого не слухали'; + + @override + String get not_following_artists => 'Ви не підписані на жодного артиста'; + + @override + String get no_favorite_albums_yet => + 'Здається, ви ще не додали жодного альбому в улюблені'; + + @override + String get no_logs_found => 'Жодних журналів не знайдено'; + + @override + String get youtube_engine => 'YouTube Двигун'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine не встановлено'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine не встановлено на вашій системі.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Переконайтесь, що він доступний у змінній PATH або\nвстановіть абсолютний шлях до виконуваного файлу $engine нижче'; + } + + @override + String get youtube_engine_unix_issue_message => + 'У macOS/Linux/Unix-подібних ОС, встановлення шляху в .zshrc/.bashrc/.bash_profile тощо не працює.\nВам потрібно налаштувати шлях у файлі конфігурації оболонки'; + + @override + String get download => 'Завантажити'; + + @override + String get file_not_found => 'Файл не знайдено'; + + @override + String get custom => 'Користувацький'; + + @override + String get add_custom_url => 'Додати користувацький URL'; + + @override + String get edit_port => 'Редагувати порт'; + + @override + String get port_helper_msg => + 'За замовчуванням -1, що означає випадкове число. Якщо у вас налаштований брандмауер, рекомендується це налаштувати.'; + + @override + String connect_request(Object client) { + return 'Дозволити $client підключення?'; + } + + @override + String get connection_request_denied => + 'Підключення відхилено. Користувач відмовив у доступі.'; + + @override + String get an_error_occurred => 'Сталася помилка'; + + @override + String get copy_to_clipboard => 'Копіювати в буфер обміну'; + + @override + String get view_logs => 'Переглянути логи'; + + @override + String get retry => 'Повторити'; + + @override + String get no_default_metadata_provider_selected => + 'Ви не встановили провайдера метаданих за замовчуванням'; + + @override + String get manage_metadata_providers => 'Керувати провайдерами метаданих'; + + @override + String get open_link_in_browser => 'Відкрити посилання в браузері?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Ви хочете відкрити наступне посилання'; + + @override + String get unsafe_url_warning => + 'Відкриття посилань з ненадійних джерел може бути небезпечним. Будьте обережні!\nВи також можете скопіювати посилання в буфер обміну.'; + + @override + String get copy_link => 'Копіювати посилання'; + + @override + String get building_your_timeline => + 'Створення вашої часової шкали на основі ваших прослуховувань...'; + + @override + String get official => 'Офіційний'; + + @override + String author_name(Object author) { + return 'Автор: $author'; + } + + @override + String get third_party => 'Сторонній'; + + @override + String get plugin_requires_authentication => 'Плагін вимагає автентифікації'; + + @override + String get update_available => 'Доступне оновлення'; + + @override + String get supports_scrobbling => 'Підтримує скроблінг'; + + @override + String get plugin_scrobbling_info => + 'Цей плагін скроббить вашу музику, щоб створити вашу історію прослуховувань.'; + + @override + String get default_metadata_source => 'Джерело метаданих за замовчуванням'; + + @override + String get set_default_metadata_source => + 'Встановити джерело метаданих за замовчуванням'; + + @override + String get default_audio_source => 'Джерело аудіо за замовчуванням'; + + @override + String get set_default_audio_source => + 'Встановити джерело аудіо за замовчуванням'; + + @override + String get set_default => 'Встановити за замовчуванням'; + + @override + String get support => 'Підтримка'; + + @override + String get support_plugin_development => 'Підтримати розробку плагіна'; + + @override + String can_access_name_api(Object name) { + return '- Може отримати доступ до **$name** API'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Ви хочете встановити цей плагін?'; + + @override + String get third_party_plugin_warning => + 'Цей плагін із стороннього репозиторію. Будь ласка, переконайтеся, що ви довіряєте джерелу перед встановленням.'; + + @override + String get author => 'Автор'; + + @override + String get this_plugin_can_do_following => 'Цей плагін може робити наступне'; + + @override + String get install => 'Встановити'; + + @override + String get install_a_metadata_provider => 'Встановити провайдера метаданих'; + + @override + String get no_tracks_playing => 'Наразі не відтворюється жоден трек'; + + @override + String get synced_lyrics_not_available => + 'Синхронізовані тексти недоступні для цієї пісні. Будь ласка, використовуйте вкладку'; + + @override + String get plain_lyrics => 'Звичайні тексти'; + + @override + String get tab_instead => 'замість цього.'; + + @override + String get disclaimer => 'Відмова від відповідальності'; + + @override + String get third_party_plugin_dmca_notice => + 'Команда Spotube не несе жодної відповідальності (включно з юридичною) за будь-які плагіни \"третіх сторін\".\nБудь ласка, використовуйте їх на свій страх і ризик. Про будь-які помилки/проблеми повідомляйте в репозиторій плагіна.\n\nЯкщо якийсь плагін \"третьої сторони\" порушує ToS/DMCA будь-якої служби/юридичної особи, будь ласка, попросіть автора плагіна \"третьої сторони\" або хостингову платформу, наприклад, GitHub/Codeberg, вжити заходів. Усі перераховані вище (позначені як \"треті сторони\") є плагінами, які підтримуються публічно/спільнотою. Ми не куруємо їх, тому не можемо вжити жодних заходів щодо них.\n\n'; + + @override + String get input_does_not_match_format => + 'Введені дані не відповідають необхідному формату'; + + @override + String get plugins => 'Плагіни'; + + @override + String get paste_plugin_download_url => + 'Вставте URL-адресу для завантаження або URL-адресу репозиторію GitHub/Codeberg або пряме посилання на файл .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Завантажити та встановити плагін з URL-адреси'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Не вдалося додати плагін: $error'; + } + + @override + String get upload_plugin_from_file => 'Завантажити плагін з файлу'; + + @override + String get installed => 'Встановлено'; + + @override + String get available_plugins => 'Доступні плагіни'; + + @override + String get configure_plugins => + 'Налаштуйте власні плагіни метаданих і аудіоджерела'; + + @override + String get audio_scrobblers => 'Аудіо скробблери'; + + @override + String get scrobbling => 'Скроблінг'; + + @override + String get source => 'Джерело: '; + + @override + String get uncompressed => 'Без стиснення'; + + @override + String get dab_music_source_description => + 'Для аудіофілів. Забезпечує високоякісні/без втрат аудіопотоки. Точна відповідність треків на основі ISRC.'; +} diff --git a/lib/l10n/generated/app_localizations_vi.dart b/lib/l10n/generated/app_localizations_vi.dart new file mode 100644 index 00000000..4d7a8945 --- /dev/null +++ b/lib/l10n/generated/app_localizations_vi.dart @@ -0,0 +1,1574 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Vietnamese (`vi`). +class AppLocalizationsVi extends AppLocalizations { + AppLocalizationsVi([String locale = 'vi']) : super(locale); + + @override + String get guest => 'Khách'; + + @override + String get browse => 'Khám phá'; + + @override + String get search => 'Tìm kiếm'; + + @override + String get library => 'Thư viên'; + + @override + String get lyrics => 'Lời bài hát'; + + @override + String get settings => 'Cài đặt'; + + @override + String get genre_categories_filter => 'Lọc theo thể loại nhạc...'; + + @override + String get genre => 'Thể loại nhạc'; + + @override + String get personalized => 'Cá nhân hóa'; + + @override + String get featured => 'Nổi bật'; + + @override + String get new_releases => 'Bản phát hành mới'; + + @override + String get songs => 'Bài hát'; + + @override + String playing_track(Object track) { + return 'Đang phát $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return 'Điều này sẽ xóa hàng đợi hiện tại. $track_length bài hát sẽ bị xóa\nBạn có muốn tiếp tục không?'; + } + + @override + String get load_more => 'Tải thêm'; + + @override + String get playlists => 'Danh sách phát'; + + @override + String get artists => 'Nghệ sĩ'; + + @override + String get albums => 'Album'; + + @override + String get tracks => 'Bài hát'; + + @override + String get downloads => 'Tải về'; + + @override + String get filter_playlists => 'Lọc danh sách phát...'; + + @override + String get liked_tracks => 'Bài hát được thích'; + + @override + String get liked_tracks_description => 'Tất cả bài hát bạn đã thích'; + + @override + String get playlist => 'Danh sách phát'; + + @override + String get create_a_playlist => 'Tạo danh sách phát'; + + @override + String get update_playlist => 'Cập nhật danh sách phát'; + + @override + String get create => 'Tạo'; + + @override + String get cancel => 'Hủy'; + + @override + String get update => 'Cập nhật'; + + @override + String get playlist_name => 'Tên danh sách phát'; + + @override + String get name_of_playlist => 'Tên của danh sách phát'; + + @override + String get description => 'Mô tả'; + + @override + String get public => 'Công khai'; + + @override + String get collaborative => 'Hợp tác'; + + @override + String get search_local_tracks => 'Tìm kiếm bài hát trong máy...'; + + @override + String get play => 'Phát'; + + @override + String get delete => 'Xóa'; + + @override + String get none => 'Không có'; + + @override + String get sort_a_z => 'Sắp xếp theo A-Z'; + + @override + String get sort_z_a => 'Sắp xếp theo Z-A'; + + @override + String get sort_artist => 'Sắp xếp theo Nghệ sĩ'; + + @override + String get sort_album => 'Sắp xếp theo Album'; + + @override + String get sort_duration => 'Sắp xếp theo Thời lượng'; + + @override + String get sort_tracks => 'Sắp xếp các bài hát'; + + @override + String currently_downloading(Object tracks_length) { + return 'Đang tải về ($tracks_length bài hát)'; + } + + @override + String get cancel_all => 'Hủy tất cả'; + + @override + String get filter_artist => 'Lọc nghệ sĩ...'; + + @override + String followers(Object followers) { + return '$followers Người theo dõi'; + } + + @override + String get add_artist_to_blacklist => 'Thêm nghệ sĩ vào blacklist'; + + @override + String get top_tracks => 'Bài hát nổi bật'; + + @override + String get fans_also_like => 'Người hâm mộ cũng thích'; + + @override + String get loading => 'Đang tải...'; + + @override + String get artist => 'Nghệ sĩ'; + + @override + String get blacklisted => 'Đã đưa vào blacklist'; + + @override + String get following => 'Đang theo dõi'; + + @override + String get follow => 'Theo dõi'; + + @override + String get artist_url_copied => 'Đã sao chép URL nghệ sĩ'; + + @override + String added_to_queue(Object tracks) { + return 'Đã thêm $tracks bài hát vào hàng đợi'; + } + + @override + String get filter_albums => 'Lọc album...'; + + @override + String get synced => 'Đồng bộ'; + + @override + String get plain => 'Bình thường'; + + @override + String get shuffle => 'Trộn'; + + @override + String get search_tracks => 'Tìm kiếm bài hát...'; + + @override + String get released => 'Phát hành'; + + @override + String error(Object error) { + return 'Lỗi $error'; + } + + @override + String get title => 'Đề mục'; + + @override + String get time => 'Thời gian'; + + @override + String get more_actions => 'Thao tác khác'; + + @override + String download_count(Object count) { + return 'Tải xuống ($count)'; + } + + @override + String add_count_to_playlist(Object count) { + return 'Thêm ($count) vào danh sách phát'; + } + + @override + String add_count_to_queue(Object count) { + return 'Thêm ($count) vào hàng đợi'; + } + + @override + String play_count_next(Object count) { + return 'Phát ($count) tiếp theo'; + } + + @override + String get album => 'Album'; + + @override + String copied_to_clipboard(Object data) { + return 'Đã sao chép $data vào clipboard'; + } + + @override + String add_to_following_playlists(Object track) { + return 'Thêm $track vào danh sách phát đang theo dõi'; + } + + @override + String get add => 'Thêm'; + + @override + String added_track_to_queue(Object track) { + return 'Đã thêm $track vào hàng đợi'; + } + + @override + String get add_to_queue => 'Thêm vào hàng đợi'; + + @override + String track_will_play_next(Object track) { + return '$track sẽ được phát tiếp theo'; + } + + @override + String get play_next => 'Phát tiếp theo'; + + @override + String removed_track_from_queue(Object track) { + return 'Đã xóa $track khỏi hàng đợi'; + } + + @override + String get remove_from_queue => 'Xóa khỏi hàng đợi'; + + @override + String get remove_from_favorites => 'Xóa khỏi bài hát yêu thích'; + + @override + String get save_as_favorite => 'Thêm vào bài hát yêu thích'; + + @override + String get add_to_playlist => 'Thêm vào danh sách phát'; + + @override + String get remove_from_playlist => 'Xóa khỏi danh sách phát'; + + @override + String get add_to_blacklist => 'Thêm vào blacklist'; + + @override + String get remove_from_blacklist => 'Xóa khỏi blacklist'; + + @override + String get share => 'Chia sẻ'; + + @override + String get mini_player => 'Trình phát thu nhỏ'; + + @override + String get slide_to_seek => 'Trượt để tìm kiếm tiến hoặc lùi'; + + @override + String get shuffle_playlist => 'Xáo trộn bài hát'; + + @override + String get unshuffle_playlist => 'Hủy xáo trộn bài hát'; + + @override + String get previous_track => 'Bài hát trước'; + + @override + String get next_track => 'Bài hát tiếp theo'; + + @override + String get pause_playback => 'Tạm dừng phát'; + + @override + String get resume_playback => 'Tiếp tục phát'; + + @override + String get loop_track => 'Lặp lại bài hát'; + + @override + String get no_loop => 'Không lặp lại'; + + @override + String get repeat_playlist => 'Lặp lại danh sách phát'; + + @override + String get queue => 'Hàng đợi'; + + @override + String get alternative_track_sources => 'Đổi nguồn bài hát'; + + @override + String get download_track => 'Tải xuống'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks bài hát trong hàng đợi'; + } + + @override + String get clear_all => 'Xóa tất cả'; + + @override + String get show_hide_ui_on_hover => + 'Hiển thị/Ẩn giao diện người dùng khi di chuột qua'; + + @override + String get always_on_top => 'Luôn ở trên cùng'; + + @override + String get exit_mini_player => 'Thoát khỏi trình phát thu nhỏ'; + + @override + String get download_location => 'Vị trí tải xuống'; + + @override + String get local_library => 'Thư viện địa phương'; + + @override + String get add_library_location => 'Thêm vào thư viện'; + + @override + String get remove_library_location => 'Xóa khỏi thư viện'; + + @override + String get account => 'Tài khoản'; + + @override + String get logout => 'Đăng xuất'; + + @override + String get logout_of_this_account => 'Đăng xuất khỏi tài khoản này'; + + @override + String get language_region => 'Ngôn ngữ và Khu vực'; + + @override + String get language => 'Ngôn ngữ'; + + @override + String get system_default => 'Mặc định hệ thống'; + + @override + String get market_place_region => 'Khu vực Marketplace'; + + @override + String get recommendation_country => 'Quốc gia gợi ý'; + + @override + String get appearance => 'Giao diện'; + + @override + String get layout_mode => 'Chế độ layout'; + + @override + String get override_layout_settings => 'Ghi đè cài đặt layout'; + + @override + String get adaptive => 'Tương thích'; + + @override + String get compact => 'Nhỏ gọn'; + + @override + String get extended => 'Mở rộng'; + + @override + String get theme => 'Chủ đề'; + + @override + String get dark => 'Tối'; + + @override + String get light => 'Sáng'; + + @override + String get system => 'Hệ thống'; + + @override + String get accent_color => 'Màu nhấn'; + + @override + String get sync_album_color => 'Đồng bộ màu album'; + + @override + String get sync_album_color_description => + 'Sử dụng màu chủ đạo của hình ảnh album làm màu nhấn'; + + @override + String get playback => 'Phát'; + + @override + String get audio_quality => 'Chất lượng âm thanh'; + + @override + String get high => 'Cao'; + + @override + String get low => 'Thấp'; + + @override + String get pre_download_play => 'Tải xuống và phát'; + + @override + String get pre_download_play_description => + 'Thay vì stream âm thanh, tải xuống trước và phát (Khuyến nghị cho người dùng có băng thông cao)'; + + @override + String get skip_non_music => 'Bỏ qua các đoạn không phải nhạc (SponsorBlock)'; + + @override + String get blacklist_description => 'Các bài hát và nghệ sĩ trong blacklist'; + + @override + String get wait_for_download_to_finish => + 'Vui lòng đợi quá trình tải xuống hiện tại hoàn thành'; + + @override + String get desktop => 'Máy tính'; + + @override + String get close_behavior => 'Thao tác đóng'; + + @override + String get close => 'Đóng'; + + @override + String get minimize_to_tray => 'Thu nhỏ vào khay hệ thống'; + + @override + String get show_tray_icon => 'Hiển thị biểu tượng trên khay hệ thống'; + + @override + String get about => 'Về chúng tôi'; + + @override + String get u_love_spotube => 'Chúng tôi biết bạn yêu Spotube'; + + @override + String get check_for_updates => 'Kiểm tra cập nhật'; + + @override + String get about_spotube => 'Về Spotube'; + + @override + String get blacklist => 'blacklist'; + + @override + String get please_sponsor => 'Vui lòng tài trợ/ủng hộ'; + + @override + String get spotube_description => + 'Spotube, một ứng dụng Spotify nhẹ, đa nền tảng và miễn phí'; + + @override + String get version => 'Phiên bản'; + + @override + String get build_number => 'Số phiên bản'; + + @override + String get founder => 'Người sáng lập'; + + @override + String get repository => 'Mã nguồn'; + + @override + String get bug_issues => 'Báo cáo lỗi'; + + @override + String get made_with => 'Được làm bằng ❤️ ở Băng-la-đét'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => 'Giấy phép'; + + @override + String get credentials_will_not_be_shared_disclaimer => + 'Đừng lo, thông tin đăng nhập của bạn sẽ không được thu thập hoặc chia sẻ với bất kỳ ai'; + + @override + String get know_how_to_login => 'Không biết cách lấy thông tin đăng nhập?'; + + @override + String get follow_step_by_step_guide => 'Các bước lấy thông tin đăng nhập'; + + @override + String cookie_name_cookie(Object name) { + return 'Cookie $name'; + } + + @override + String get fill_in_all_fields => 'Vui lòng điền đầy đủ thông tin'; + + @override + String get submit => 'Gửi'; + + @override + String get exit => 'Thoát'; + + @override + String get previous => 'Trước'; + + @override + String get next => 'Tiếp'; + + @override + String get done => 'Hoàn tất'; + + @override + String get step_1 => 'Bước 1'; + + @override + String get first_go_to => 'Đầu tiên, truy cập'; + + @override + String get something_went_wrong => 'Đã xảy ra lỗi'; + + @override + String get piped_instance => 'Phiên bản Server Piped'; + + @override + String get piped_description => + 'Phiên bản Piped để sử dụng cho Track matching'; + + @override + String get piped_warning => + 'Một số phiên bản Piped có thể không hoạt động tốt'; + + @override + String get invidious_instance => 'Phiên bản máy chủ Invidious'; + + @override + String get invidious_description => + 'Phiên bản máy chủ Invidious để sử dụng để so khớp bản nhạc'; + + @override + String get invidious_warning => + 'Một số có thể sẽ không hoạt động tốt. Vì vậy hãy sử dụng với rủi ro của riêng bạn'; + + @override + String get generate => 'Tạo'; + + @override + String track_exists(Object track) { + return 'Bài hát $track đã tồn tại'; + } + + @override + String get replace_downloaded_tracks => 'Thay thế tất cả các bài hát đã tải'; + + @override + String get skip_download_tracks => + 'Bỏ qua tải xuống tất cả các bài hát đã tải'; + + @override + String get do_you_want_to_replace => + 'Bạn có muốn thay thế bài hát hiện có không?'; + + @override + String get replace => 'Thay thế'; + + @override + String get skip => 'Bỏ qua'; + + @override + String select_up_to_count_type(Object count, Object type) { + return 'Chọn tối đa $count $type'; + } + + @override + String get select_genres => 'Chọn Thể loại'; + + @override + String get add_genres => 'Thêm Thể loại'; + + @override + String get country => 'Quốc gia'; + + @override + String get number_of_tracks_generate => 'Số lượng bài hát để tạo'; + + @override + String get acousticness => 'Độ âm thanh'; + + @override + String get danceability => 'Khả năng nhảy'; + + @override + String get energy => 'Năng lượng'; + + @override + String get instrumentalness => 'Độ nhạc cụ'; + + @override + String get liveness => 'Sống động'; + + @override + String get loudness => 'Độ ồn'; + + @override + String get speechiness => 'Độ nói'; + + @override + String get valence => 'Tính tích cực'; + + @override + String get popularity => 'Độ phổ biến'; + + @override + String get key => 'Tông'; + + @override + String get duration => 'Thời lượng (giây)'; + + @override + String get tempo => 'Nhịp độ (BPM)'; + + @override + String get mode => 'Chế độ'; + + @override + String get time_signature => 'Chữ ký thời gian'; + + @override + String get short => 'Ngắn'; + + @override + String get medium => 'Trung bình'; + + @override + String get long => 'Dài'; + + @override + String get min => 'Tối thiểu'; + + @override + String get max => 'Tối đa'; + + @override + String get target => 'Mục tiêu'; + + @override + String get moderate => 'Trung bình'; + + @override + String get deselect_all => 'Bỏ chọn tất cả'; + + @override + String get select_all => 'Chọn tất cả'; + + @override + String get are_you_sure => 'Bạn có chắc chắn?'; + + @override + String get generating_playlist => + 'Đang tạo danh sách phát tùy chỉnh của bạn...'; + + @override + String selected_count_tracks(Object count) { + return 'Đã chọn $count bài hát'; + } + + @override + String get download_warning => + 'Tải xuống tất cả các bài hát một lần, sẽ vi phạm bản quyền âm nhạc và gây thiệt hại cho xã hội sáng tạo âm nhạc. Hy vọng bạn nhận thức được điều này. Hãy luôn tôn trọng và ủng hộ công sức của nghệ sĩ'; + + @override + String get download_ip_ban_warning => + 'Địa chỉ IP của bạn có thể bị chặn trên YouTube do yêu cầu tải xuống quá mức so với bình thường. Chặn IP có nghĩa là bạn không thể sử dụng YouTube (ngay cả khi bạn đã đăng nhập) ít nhất 2-3 tháng từ thiết bị IP đó. Và Spotube không chịu trách nhiệm nếu điều này xảy ra'; + + @override + String get by_clicking_accept_terms => + 'Bằng cách nhấp vào \'Chấp nhận\', bạn đồng ý với các điều khoản sau:'; + + @override + String get download_agreement_1 => + 'Tôi biết mình đang vi phạm bản quyền âm nhạc. Đó là không tốt.'; + + @override + String get download_agreement_2 => + 'Tôi sẽ ủng hộ nghệ sĩ bất cứ nơi nào tôi có thể và tôi chỉ làm điều này vì tôi không có tiền để mua tác phẩm của họ'; + + @override + String get download_agreement_3 => + 'Tôi hoàn toàn nhận thức được rằng địa chỉ IP của tôi có thể bị chặn trên YouTube và tôi không đổ lỗi cho Spotube hoặc chủ sở hữu/người đóng góp của nó về bất kỳ tai nạn nào do hành động này của tôi'; + + @override + String get decline => 'Từ chối'; + + @override + String get accept => 'Chấp nhận'; + + @override + String get details => 'Chi tiết'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => 'Kênh'; + + @override + String get likes => 'Thích'; + + @override + String get dislikes => 'Không thích'; + + @override + String get views => 'Lượt xem'; + + @override + String get streamUrl => 'URL phát trực tiếp'; + + @override + String get stop => 'Dừng'; + + @override + String get sort_newest => 'Sắp xếp theo mới nhất'; + + @override + String get sort_oldest => 'Sắp xếp theo cũ nhất'; + + @override + String get sleep_timer => 'Hẹn giờ tắt'; + + @override + String mins(Object minutes) { + return '$minutes Phút'; + } + + @override + String hours(Object hours) { + return '$hours Giờ'; + } + + @override + String hour(Object hours) { + return '$hours Giờ'; + } + + @override + String get custom_hours => 'Giờ Tùy chỉnh'; + + @override + String get logs => 'Nhật ký'; + + @override + String get developers => 'Nhà phát triển'; + + @override + String get not_logged_in => 'Bạn chưa đăng nhập'; + + @override + String get search_mode => 'Chế độ tìm kiếm'; + + @override + String get audio_source => 'Nguồn âm thanh'; + + @override + String get ok => 'Ok'; + + @override + String get failed_to_encrypt => 'Mã hóa không thành công'; + + @override + String get encryption_failed_warning => + 'Spotube không thành công trong việc mã hóa nhằm lưu trữ dữ liêu an toàn. vậy nên sẽ chuyển về lưu trữ không an toàn\nNếu bạn đang sử dụng Linux, đảm bảo rằng bạn có sử dụng dịch vụ bảo mật (gnome-keyring, kde-wallet, keepassxc, v.v.)'; + + @override + String get querying_info => 'Đang truy vấn thông tin...'; + + @override + String get piped_api_down => 'API Piped đang gặp sự cố'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return 'Phiên bản Piped $pipedInstance hiện đang gặp sự cố\n\nThay đổi phiên bản hoặc thay đổi \'Loại API\' thành API YouTube official\n\nKhởi động lai ứng dụng sau khi thay đổi.'; + } + + @override + String get you_are_offline => 'Bạn đang ngoại tuyến'; + + @override + String get connection_restored => + 'Kết nối internet của bạn đã được khôi phục'; + + @override + String get use_system_title_bar => 'Sử dụng thanh tiêu đề hệ thống'; + + @override + String get crunching_results => 'Đang tìm kiếm...'; + + @override + String get search_to_get_results => 'Chưa tìm kiếm'; + + @override + String get use_amoled_mode => 'Chủ đề tối hoàn toàn'; + + @override + String get pitch_dark_theme => 'Chế độ AMOLED'; + + @override + String get normalize_audio => 'Bình thường hóa âm thanh'; + + @override + String get change_cover => 'Thay đổi ảnh bìa'; + + @override + String get add_cover => 'Thêm ảnh bìa'; + + @override + String get restore_defaults => 'Khôi phục mặc định'; + + @override + String get download_music_format => 'Định dạng nhạc tải về'; + + @override + String get streaming_music_format => 'Định dạng nhạc phát trực tuyến'; + + @override + String get download_music_quality => 'Chất lượng nhạc tải về'; + + @override + String get streaming_music_quality => 'Chất lượng nhạc phát trực tuyến'; + + @override + String get login_with_lastfm => 'Đăng nhập bằng tài khoản Last.fm'; + + @override + String get connect => 'Liên kết'; + + @override + String get disconnect_lastfm => 'Dừng liên kết Last.fm'; + + @override + String get disconnect => 'Ngắt kết nối'; + + @override + String get username => 'Tên người dùng'; + + @override + String get password => 'Mật khẩu'; + + @override + String get login => 'Đăng nhập'; + + @override + String get login_with_your_lastfm => + 'Đăng nhập bằng tài khoản Last.fm của bạn'; + + @override + String get scrobble_to_lastfm => 'Scrobble đến Last.fm'; + + @override + String get go_to_album => 'Đi đến Album'; + + @override + String get discord_rich_presence => 'Hiển thị trạng thái Discord'; + + @override + String get browse_all => 'Duyệt tất cả'; + + @override + String get genres => 'Thể loại'; + + @override + String get explore_genres => 'Khám phá Thể loại'; + + @override + String get friends => 'Bạn bè'; + + @override + String get no_lyrics_available => + 'Xin lỗi, không tìm thấy lời cho bài hát này'; + + @override + String get start_a_radio => 'Bắt đầu Một Đài phát thanh'; + + @override + String get how_to_start_radio => + 'Bạn muốn bắt đầu đài phát thanh như thế nào?'; + + @override + String get replace_queue_question => + 'Bạn muốn thay thế hàng đợi hiện tại hay thêm vào?'; + + @override + String get endless_playback => 'Phát không giới hạn'; + + @override + String get delete_playlist => 'Xóa Danh sách phát'; + + @override + String get delete_playlist_confirmation => + 'Bạn có chắc chắn muốn xóa danh sách phát này không?'; + + @override + String get local_tracks => 'Bài hát Địa phương'; + + @override + String get local_tab => 'Địa phương'; + + @override + String get song_link => 'Liên kết Bài hát'; + + @override + String get skip_this_nonsense => 'Bỏ qua bớt rối này'; + + @override + String get freedom_of_music => '“Sự Tự do của Âm nhạc”'; + + @override + String get freedom_of_music_palm => + '“Sự Tự do của Âm nhạc trong lòng bàn tay của bạn”'; + + @override + String get get_started => 'Bắt đầu thôi'; + + @override + String get youtube_source_description => + 'Được đề xuất và hoạt động tốt nhất.'; + + @override + String get piped_source_description => + 'Cảm thấy tự do? Giống như YouTube nhưng miễn phí hơn rất nhiều.'; + + @override + String get jiosaavn_source_description => 'Tốt nhất cho khu vực Nam Á.'; + + @override + String get invidious_source_description => + 'Tương tự như Piped nhưng có tính khả dụng cao hơn.'; + + @override + String highest_quality(Object quality) { + return 'Chất lượng Tốt nhất: $quality'; + } + + @override + String get select_audio_source => 'Chọn Nguồn Âm thanh'; + + @override + String get endless_playback_description => + 'Tự động thêm các bài hát mới\nvào cuối hàng đợi'; + + @override + String get choose_your_region => 'Chọn khu vực của bạn'; + + @override + String get choose_your_region_description => + 'Điều này sẽ giúp Spotube hiển thị nội dung phù hợp cho vị trí của bạn.'; + + @override + String get choose_your_language => 'Chọn ngôn ngữ của bạn'; + + @override + String get help_project_grow => 'Hãy giúp dự án này phát triển'; + + @override + String get help_project_grow_description => + 'Spotube là một dự án mã nguồn mở. Bạn có thể giúp dự án này phát triển bằng cách đóng góp vào dự án, báo cáo lỗi hoặc đề xuất tính năng mới.'; + + @override + String get contribute_on_github => 'Đóng góp trên GitHub'; + + @override + String get donate_on_open_collective => 'Quyên góp trên Open Collective'; + + @override + String get browse_anonymously => 'Duyệt Anonymously'; + + @override + String get enable_connect => 'Kích hoạt kết nối'; + + @override + String get enable_connect_description => + 'Điều khiển Spotube từ các thiết bị khác'; + + @override + String get devices => 'Thiết bị'; + + @override + String get select => 'Chọn'; + + @override + String connect_client_alert(Object client) { + return 'Bạn đang được điều khiển bởi $client'; + } + + @override + String get this_device => 'Thiết bị này'; + + @override + String get remote => 'Từ xa'; + + @override + String get stats => 'Thống kê'; + + @override + String and_n_more(Object count) { + return 'và $count cái khác'; + } + + @override + String get recently_played => 'Gần đây đã phát'; + + @override + String get browse_more => 'Xem thêm'; + + @override + String get no_title => 'Không có tiêu đề'; + + @override + String get not_playing => 'Không phát'; + + @override + String get epic_failure => 'Thất bại hoàn toàn!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return 'Đã thêm $tracks_length bài hát vào danh sách phát'; + } + + @override + String get spotube_has_an_update => 'Spotube có bản cập nhật'; + + @override + String get download_now => 'Tải về ngay'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum đã được phát hành'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version đã được phát hành'; + } + + @override + String get read_the_latest => 'Đọc tin mới nhất'; + + @override + String get release_notes => 'ghi chú phát hành'; + + @override + String get pick_color_scheme => 'Chọn chủ đề màu sắc'; + + @override + String get save => 'Lưu'; + + @override + String get choose_the_device => 'Chọn thiết bị:'; + + @override + String get multiple_device_connected => + 'Có nhiều thiết bị kết nối.\nChọn thiết bị mà bạn muốn thực hiện hành động này'; + + @override + String get nothing_found => 'Không tìm thấy gì'; + + @override + String get the_box_is_empty => 'Hộp trống'; + + @override + String get top_artists => 'Những Nghệ Sĩ Hàng Đầu'; + + @override + String get top_albums => 'Những Album Hàng Đầu'; + + @override + String get this_week => 'Tuần này'; + + @override + String get this_month => 'Tháng này'; + + @override + String get last_6_months => '6 tháng qua'; + + @override + String get this_year => 'Năm nay'; + + @override + String get last_2_years => '2 năm qua'; + + @override + String get all_time => 'Mọi thời đại'; + + @override + String powered_by_provider(Object providerName) { + return 'Cung cấp bởi $providerName'; + } + + @override + String get email => 'Email'; + + @override + String get profile_followers => 'Người theo dõi'; + + @override + String get birthday => 'Ngày sinh'; + + @override + String get subscription => 'Gói cước'; + + @override + String get not_born => 'Chưa sinh'; + + @override + String get hacker => 'Tin tặc'; + + @override + String get profile => 'Hồ sơ'; + + @override + String get no_name => 'Không có tên'; + + @override + String get edit => 'Chỉnh sửa'; + + @override + String get user_profile => 'Hồ sơ người dùng'; + + @override + String count_plays(Object count) { + return '$count lần phát'; + } + + @override + String get streaming_fees_hypothetical => + '*Tính toán dựa trên thanh toán của Spotify cho mỗi lần phát\ntừ \$0.003 đến \$0.005. Đây là một tính toán giả định để\ngive người dùng cái nhìn về số tiền họ sẽ chi trả cho các nghệ sĩ nếu họ nghe\nbài hát của họ trên Spotify.'; + + @override + String get minutes_listened => 'Thời gian nghe'; + + @override + String get streamed_songs => 'Bài hát đã phát'; + + @override + String count_streams(Object count) { + return '$count lượt phát'; + } + + @override + String get owned_by_you => 'Thuộc sở hữu của bạn'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl đã sao chép vào bảng tạm'; + } + + @override + String get hipotetical_calculation => + '*Điều này được tính toán dựa trên khoản thanh toán trung bình mỗi luồng của nền tảng phát nhạc trực tuyến là \$0,003 đến \$0,005. Đây là một phép tính giả định để cung cấp cho người dùng cái nhìn sâu sắc về số tiền họ đã trả cho các nghệ sĩ nếu họ nghe bài hát của họ trên các nền tảng phát nhạc trực tuyến khác nhau.'; + + @override + String count_mins(Object minutes) { + return '$minutes phút'; + } + + @override + String get summary_minutes => 'phút'; + + @override + String get summary_listened_to_music => 'Đã nghe nhạc'; + + @override + String get summary_songs => 'bài hát'; + + @override + String get summary_streamed_overall => 'Stream tổng cộng'; + + @override + String get summary_owed_to_artists => 'Nợ nghệ sĩ\ntrong tháng này'; + + @override + String get summary_artists => 'nghệ sĩ'; + + @override + String get summary_music_reached_you => 'Âm nhạc đã đến với bạn'; + + @override + String get summary_full_albums => 'album đầy đủ'; + + @override + String get summary_got_your_love => 'Nhận được tình yêu của bạn'; + + @override + String get summary_playlists => 'danh sách phát'; + + @override + String get summary_were_on_repeat => 'Đã được phát lại'; + + @override + String total_money(Object money) { + return 'Tổng cộng $money'; + } + + @override + String get webview_not_found => 'Không tìm thấy Webview'; + + @override + String get webview_not_found_description => + 'Không có runtime Webview nào được cài đặt trên thiết bị của bạn.\nNếu đã cài đặt, hãy đảm bảo rằng nó nằm trong environment PATH\n\nSau khi cài đặt, hãy khởi động lại ứng dụng'; + + @override + String get unsupported_platform => 'Nền tảng không được hỗ trợ'; + + @override + String get cache_music => 'Lưu nhạc vào bộ nhớ đệm'; + + @override + String get open => 'Mở'; + + @override + String get cache_folder => 'Thư mục bộ nhớ đệm'; + + @override + String get export => 'Xuất'; + + @override + String get clear_cache => 'Xóa bộ nhớ đệm'; + + @override + String get clear_cache_confirmation => 'Bạn có muốn xóa bộ nhớ đệm không?'; + + @override + String get export_cache_files => 'Xuất các tệp được lưu trong bộ nhớ đệm'; + + @override + String found_n_files(Object count) { + return 'Tìm thấy $count tệp'; + } + + @override + String get export_cache_confirmation => 'Bạn có muốn xuất các tệp này đến'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return 'Đã xuất $filesExported trên $files tệp'; + } + + @override + String get undo => 'Hoàn tác'; + + @override + String get download_all => 'Tải xuống tất cả'; + + @override + String get add_all_to_playlist => 'Thêm tất cả vào danh sách phát'; + + @override + String get add_all_to_queue => 'Thêm tất cả vào danh sách chờ'; + + @override + String get play_all_next => 'Chơi tất cả tiếp theo'; + + @override + String get pause => 'Tạm dừng'; + + @override + String get view_all => 'Xem tất cả'; + + @override + String get no_tracks_added_yet => 'Có vẻ bạn chưa thêm bất kỳ bài hát nào'; + + @override + String get no_tracks => 'Có vẻ không có bài hát nào ở đây'; + + @override + String get no_tracks_listened_yet => 'Có vẻ bạn chưa nghe gì cả'; + + @override + String get not_following_artists => + 'Bạn không đang theo dõi bất kỳ nghệ sĩ nào'; + + @override + String get no_favorite_albums_yet => + 'Có vẻ bạn chưa thêm album nào vào danh sách yêu thích'; + + @override + String get no_logs_found => 'Không tìm thấy nhật ký'; + + @override + String get youtube_engine => 'Công cụ YouTube'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine chưa được cài đặt'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine chưa được cài đặt trong hệ thống của bạn.'; + } + + @override + String youtube_engine_set_path(Object engine) { + return 'Đảm bảo nó có sẵn trong biến PATH hoặc\nđặt đường dẫn tuyệt đối đến tệp thực thi $engine dưới đây'; + } + + @override + String get youtube_engine_unix_issue_message => + 'Trên macOS/Linux/Unix, việc thiết lập đường dẫn trong .zshrc/.bashrc/.bash_profile v.v. sẽ không hoạt động.\nBạn cần thiết lập đường dẫn trong tệp cấu hình shell'; + + @override + String get download => 'Tải xuống'; + + @override + String get file_not_found => 'Không tìm thấy tệp'; + + @override + String get custom => 'Tùy chỉnh'; + + @override + String get add_custom_url => 'Thêm URL tùy chỉnh'; + + @override + String get edit_port => 'Chỉnh sửa cổng'; + + @override + String get port_helper_msg => + 'Mặc định là -1, có nghĩa là số ngẫu nhiên. Nếu bạn đã cấu hình tường lửa, nên đặt điều này.'; + + @override + String connect_request(Object client) { + return 'Cho phép $client kết nối?'; + } + + @override + String get connection_request_denied => + 'Kết nối bị từ chối. Người dùng đã từ chối quyền truy cập.'; + + @override + String get an_error_occurred => 'Đã xảy ra lỗi'; + + @override + String get copy_to_clipboard => 'Sao chép vào khay nhớ tạm'; + + @override + String get view_logs => 'Xem nhật ký'; + + @override + String get retry => 'Thử lại'; + + @override + String get no_default_metadata_provider_selected => + 'Bạn chưa đặt nhà cung cấp siêu dữ liệu mặc định nào'; + + @override + String get manage_metadata_providers => 'Quản lý nhà cung cấp siêu dữ liệu'; + + @override + String get open_link_in_browser => 'Mở liên kết trong Trình duyệt?'; + + @override + String get do_you_want_to_open_the_following_link => + 'Bạn có muốn mở liên kết sau không'; + + @override + String get unsafe_url_warning => + 'Việc mở các liên kết từ các nguồn không đáng tin cậy có thể không an toàn. Hãy thận trọng!\nBạn cũng có thể sao chép liên kết vào khay nhớ tạm của mình.'; + + @override + String get copy_link => 'Sao chép liên kết'; + + @override + String get building_your_timeline => + 'Đang xây dựng dòng thời gian của bạn dựa trên những gì bạn đã nghe...'; + + @override + String get official => 'Chính thức'; + + @override + String author_name(Object author) { + return 'Tác giả: $author'; + } + + @override + String get third_party => 'Bên thứ ba'; + + @override + String get plugin_requires_authentication => 'Plugin yêu cầu xác thực'; + + @override + String get update_available => 'Có bản cập nhật'; + + @override + String get supports_scrobbling => 'Hỗ trợ scrobbling'; + + @override + String get plugin_scrobbling_info => + 'Plugin này scrobble nhạc của bạn để tạo lịch sử nghe của bạn.'; + + @override + String get default_metadata_source => 'Nguồn siêu dữ liệu mặc định'; + + @override + String get set_default_metadata_source => 'Đặt nguồn siêu dữ liệu mặc định'; + + @override + String get default_audio_source => 'Nguồn âm thanh mặc định'; + + @override + String get set_default_audio_source => 'Đặt nguồn âm thanh mặc định'; + + @override + String get set_default => 'Đặt làm mặc định'; + + @override + String get support => 'Hỗ trợ'; + + @override + String get support_plugin_development => 'Hỗ trợ phát triển plugin'; + + @override + String can_access_name_api(Object name) { + return '- Có thể truy cập API **$name**'; + } + + @override + String get do_you_want_to_install_this_plugin => + 'Bạn có muốn cài đặt plugin này không?'; + + @override + String get third_party_plugin_warning => + 'Plugin này đến từ một kho lưu trữ của bên thứ ba. Vui lòng đảm bảo rằng bạn tin tưởng nguồn trước khi cài đặt.'; + + @override + String get author => 'Tác giả'; + + @override + String get this_plugin_can_do_following => + 'Plugin này có thể làm những việc sau'; + + @override + String get install => 'Cài đặt'; + + @override + String get install_a_metadata_provider => + 'Cài đặt một Nhà cung cấp siêu dữ liệu'; + + @override + String get no_tracks_playing => 'Hiện không có bản nhạc nào đang phát'; + + @override + String get synced_lyrics_not_available => + 'Lời bài hát được đồng bộ hóa không có sẵn cho bài hát này. Vui lòng sử dụng'; + + @override + String get plain_lyrics => 'Lời bài hát thuần túy'; + + @override + String get tab_instead => 'thay thế.'; + + @override + String get disclaimer => 'Miễn trừ trách nhiệm'; + + @override + String get third_party_plugin_dmca_notice => + 'Nhóm Spotube không chịu bất kỳ trách nhiệm nào (bao gồm cả pháp lý) đối với bất kỳ plugin \"Bên thứ ba\" nào.\nVui lòng sử dụng chúng với rủi ro của riêng bạn. Đối với bất kỳ lỗi/vấn đề nào, vui lòng báo cáo chúng cho kho lưu trữ plugin.\n\nNếu bất kỳ plugin \"Bên thứ ba\" nào vi phạm ToS/DMCA của bất kỳ dịch vụ/thực thể pháp lý nào, vui lòng yêu cầu tác giả plugin \"Bên thứ ba\" hoặc nền tảng lưu trữ, ví dụ: GitHub/Codeberg, thực hiện hành động. Tất cả các plugin được liệt kê ở trên (được gắn nhãn \"Bên thứ ba\") đều là các plugin công cộng/do cộng đồng duy trì. Chúng tôi không quản lý chúng, vì vậy chúng tôi không thể thực hiện bất kỳ hành động nào đối với chúng.\n\n'; + + @override + String get input_does_not_match_format => + 'Đầu vào không khớp với định dạng yêu cầu'; + + @override + String get plugins => 'Tiện ích bổ sung'; + + @override + String get paste_plugin_download_url => + 'Dán url tải xuống hoặc url kho lưu trữ GitHub/Codeberg hoặc liên kết trực tiếp đến tệp .smplug'; + + @override + String get download_and_install_plugin_from_url => + 'Tải xuống và cài đặt plugin từ url'; + + @override + String failed_to_add_plugin_error(Object error) { + return 'Không thể thêm plugin: $error'; + } + + @override + String get upload_plugin_from_file => 'Tải lên plugin từ tệp'; + + @override + String get installed => 'Đã cài đặt'; + + @override + String get available_plugins => 'Các plugin có sẵn'; + + @override + String get configure_plugins => + 'Cấu hình nhà cung cấp siêu dữ liệu và tiện ích nguồn âm thanh riêng'; + + @override + String get audio_scrobblers => 'Bộ scrobbler âm thanh'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => 'Nguồn: '; + + @override + String get uncompressed => 'Không nén'; + + @override + String get dab_music_source_description => + 'Dành cho người yêu âm nhạc chất lượng cao. Cung cấp luồng âm thanh chất lượng cao/không nén. Phù hợp bài hát dựa trên ISRC chính xác.'; +} diff --git a/lib/l10n/generated/app_localizations_zh.dart b/lib/l10n/generated/app_localizations_zh.dart new file mode 100644 index 00000000..ac7d4890 --- /dev/null +++ b/lib/l10n/generated/app_localizations_zh.dart @@ -0,0 +1,3051 @@ +// ignore: unused_import +import 'package:intl/intl.dart' as intl; +import 'app_localizations.dart'; + +// ignore_for_file: type=lint + +/// The translations for Chinese (`zh`). +class AppLocalizationsZh extends AppLocalizations { + AppLocalizationsZh([String locale = 'zh']) : super(locale); + + @override + String get guest => '访客'; + + @override + String get browse => '浏览'; + + @override + String get search => '搜索'; + + @override + String get library => '音乐库'; + + @override + String get lyrics => '歌词'; + + @override + String get settings => '设置'; + + @override + String get genre_categories_filter => '筛选类别...'; + + @override + String get genre => '探索歌单'; + + @override + String get personalized => '为你打造'; + + @override + String get featured => '推荐'; + + @override + String get new_releases => '新歌热播'; + + @override + String get songs => '歌曲'; + + @override + String playing_track(Object track) { + return '播放 $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return '这将清空当前的播放队列。$track_length 首歌曲将被移除\n你确定要继续吗?'; + } + + @override + String get load_more => '加载更多'; + + @override + String get playlists => '歌单'; + + @override + String get artists => '艺人'; + + @override + String get albums => '专辑'; + + @override + String get tracks => '歌曲'; + + @override + String get downloads => '下载'; + + @override + String get filter_playlists => '筛选歌单...'; + + @override + String get liked_tracks => '已点赞的歌曲'; + + @override + String get liked_tracks_description => '你点赞过的所有歌曲'; + + @override + String get playlist => '播放列表'; + + @override + String get create_a_playlist => '创建一个歌单'; + + @override + String get update_playlist => '更新播放列表'; + + @override + String get create => '创建'; + + @override + String get cancel => '取消'; + + @override + String get update => '更新'; + + @override + String get playlist_name => '歌单名称'; + + @override + String get name_of_playlist => '歌单的名称'; + + @override + String get description => '描述'; + + @override + String get public => '公开'; + + @override + String get collaborative => '共享协作'; + + @override + String get search_local_tracks => '搜索本地歌曲...'; + + @override + String get play => '播放'; + + @override + String get delete => '删除'; + + @override + String get none => '无'; + + @override + String get sort_a_z => '按字母正序'; + + @override + String get sort_z_a => '按字母倒序'; + + @override + String get sort_artist => '按艺人'; + + @override + String get sort_album => '按专辑'; + + @override + String get sort_duration => '按时长排序'; + + @override + String get sort_tracks => '排序方式'; + + @override + String currently_downloading(Object tracks_length) { + return '正在下载 ($tracks_length)'; + } + + @override + String get cancel_all => '取消全部'; + + @override + String get filter_artist => '筛选艺人...'; + + @override + String followers(Object followers) { + return '$followers 名关注者'; + } + + @override + String get add_artist_to_blacklist => '屏蔽该艺人'; + + @override + String get top_tracks => '热门歌曲'; + + @override + String get fans_also_like => '粉丝也喜欢'; + + @override + String get loading => '加载中...'; + + @override + String get artist => '艺人'; + + @override + String get blacklisted => '已屏蔽'; + + @override + String get following => '关注中'; + + @override + String get follow => '关注'; + + @override + String get artist_url_copied => '艺人的分享链接已复制至剪贴板'; + + @override + String added_to_queue(Object tracks) { + return '已添加 $tracks 首歌曲到播放队列'; + } + + @override + String get filter_albums => '筛选专辑...'; + + @override + String get synced => '同步'; + + @override + String get plain => '无同步'; + + @override + String get shuffle => '随机播放'; + + @override + String get search_tracks => '搜索歌曲...'; + + @override + String get released => '发行时间'; + + @override + String error(Object error) { + return '错误 $error'; + } + + @override + String get title => '标题'; + + @override + String get time => '时长'; + + @override + String get more_actions => '更多操作'; + + @override + String download_count(Object count) { + return '下载 ($count) 首歌曲'; + } + + @override + String add_count_to_playlist(Object count) { + return '添加 ($count) 首歌曲到歌单中'; + } + + @override + String add_count_to_queue(Object count) { + return '添加 ($count) 首歌曲到播放队列中'; + } + + @override + String play_count_next(Object count) { + return '接下来播放 ($count) 首歌曲'; + } + + @override + String get album => '专辑'; + + @override + String copied_to_clipboard(Object data) { + return '已将 $data 复制至剪贴板'; + } + + @override + String add_to_following_playlists(Object track) { + return '添加 $track 到以下播放列表'; + } + + @override + String get add => '添加'; + + @override + String added_track_to_queue(Object track) { + return '添加 $track 到播放队列'; + } + + @override + String get add_to_queue => '添加到播放队列'; + + @override + String track_will_play_next(Object track) { + return '$track 将在下一首播放'; + } + + @override + String get play_next => '下一首播放'; + + @override + String removed_track_from_queue(Object track) { + return '将 $track 从播放队列中移除'; + } + + @override + String get remove_from_queue => '从播放队列移除'; + + @override + String get remove_from_favorites => '取消点赞'; + + @override + String get save_as_favorite => '点赞'; + + @override + String get add_to_playlist => '添加到歌单'; + + @override + String get remove_from_playlist => '从歌单中移除'; + + @override + String get add_to_blacklist => '添加到屏蔽列表'; + + @override + String get remove_from_blacklist => '从屏蔽列表中移除'; + + @override + String get share => '分享'; + + @override + String get mini_player => '小窗模式'; + + @override + String get slide_to_seek => '滑动以前进或后退'; + + @override + String get shuffle_playlist => '随机播放歌单'; + + @override + String get unshuffle_playlist => '取消随机播放歌单'; + + @override + String get previous_track => '上一首歌曲'; + + @override + String get next_track => '下一首歌曲'; + + @override + String get pause_playback => '暂停播放'; + + @override + String get resume_playback => '恢复播放'; + + @override + String get loop_track => '单曲循环'; + + @override + String get no_loop => '无循环'; + + @override + String get repeat_playlist => '歌单循环'; + + @override + String get queue => '播放队列'; + + @override + String get alternative_track_sources => '其它音源'; + + @override + String get download_track => '下载歌曲'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks 首歌曲在播放队列中'; + } + + @override + String get clear_all => '清除全部'; + + @override + String get show_hide_ui_on_hover => '悬停时显示/隐藏控制栏'; + + @override + String get always_on_top => '置顶'; + + @override + String get exit_mini_player => '退出小窗模式'; + + @override + String get download_location => '下载路径'; + + @override + String get local_library => '本地图书馆'; + + @override + String get add_library_location => '添加到图书馆'; + + @override + String get remove_library_location => '从图书馆中删除'; + + @override + String get account => '账户'; + + @override + String get logout => '退出'; + + @override + String get logout_of_this_account => '退出该账户'; + + @override + String get language_region => '语言和地区'; + + @override + String get language => '语言'; + + @override + String get system_default => '系统默认'; + + @override + String get market_place_region => '市场地区'; + + @override + String get recommendation_country => '选择国家与地区以获取对应推荐'; + + @override + String get appearance => '外观'; + + @override + String get layout_mode => '布局类型'; + + @override + String get override_layout_settings => '将覆盖响应式布局设置'; + + @override + String get adaptive => '自适应'; + + @override + String get compact => '紧凑'; + + @override + String get extended => '宽广'; + + @override + String get theme => '主题'; + + @override + String get dark => '深色'; + + @override + String get light => '浅色'; + + @override + String get system => '系统'; + + @override + String get accent_color => '主色调'; + + @override + String get sync_album_color => '匹配封面颜色'; + + @override + String get sync_album_color_description => '选取专辑封面主题色作为主色调'; + + @override + String get playback => '播放'; + + @override + String get audio_quality => '音质'; + + @override + String get high => '高'; + + @override + String get low => '低'; + + @override + String get pre_download_play => '先下后播'; + + @override + String get pre_download_play_description => '先下载歌曲后再播放而非流式播放(推荐带宽较高用户使用)'; + + @override + String get skip_non_music => '跳过非音乐片段(屏蔽赞助商)'; + + @override + String get blacklist_description => '已屏蔽的歌曲与艺人'; + + @override + String get wait_for_download_to_finish => '请等待当前下载任务完成'; + + @override + String get desktop => '桌面端设置'; + + @override + String get close_behavior => '点击关闭按钮行为'; + + @override + String get close => '关闭'; + + @override + String get minimize_to_tray => '最小化到托盘'; + + @override + String get show_tray_icon => '显示托盘图标'; + + @override + String get about => '关于'; + + @override + String get u_love_spotube => '我们明白你喜欢 Spotube'; + + @override + String get check_for_updates => '检查更新'; + + @override + String get about_spotube => '关于 Spotube'; + + @override + String get blacklist => '屏蔽列表'; + + @override + String get please_sponsor => '请赞助/捐赠'; + + @override + String get spotube_description => 'Spotube,一个轻量、跨平台且完全免费的 Spotify 客户端。'; + + @override + String get version => '版本'; + + @override + String get build_number => '构建代码'; + + @override + String get founder => '发起人'; + + @override + String get repository => '源码'; + + @override + String get bug_issues => '缺陷和问题报告'; + + @override + String get made_with => '于孟加拉🇧🇩用 ❤️ 发电'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => '许可证'; + + @override + String get credentials_will_not_be_shared_disclaimer => + '不用担心,软件不会收集或分享任何个人数据给第三方'; + + @override + String get know_how_to_login => '不知道该怎么做?'; + + @override + String get follow_step_by_step_guide => '请按照以下指南进行'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => '请填写所有栏目'; + + @override + String get submit => '提交'; + + @override + String get exit => '退出'; + + @override + String get previous => '上一步'; + + @override + String get next => '下一步'; + + @override + String get done => '完成'; + + @override + String get step_1 => '步骤 1'; + + @override + String get first_go_to => '首先,前往'; + + @override + String get something_went_wrong => '某些地方出现了问题'; + + @override + String get piped_instance => 'Piped 服务器实例'; + + @override + String get piped_description => 'Piped 服务器实例用于匹配歌曲'; + + @override + String get piped_warning => '它们中的一部分可能并不能正常工作。使用时请自行承担风险'; + + @override + String get invidious_instance => 'Invidious服务器实例'; + + @override + String get invidious_description => '用于音轨匹配的Invidious服务器实例'; + + @override + String get invidious_warning => '有些可能无法正常工作。请自行承担风险'; + + @override + String get generate => '生成'; + + @override + String track_exists(Object track) { + return '歌曲 $track 已存在'; + } + + @override + String get replace_downloaded_tracks => '替换已下载的歌曲'; + + @override + String get skip_download_tracks => '下载时跳过已下载的歌曲'; + + @override + String get do_you_want_to_replace => '你确定要替换已下载的歌曲吗??'; + + @override + String get replace => '替换'; + + @override + String get skip => '跳过'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '选择多达 $count 种的类型 $type'; + } + + @override + String get select_genres => '选择曲风'; + + @override + String get add_genres => '添加曲风'; + + @override + String get country => '国家和地区'; + + @override + String get number_of_tracks_generate => '生成歌曲的数目'; + + @override + String get acousticness => '原声程度'; + + @override + String get danceability => '律动感'; + + @override + String get energy => '冲击感'; + + @override + String get instrumentalness => '歌唱部分占比'; + + @override + String get liveness => '现场感'; + + @override + String get loudness => '响度'; + + @override + String get speechiness => '朗诵比例'; + + @override + String get valence => '心理感受'; + + @override + String get popularity => '流行度'; + + @override + String get key => '曲调'; + + @override + String get duration => '歌曲时长 (s)'; + + @override + String get tempo => '分钟节拍数 (BPM)'; + + @override + String get mode => '旋律重复度'; + + @override + String get time_signature => '音符时值'; + + @override + String get short => '短'; + + @override + String get medium => '中'; + + @override + String get long => '长'; + + @override + String get min => '最低'; + + @override + String get max => '最高'; + + @override + String get target => '目标'; + + @override + String get moderate => '中'; + + @override + String get deselect_all => '取消全选'; + + @override + String get select_all => '全选'; + + @override + String get are_you_sure => '你确定吗?'; + + @override + String get generating_playlist => '正在生成你的自定义歌单...'; + + @override + String selected_count_tracks(Object count) { + return '已选择 $count 首歌曲'; + } + + @override + String get download_warning => + '如果你大量下载这些歌曲,你显然在侵犯音乐的版权并对音乐创作社区造成了伤害。我希望你能意识到这一点。永远要尊重并支持艺术家们的辛勤工作'; + + @override + String get download_ip_ban_warning => + '小心,如果出现超出正常的下载请求那你的 IP 可能会被 YouTube 封禁,这意味着你的设备将在长达 2-3 个月的时间内无法使用该 IP 访问 YouTube(即使你没登录)。Spotube 对此不承担任何责任'; + + @override + String get by_clicking_accept_terms => '点击 \'同意\' 代表着你同意以下的条款'; + + @override + String get download_agreement_1 => '我明白侵犯音乐版权是一件不好的事情'; + + @override + String get download_agreement_2 => '我将尽可能支持艺术家的工作。我现在之所以做不到是因为缺乏资金来购买正版'; + + @override + String get download_agreement_3 => + '我完全了解我的 IP 存在被 YouTube的风险。我同意 Spotube 的所有者与贡献者们无须对我目前的行为所导致的任何后果负责'; + + @override + String get decline => '拒绝'; + + @override + String get accept => '同意'; + + @override + String get details => '详情'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => '频道'; + + @override + String get likes => '赞'; + + @override + String get dislikes => '踩'; + + @override + String get views => '浏览次数'; + + @override + String get streamUrl => '播放流 URL'; + + @override + String get stop => '停止'; + + @override + String get sort_newest => '按添加日期正序'; + + @override + String get sort_oldest => '按添加日期倒序'; + + @override + String get sleep_timer => '睡眠定时器'; + + @override + String mins(Object minutes) { + return '$minutes 分'; + } + + @override + String hours(Object hours) { + return '$hours 时'; + } + + @override + String hour(Object hours) { + return '$hours 时'; + } + + @override + String get custom_hours => '自定义时间'; + + @override + String get logs => '日志'; + + @override + String get developers => '开发者'; + + @override + String get not_logged_in => '你尚未登录'; + + @override + String get search_mode => '搜索模式'; + + @override + String get audio_source => '音频源'; + + @override + String get ok => '确定'; + + @override + String get failed_to_encrypt => '加密失败'; + + @override + String get encryption_failed_warning => + 'Spotube使用加密来安全地存储您的数据。但是失败了。因此,它将回退到不安全的存储\n如果您使用Linux,请确保已安装gnome-keyring、kde-wallet和keepassxc等秘密服务'; + + @override + String get querying_info => '正在查询信息...'; + + @override + String get piped_api_down => 'Piped API不可用'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return '当前Piped实例$pipedInstance不可用\n\n请更改实例或将\'API类型\'更改为官方YouTube API\n\n更改后请确保重新启动应用程序'; + } + + @override + String get you_are_offline => '您当前处于离线状态'; + + @override + String get connection_restored => '您的互联网连接已恢复'; + + @override + String get use_system_title_bar => '使用系统标题栏'; + + @override + String get crunching_results => '处理结果中...'; + + @override + String get search_to_get_results => '搜索以获取结果'; + + @override + String get use_amoled_mode => '使用 AMOLED 模式'; + + @override + String get pitch_dark_theme => '深色主题'; + + @override + String get normalize_audio => '标准化音频'; + + @override + String get change_cover => '更改封面'; + + @override + String get add_cover => '添加封面'; + + @override + String get restore_defaults => '恢复默认值'; + + @override + String get download_music_format => '下载音乐格式'; + + @override + String get streaming_music_format => '流媒体音乐格式'; + + @override + String get download_music_quality => '下载音乐质量'; + + @override + String get streaming_music_quality => '流媒体音乐质量'; + + @override + String get login_with_lastfm => '使用 Last.fm 登录'; + + @override + String get connect => '连接'; + + @override + String get disconnect_lastfm => '断开 Last.fm 连接'; + + @override + String get disconnect => '断开连接'; + + @override + String get username => '用户名'; + + @override + String get password => '密码'; + + @override + String get login => '登录'; + + @override + String get login_with_your_lastfm => '使用您的 Last.fm 帐户登录'; + + @override + String get scrobble_to_lastfm => '在 Last.fm 上记录播放'; + + @override + String get go_to_album => '前往专辑'; + + @override + String get discord_rich_presence => 'Discord 丰富展现'; + + @override + String get browse_all => '浏览全部'; + + @override + String get genres => '音乐类型'; + + @override + String get explore_genres => '探索音乐类型'; + + @override + String get friends => '朋友'; + + @override + String get no_lyrics_available => '抱歉,无法找到此曲的歌词'; + + @override + String get start_a_radio => '开始收听电台'; + + @override + String get how_to_start_radio => '您想如何开始收听电台?'; + + @override + String get replace_queue_question => '您想要替换当前队列还是追加到队列?'; + + @override + String get endless_playback => '无尽播放'; + + @override + String get delete_playlist => '删除播放列表'; + + @override + String get delete_playlist_confirmation => '您确定要删除此播放列表吗?'; + + @override + String get local_tracks => '本地音轨'; + + @override + String get local_tab => '本地'; + + @override + String get song_link => '歌曲链接'; + + @override + String get skip_this_nonsense => '跳过此无聊内容'; + + @override + String get freedom_of_music => '“音乐的自由”'; + + @override + String get freedom_of_music_palm => '“音乐的自由掌握在您手中”'; + + @override + String get get_started => '让我们开始吧'; + + @override + String get youtube_source_description => '推荐并且效果最佳。'; + + @override + String get piped_source_description => '感觉自由?与YouTube一样但更自由。'; + + @override + String get jiosaavn_source_description => '最适合南亚地区。'; + + @override + String get invidious_source_description => '类似于Piped,但可用性更高。'; + + @override + String highest_quality(Object quality) { + return '最高音质:$quality'; + } + + @override + String get select_audio_source => '选择音频源'; + + @override + String get endless_playback_description => '自动将新歌曲添加到队列的末尾'; + + @override + String get choose_your_region => '选择您的地区'; + + @override + String get choose_your_region_description => '这将帮助Spotube为您的位置显示正确的内容。'; + + @override + String get choose_your_language => '选择您的语言'; + + @override + String get help_project_grow => '帮助这个项目成长'; + + @override + String get help_project_grow_description => + 'Spotube是一个开源项目。您可以通过为项目做出贡献、报告错误或建议新功能来帮助该项目成长。'; + + @override + String get contribute_on_github => '在GitHub上做出贡献'; + + @override + String get donate_on_open_collective => '在Open Collective上捐款'; + + @override + String get browse_anonymously => '匿名浏览'; + + @override + String get enable_connect => '启用连接'; + + @override + String get enable_connect_description => '从其他设备控制Spotube'; + + @override + String get devices => '设备'; + + @override + String get select => '选择'; + + @override + String connect_client_alert(Object client) { + return '您正在被 $client 控制'; + } + + @override + String get this_device => '此设备'; + + @override + String get remote => '远程'; + + @override + String get stats => '统计'; + + @override + String and_n_more(Object count) { + return '和 $count 更多'; + } + + @override + String get recently_played => '最近播放'; + + @override + String get browse_more => '浏览更多'; + + @override + String get no_title => '没有标题'; + + @override + String get not_playing => '未播放'; + + @override + String get epic_failure => '史诗级失败!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '已将 $tracks_length 首曲目添加到队列'; + } + + @override + String get spotube_has_an_update => 'Spotube 有更新'; + + @override + String get download_now => '立即下载'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum 已发布'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version 已发布'; + } + + @override + String get read_the_latest => '阅读最新'; + + @override + String get release_notes => '版本说明'; + + @override + String get pick_color_scheme => '选择配色方案'; + + @override + String get save => '保存'; + + @override + String get choose_the_device => '选择设备:'; + + @override + String get multiple_device_connected => '已连接多个设备。\n选择您希望执行此操作的设备'; + + @override + String get nothing_found => '未找到任何内容'; + + @override + String get the_box_is_empty => '箱子为空'; + + @override + String get top_artists => '热门艺术家'; + + @override + String get top_albums => '热门专辑'; + + @override + String get this_week => '本周'; + + @override + String get this_month => '本月'; + + @override + String get last_6_months => '过去6个月'; + + @override + String get this_year => '今年'; + + @override + String get last_2_years => '过去2年'; + + @override + String get all_time => '所有时间'; + + @override + String powered_by_provider(Object providerName) { + return '由 $providerName 提供支持'; + } + + @override + String get email => '电子邮件'; + + @override + String get profile_followers => '关注者'; + + @override + String get birthday => '生日'; + + @override + String get subscription => '订阅'; + + @override + String get not_born => '尚未出生'; + + @override + String get hacker => '黑客'; + + @override + String get profile => '个人资料'; + + @override + String get no_name => '无名'; + + @override + String get edit => '编辑'; + + @override + String get user_profile => '用户资料'; + + @override + String count_plays(Object count) { + return '$count 次播放'; + } + + @override + String get streaming_fees_hypothetical => + '*基于 Spotify 每次播放的支付金额\n从 \$0.003 到 \$0.005 计算。这是一个假设性的\n计算,旨在让用户了解如果他们在 Spotify 上收听\n这些歌曲,可能会付给艺术家的金额。'; + + @override + String get minutes_listened => '听的分钟数'; + + @override + String get streamed_songs => '已流媒体歌曲'; + + @override + String count_streams(Object count) { + return '$count 次流媒体'; + } + + @override + String get owned_by_you => '由您拥有'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl 已复制到剪贴板'; + } + + @override + String get hipotetical_calculation => + '*这是根据在线音乐流媒体平台每流平均支付0.003美元至0.005美元计算得出的。这是一个假设性的计算,旨在让用户了解如果他们在不同的音乐流媒体平台上收听歌曲,他们将需要向艺人支付多少费用。'; + + @override + String count_mins(Object minutes) { + return '$minutes 分钟'; + } + + @override + String get summary_minutes => '分钟'; + + @override + String get summary_listened_to_music => '听音乐'; + + @override + String get summary_songs => '歌曲'; + + @override + String get summary_streamed_overall => '总体流媒体'; + + @override + String get summary_owed_to_artists => '本月欠艺术家的'; + + @override + String get summary_artists => '艺术家的'; + + @override + String get summary_music_reached_you => '音乐触及了你'; + + @override + String get summary_full_albums => '完整专辑'; + + @override + String get summary_got_your_love => '获得了你的爱'; + + @override + String get summary_playlists => '播放列表'; + + @override + String get summary_were_on_repeat => '已重复播放'; + + @override + String total_money(Object money) { + return '总计 $money'; + } + + @override + String get webview_not_found => '未找到 Webview'; + + @override + String get webview_not_found_description => + '您的设备中未安装 Webview 运行时。\n如果已安装,请确保它在 environment PATH 中\n\n安装后,重新启动应用程序'; + + @override + String get unsupported_platform => '不支持的平台'; + + @override + String get cache_music => '缓存音乐'; + + @override + String get open => '打开'; + + @override + String get cache_folder => '缓存文件夹'; + + @override + String get export => '导出'; + + @override + String get clear_cache => '清除缓存'; + + @override + String get clear_cache_confirmation => '您要清除缓存吗?'; + + @override + String get export_cache_files => '导出缓存文件'; + + @override + String found_n_files(Object count) { + return '找到 $count 个文件'; + } + + @override + String get export_cache_confirmation => '您要导出这些文件到'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '导出了 $filesExported / $files 个文件'; + } + + @override + String get undo => '撤销'; + + @override + String get download_all => '下载全部'; + + @override + String get add_all_to_playlist => '将全部添加到播放列表'; + + @override + String get add_all_to_queue => '将全部添加到队列'; + + @override + String get play_all_next => '播放全部下一首'; + + @override + String get pause => '暂停'; + + @override + String get view_all => '查看所有'; + + @override + String get no_tracks_added_yet => '看起来你还没有添加任何曲目'; + + @override + String get no_tracks => '看起来这里没有任何曲目'; + + @override + String get no_tracks_listened_yet => '看起来你还没有听任何东西'; + + @override + String get not_following_artists => '你没有关注任何艺术家'; + + @override + String get no_favorite_albums_yet => '看起来你还没有将任何专辑添加到收藏夹'; + + @override + String get no_logs_found => '未找到日志'; + + @override + String get youtube_engine => 'YouTube 引擎'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine 未安装'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine 未在您的系统中安装。'; + } + + @override + String youtube_engine_set_path(Object engine) { + return '确保它可用在 PATH 变量中,或\n设置 $engine 可执行文件的绝对路径'; + } + + @override + String get youtube_engine_unix_issue_message => + '在 macOS/Linux/Unix 类操作系统中,在 .zshrc/.bashrc/.bash_profile 等文件中设置路径无效。\n您需要在 shell 配置文件中设置路径'; + + @override + String get download => '下载'; + + @override + String get file_not_found => '文件未找到'; + + @override + String get custom => '自定义'; + + @override + String get add_custom_url => '添加自定义 URL'; + + @override + String get edit_port => '编辑端口'; + + @override + String get port_helper_msg => '默认值为-1,表示随机数。如果您已配置防火墙,建议设置此项。'; + + @override + String connect_request(Object client) { + return '允许 $client 连接吗?'; + } + + @override + String get connection_request_denied => '连接被拒绝。用户拒绝访问。'; + + @override + String get an_error_occurred => '发生错误'; + + @override + String get copy_to_clipboard => '复制到剪贴板'; + + @override + String get view_logs => '查看日志'; + + @override + String get retry => '重试'; + + @override + String get no_default_metadata_provider_selected => '您未设置默认元数据提供者'; + + @override + String get manage_metadata_providers => '管理元数据提供者'; + + @override + String get open_link_in_browser => '在浏览器中打开链接?'; + + @override + String get do_you_want_to_open_the_following_link => '您想打开以下链接吗'; + + @override + String get unsafe_url_warning => '从不受信任的来源打开链接可能不安全。请谨慎!\n您也可以将链接复制到剪贴板。'; + + @override + String get copy_link => '复制链接'; + + @override + String get building_your_timeline => '正在根据您的收听记录构建您的时间线...'; + + @override + String get official => '官方'; + + @override + String author_name(Object author) { + return '作者:$author'; + } + + @override + String get third_party => '第三方'; + + @override + String get plugin_requires_authentication => '插件需要身份验证'; + + @override + String get update_available => '有可用更新'; + + @override + String get supports_scrobbling => '支持 Scrobbling'; + + @override + String get plugin_scrobbling_info => '此插件会 scrobble 您的音乐以生成您的收听历史记录。'; + + @override + String get default_metadata_source => '默认元数据源'; + + @override + String get set_default_metadata_source => '设置默认元数据源'; + + @override + String get default_audio_source => '默认音频源'; + + @override + String get set_default_audio_source => '设置默认音频源'; + + @override + String get set_default => '设为默认'; + + @override + String get support => '支持'; + + @override + String get support_plugin_development => '支持插件开发'; + + @override + String can_access_name_api(Object name) { + return '- 可以访问 **$name** API'; + } + + @override + String get do_you_want_to_install_this_plugin => '您想安装此插件吗?'; + + @override + String get third_party_plugin_warning => '此插件来自第三方存储库。请在安装前确保您信任此来源。'; + + @override + String get author => '作者'; + + @override + String get this_plugin_can_do_following => '此插件可以执行以下操作'; + + @override + String get install => '安装'; + + @override + String get install_a_metadata_provider => '安装元数据提供者'; + + @override + String get no_tracks_playing => '当前没有播放任何曲目'; + + @override + String get synced_lyrics_not_available => '此歌曲的同步歌词不可用。请使用'; + + @override + String get plain_lyrics => '纯歌词'; + + @override + String get tab_instead => '选项卡。'; + + @override + String get disclaimer => '免责声明'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube 团队对任何“第三方”插件不承担任何责任(包括法律责任)。\n请自行承担风险使用。对于任何错误/问题,请向插件存储库报告。\n\n如果任何“第三方”插件违反了任何服务/法律实体的服务条款/DMCA,请要求该“第三方”插件作者或托管平台(例如 GitHub/Codeberg)采取行动。上面列出的(标记为“第三方”)都是公共/社区维护的插件。我们不对此类插件进行管理,因此无法对其采取任何行动。\n\n'; + + @override + String get input_does_not_match_format => '输入与所需格式不匹配'; + + @override + String get plugins => '插件'; + + @override + String get paste_plugin_download_url => + '粘贴下载 URL、GitHub/Codeberg 存储库 URL 或 .smplug 文件的直接链接'; + + @override + String get download_and_install_plugin_from_url => '从 URL 下载并安装插件'; + + @override + String failed_to_add_plugin_error(Object error) { + return '添加插件失败:$error'; + } + + @override + String get upload_plugin_from_file => '从文件上传插件'; + + @override + String get installed => '已安装'; + + @override + String get available_plugins => '可用插件'; + + @override + String get configure_plugins => '配置您自己的元数据提供者和音频源插件'; + + @override + String get audio_scrobblers => '音频 Scrobblers'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => '来源:'; + + @override + String get uncompressed => '无损'; + + @override + String get dab_music_source_description => + '适合发烧友。提供高质量/无损音频流。基于 ISRC 的精确曲目匹配。'; +} + +/// The translations for Chinese, as used in Taiwan (`zh_TW`). +class AppLocalizationsZhTw extends AppLocalizationsZh { + AppLocalizationsZhTw() : super('zh_TW'); + + @override + String get guest => '訪客'; + + @override + String get browse => '瀏覽'; + + @override + String get search => '搜尋'; + + @override + String get library => '音樂庫'; + + @override + String get lyrics => '歌詞'; + + @override + String get settings => '設定'; + + @override + String get genre_categories_filter => '過濾分類...'; + + @override + String get genre => '探索歌單'; + + @override + String get personalized => '為你打造'; + + @override + String get featured => '推薦'; + + @override + String get new_releases => '新歌熱播'; + + @override + String get songs => '歌曲'; + + @override + String playing_track(Object track) { + return '播放 $track'; + } + + @override + String queue_clear_alert(Object track_length) { + return '這將清空目前的播放清單。$track_length 首歌曲將被移除\n你確定要繼續嗎?'; + } + + @override + String get load_more => '載入更多'; + + @override + String get playlists => '歌單'; + + @override + String get artists => '藝人'; + + @override + String get albums => '專輯'; + + @override + String get tracks => '歌曲'; + + @override + String get downloads => '下載'; + + @override + String get filter_playlists => '過濾歌單...'; + + @override + String get liked_tracks => '已按讚的歌曲'; + + @override + String get liked_tracks_description => '你按過讚的所有歌曲'; + + @override + String get playlist => '播放清單'; + + @override + String get create_a_playlist => '建立一個歌單'; + + @override + String get update_playlist => '更新播放清單'; + + @override + String get create => '建立'; + + @override + String get cancel => '取消'; + + @override + String get update => '更新'; + + @override + String get playlist_name => '歌單名稱'; + + @override + String get name_of_playlist => '歌單的名稱'; + + @override + String get description => '說明'; + + @override + String get public => '公開'; + + @override + String get collaborative => '共享協作'; + + @override + String get search_local_tracks => '搜尋本地歌曲...'; + + @override + String get play => '播放'; + + @override + String get delete => '刪除'; + + @override + String get none => '無'; + + @override + String get sort_a_z => '依字母順序'; + + @override + String get sort_z_a => '依字母倒序'; + + @override + String get sort_artist => '按藝人'; + + @override + String get sort_album => '按專輯'; + + @override + String get sort_duration => '依長度排序'; + + @override + String get sort_tracks => '排序方式'; + + @override + String currently_downloading(Object tracks_length) { + return '正在下載 ($tracks_length)'; + } + + @override + String get cancel_all => '取消全部'; + + @override + String get filter_artist => '過濾藝人...'; + + @override + String followers(Object followers) { + return '$followers 名追蹤者'; + } + + @override + String get add_artist_to_blacklist => '封鎖該藝人'; + + @override + String get top_tracks => '熱門歌曲'; + + @override + String get fans_also_like => '粉絲也喜歡'; + + @override + String get loading => '載入中...'; + + @override + String get artist => '藝人'; + + @override + String get blacklisted => '已封鎖'; + + @override + String get following => '關注中'; + + @override + String get follow => '關注'; + + @override + String get artist_url_copied => '此名藝人的分享連結已複製至剪貼簿'; + + @override + String added_to_queue(Object tracks) { + return '已新增 $tracks 首歌曲到播放清單'; + } + + @override + String get filter_albums => '過濾專輯...'; + + @override + String get synced => '同步'; + + @override + String get plain => '未同步'; + + @override + String get shuffle => '隨機播放'; + + @override + String get search_tracks => '搜尋歌曲...'; + + @override + String get released => '發表時間'; + + @override + String error(Object error) { + return '發生錯誤: $error'; + } + + @override + String get title => '標題'; + + @override + String get time => '時長'; + + @override + String get more_actions => '更多動作'; + + @override + String download_count(Object count) { + return '下載 ($count) 首歌曲'; + } + + @override + String add_count_to_playlist(Object count) { + return '將 ($count) 首歌曲新增到歌單中'; + } + + @override + String add_count_to_queue(Object count) { + return '新增 ($count) 首歌曲到播放清單'; + } + + @override + String play_count_next(Object count) { + return '接下來將播放 ($count) 首歌曲'; + } + + @override + String get album => '專輯'; + + @override + String copied_to_clipboard(Object data) { + return '已將 $data 複製至剪貼簿'; + } + + @override + String add_to_following_playlists(Object track) { + return '新增 $track 到以下播放清單'; + } + + @override + String get add => '新增'; + + @override + String added_track_to_queue(Object track) { + return '新增 $track 到播放清單'; + } + + @override + String get add_to_queue => '新增至播放清單'; + + @override + String track_will_play_next(Object track) { + return '$track 將在下一首播放'; + } + + @override + String get play_next => '下一首播放'; + + @override + String removed_track_from_queue(Object track) { + return '將 $track 從播放清單移除'; + } + + @override + String get remove_from_queue => '從播放清單移除'; + + @override + String get remove_from_favorites => '取消按讚'; + + @override + String get save_as_favorite => '按讚'; + + @override + String get add_to_playlist => '新增到歌單'; + + @override + String get remove_from_playlist => '從歌單移除'; + + @override + String get add_to_blacklist => '新增到已封鎖清單'; + + @override + String get remove_from_blacklist => '從已封鎖清單移除'; + + @override + String get share => '分享'; + + @override + String get mini_player => '小窗模式'; + + @override + String get slide_to_seek => '滑動以前進或後退'; + + @override + String get shuffle_playlist => '隨機播放歌單'; + + @override + String get unshuffle_playlist => '取消隨機播放歌單'; + + @override + String get previous_track => '上一首歌曲'; + + @override + String get next_track => '下一首歌'; + + @override + String get pause_playback => '暫停播放'; + + @override + String get resume_playback => '恢復播放'; + + @override + String get loop_track => '單曲循環'; + + @override + String get no_loop => '無循環'; + + @override + String get repeat_playlist => '歌單循環'; + + @override + String get queue => '播放清單'; + + @override + String get alternative_track_sources => '其它音源'; + + @override + String get download_track => '下載歌曲'; + + @override + String tracks_in_queue(Object tracks) { + return '$tracks 首歌曲在播放清單中'; + } + + @override + String get clear_all => '清除全部'; + + @override + String get show_hide_ui_on_hover => '游標暫留時顯示 / 隱藏控制列'; + + @override + String get always_on_top => '置頂'; + + @override + String get exit_mini_player => '退出小窗模式'; + + @override + String get download_location => '下載路徑'; + + @override + String get local_library => '本地媒體庫'; + + @override + String get add_library_location => '新增至媒體庫'; + + @override + String get remove_library_location => '從媒體庫移除'; + + @override + String get account => '帳戶'; + + @override + String get logout => '退出'; + + @override + String get logout_of_this_account => '退出該帳戶'; + + @override + String get language_region => '語言與地區'; + + @override + String get language => '語言'; + + @override + String get system_default => '系統預設'; + + @override + String get market_place_region => '市集地區'; + + @override + String get recommendation_country => '請選擇國家與地區以取得對應的音樂推薦'; + + @override + String get appearance => '外觀'; + + @override + String get layout_mode => '佈局類型'; + + @override + String get override_layout_settings => '將覆寫響應式佈局設定'; + + @override + String get adaptive => '響應式'; + + @override + String get compact => '緊湊'; + + @override + String get extended => '寬闊'; + + @override + String get theme => '主題'; + + @override + String get dark => '深色'; + + @override + String get light => '淺色'; + + @override + String get system => '依循系統'; + + @override + String get accent_color => '主色調'; + + @override + String get sync_album_color => '符合封面顏色'; + + @override + String get sync_album_color_description => '選取專輯封面主題色為主色調'; + + @override + String get playback => '播放'; + + @override + String get audio_quality => '音質'; + + @override + String get high => '高'; + + @override + String get low => '低'; + + @override + String get pre_download_play => '下載後播放'; + + @override + String get pre_download_play_description => '先下載歌曲後再播放而非串流播放(建議頻寬較高使用者使用)'; + + @override + String get skip_non_music => '跳過非音樂片段(跳過贊助商廣告)'; + + @override + String get blacklist_description => '已封鎖的歌曲與藝人'; + + @override + String get wait_for_download_to_finish => '請等待目前下載工作完成'; + + @override + String get desktop => '桌面版設定'; + + @override + String get close_behavior => '點選關閉按鈕行為'; + + @override + String get close => '關閉'; + + @override + String get minimize_to_tray => '最小化到工作列'; + + @override + String get show_tray_icon => '顯示工作列圖示'; + + @override + String get about => '關於'; + + @override + String get u_love_spotube => '我們明白你喜歡 Spotube'; + + @override + String get check_for_updates => '檢查更新'; + + @override + String get about_spotube => '關於 Spotube'; + + @override + String get blacklist => '黑名單'; + + @override + String get please_sponsor => '請考慮贊助或捐款'; + + @override + String get spotube_description => 'Spotube,一款輕量、跨平台且完全免費的 Spotify 用戶端。'; + + @override + String get version => '版本'; + + @override + String get build_number => '建置編號'; + + @override + String get founder => '發起人'; + + @override + String get repository => '專案儲存庫'; + + @override + String get bug_issues => '缺陷與問題報告'; + + @override + String get made_with => '於孟加拉🇧🇩用 ❤️ 發電'; + + @override + String get kingkor_roy_tirtho => 'Kingkor Roy Tirtho'; + + @override + String copyright(Object current_year) { + return '© 2021-$current_year Kingkor Roy Tirtho'; + } + + @override + String get license => '授權'; + + @override + String get credentials_will_not_be_shared_disclaimer => + '您大可放心,軟體不會收集或分享任何個人資料給第三方'; + + @override + String get know_how_to_login => '不知道該怎麼辦?'; + + @override + String get follow_step_by_step_guide => '請依照以下說明進行'; + + @override + String cookie_name_cookie(Object name) { + return '$name Cookie'; + } + + @override + String get fill_in_all_fields => '請填入所有欄位'; + + @override + String get submit => '提交'; + + @override + String get exit => '退出'; + + @override + String get previous => '上一步'; + + @override + String get next => '下一步'; + + @override + String get done => '完成'; + + @override + String get step_1 => '步驟 1'; + + @override + String get first_go_to => '首先,前往'; + + @override + String get something_went_wrong => '某些地方出現了問題'; + + @override + String get piped_instance => 'Piped 伺服器實例'; + + @override + String get piped_description => 'Piped 伺服器實例用於匹配歌曲'; + + @override + String get piped_warning => '它們之中的一部分可能無法正常運作。使用時請自行承擔風險'; + + @override + String get invidious_instance => 'Invidious 伺服器實例'; + + @override + String get invidious_description => '用於音軌匹配的 Invidious 伺服器實例'; + + @override + String get invidious_warning => '有些可能無法正常運作。請自行承擔風險'; + + @override + String get generate => '生成'; + + @override + String track_exists(Object track) { + return '曲目 $track 已存在'; + } + + @override + String get replace_downloaded_tracks => '替換已下載的歌曲'; + + @override + String get skip_download_tracks => '下載時跳過已下載的歌曲'; + + @override + String get do_you_want_to_replace => '你確定要取代已下載的歌曲嗎??'; + + @override + String get replace => '取代'; + + @override + String get skip => '跳過'; + + @override + String select_up_to_count_type(Object count, Object type) { + return '選擇最多 $count 種的類型 $type'; + } + + @override + String get select_genres => '選擇曲風'; + + @override + String get add_genres => '新增曲風'; + + @override + String get country => '國家和地區'; + + @override + String get number_of_tracks_generate => '產生歌曲的數目'; + + @override + String get acousticness => '原聲程度'; + + @override + String get danceability => '律動感'; + + @override + String get energy => '衝擊感'; + + @override + String get instrumentalness => '歌唱部分佔比'; + + @override + String get liveness => '現場感'; + + @override + String get loudness => '響度'; + + @override + String get speechiness => '朗誦比例'; + + @override + String get valence => '心理感受'; + + @override + String get popularity => '流行度'; + + @override + String get key => '曲調'; + + @override + String get duration => '歌曲長度 (s)'; + + @override + String get tempo => '每分鐘拍數 (BPM)'; + + @override + String get mode => '旋律重複度'; + + @override + String get time_signature => '音符時值'; + + @override + String get short => '短'; + + @override + String get medium => '中'; + + @override + String get long => '長'; + + @override + String get min => '最低'; + + @override + String get max => '最高'; + + @override + String get target => '目標'; + + @override + String get moderate => '中'; + + @override + String get deselect_all => '取消全選'; + + @override + String get select_all => '全選'; + + @override + String get are_you_sure => '你確定嗎?'; + + @override + String get generating_playlist => '正在產生你的自訂歌單...'; + + @override + String selected_count_tracks(Object count) { + return '已選取 $count 首歌曲'; + } + + @override + String get download_warning => + '如果你大量下載這些歌曲,你顯然在侵犯音樂的版權並對音樂創作社區造成了傷害。我希望你能意識到這一點。永遠要尊重並支持藝術家們的辛勤工作'; + + @override + String get download_ip_ban_warning => + '小心,如果出現超出正常的下載請求,那你的 IP 可能會被 YouTube 封鎖,這意味著你的裝置將在長達 2-3 個月的時間內無法使用該 IP 訪問 YouTube(即使你沒登入)。Spotube 不會因而承擔任何責任'; + + @override + String get by_clicking_accept_terms => '點擊 \'同意\' 代表你同意以下的條款'; + + @override + String get download_agreement_1 => '我明白侵害音樂版權是一件不好的事'; + + @override + String get download_agreement_2 => '我將盡可能支持藝術家的工作。我現在之所以做不到是因為缺乏資金來購買正版'; + + @override + String get download_agreement_3 => + '我完全了解我的 IP 存在被 YouTube 封鎖的風險。並且我明白 Spotube 的擁有者與貢獻者們無須對我目前的行為所導致的任何後果負責'; + + @override + String get decline => '拒絕'; + + @override + String get accept => '同意'; + + @override + String get details => '詳細資訊'; + + @override + String get youtube => 'YouTube'; + + @override + String get channel => '頻道'; + + @override + String get likes => '讚'; + + @override + String get dislikes => '倒讚'; + + @override + String get views => '瀏覽次數'; + + @override + String get streamUrl => '播放串流 URL'; + + @override + String get stop => '停止'; + + @override + String get sort_newest => '依新增日期順序'; + + @override + String get sort_oldest => '依新增日期倒序'; + + @override + String get sleep_timer => '睡眠計時器'; + + @override + String mins(Object minutes) { + return '$minutes 分'; + } + + @override + String hours(Object hours) { + return '$hours 時'; + } + + @override + String hour(Object hours) { + return '$hours 時'; + } + + @override + String get custom_hours => '自訂時長'; + + @override + String get logs => '記錄檔(Log)'; + + @override + String get developers => '開發者'; + + @override + String get not_logged_in => '你尚未登入'; + + @override + String get search_mode => '搜尋模式'; + + @override + String get audio_source => '音訊來源'; + + @override + String get ok => '確定'; + + @override + String get failed_to_encrypt => '加密失敗'; + + @override + String get encryption_failed_warning => + 'Spotube使用加密來安全地儲存您的資料。但是失敗了。因此,它將回退到不安全的儲存空間\n如果您使用Linux,請確保已安裝gnome-keyring、kde-wallet和keepassxc等加密服務'; + + @override + String get querying_info => '正在查詢資訊...'; + + @override + String get piped_api_down => 'Piped API 無法使用'; + + @override + String piped_down_error_instructions(Object pipedInstance) { + return '當前Piped實例 $pipedInstance 不可用\n\n請更改實例或將\'API類型\'更改為官方YouTube API\n\n更改後請確保重新啟動應用程式'; + } + + @override + String get you_are_offline => '您目前處於離線狀態'; + + @override + String get connection_restored => '您的網路連線已恢復'; + + @override + String get use_system_title_bar => '使用作業系統的預設視窗標題列'; + + @override + String get crunching_results => '處理結果中...'; + + @override + String get search_to_get_results => '搜尋以取得結果'; + + @override + String get use_amoled_mode => '使用 AMOLED 模式'; + + @override + String get pitch_dark_theme => '漆黑主題'; + + @override + String get normalize_audio => '標準化音訊'; + + @override + String get change_cover => '更改封面'; + + @override + String get add_cover => '新增封面'; + + @override + String get restore_defaults => '恢復預設值'; + + @override + String get download_music_format => '下載音樂格式'; + + @override + String get streaming_music_format => '串流音樂格式'; + + @override + String get download_music_quality => '下載音樂品質'; + + @override + String get streaming_music_quality => '串流音樂品質'; + + @override + String get login_with_lastfm => '使用 Last.fm 登入'; + + @override + String get connect => '連線'; + + @override + String get disconnect_lastfm => '切斷 Last.fm 連線'; + + @override + String get disconnect => '斷開連線'; + + @override + String get username => '帳號'; + + @override + String get password => '密碼'; + + @override + String get login => '登入'; + + @override + String get login_with_your_lastfm => '使用您的 Last.fm 帳號登入'; + + @override + String get scrobble_to_lastfm => '在 Last.fm 上記錄你的播放'; + + @override + String get go_to_album => '前往專輯'; + + @override + String get discord_rich_presence => 'Discord Rick Presence(Discord 狀態)'; + + @override + String get browse_all => '瀏覽全部'; + + @override + String get genres => '音樂類型'; + + @override + String get explore_genres => '探索音樂類型'; + + @override + String get friends => '好友'; + + @override + String get no_lyrics_available => '抱歉,無法找到這首歌的歌詞'; + + @override + String get start_a_radio => '開始收聽電台'; + + @override + String get how_to_start_radio => '您想如何開始收聽電台?'; + + @override + String get replace_queue_question => '您想要取代目前清單還是追加到清單?'; + + @override + String get endless_playback => '無限播放'; + + @override + String get delete_playlist => '刪除播放清單'; + + @override + String get delete_playlist_confirmation => '您確定要刪除此播放清單嗎?'; + + @override + String get local_tracks => '本地音訊'; + + @override + String get local_tab => '本地'; + + @override + String get song_link => '歌曲連結'; + + @override + String get skip_this_nonsense => '跳過這個無聊內容'; + + @override + String get freedom_of_music => '“音樂的自由”'; + + @override + String get freedom_of_music_palm => '「音樂的自由掌握在您手中」'; + + @override + String get get_started => '我們開始吧'; + + @override + String get youtube_source_description => '建議且效果最佳。'; + + @override + String get piped_source_description => '感覺自由?與 YouTube 一樣,但更自由。'; + + @override + String get jiosaavn_source_description => '最適合南亞地區。'; + + @override + String get invidious_source_description => '類似 Piped,但可用性更高。'; + + @override + String highest_quality(Object quality) { + return '最高音質:$quality'; + } + + @override + String get select_audio_source => '選擇音訊來源'; + + @override + String get endless_playback_description => '自動將新歌曲加入清單的結尾'; + + @override + String get choose_your_region => '選擇您的所在地區'; + + @override + String get choose_your_region_description => '這能幫助 Spotube 為您的所在位置顯示正確的內容。'; + + @override + String get choose_your_language => '選擇您的語言'; + + @override + String get help_project_grow => '幫助這個專案成長'; + + @override + String get help_project_grow_description => + 'Spotube是一個開源專案。您可以透過為專案做出貢獻、回報錯誤或建議新功能來幫助專案成長。'; + + @override + String get contribute_on_github => '在GitHub上做出貢獻'; + + @override + String get donate_on_open_collective => '在Open Collective上捐款'; + + @override + String get browse_anonymously => '匿名瀏覽'; + + @override + String get enable_connect => '啟用連線'; + + @override + String get enable_connect_description => '從其他裝置控制Spotube'; + + @override + String get devices => '裝置'; + + @override + String get select => '選擇'; + + @override + String connect_client_alert(Object client) { + return '您正在被 $client 控制'; + } + + @override + String get this_device => '此裝置'; + + @override + String get remote => '遠端'; + + @override + String get stats => '統計'; + + @override + String and_n_more(Object count) { + return '還有 $count 個'; + } + + @override + String get recently_played => '最近播放'; + + @override + String get browse_more => '瀏覽更多'; + + @override + String get no_title => '無標題'; + + @override + String get not_playing => '未播放'; + + @override + String get epic_failure => '史詩級的失敗!'; + + @override + String added_num_tracks_to_queue(Object tracks_length) { + return '已將 $tracks_length 首曲目新增至清單'; + } + + @override + String get spotube_has_an_update => 'Spotube 有更新版本'; + + @override + String get download_now => '立即下載'; + + @override + String nightly_version(Object nightlyBuildNum) { + return 'Spotube Nightly $nightlyBuildNum 已發佈'; + } + + @override + String release_version(Object version) { + return 'Spotube v$version 已發布'; + } + + @override + String get read_the_latest => '閱讀最新'; + + @override + String get release_notes => '版本說明'; + + @override + String get pick_color_scheme => '選擇配色方案'; + + @override + String get save => '儲存'; + + @override + String get choose_the_device => '選擇裝置:'; + + @override + String get multiple_device_connected => '已連接多個裝置。\n選擇您希望執行此操作的裝置'; + + @override + String get nothing_found => '未找到任何內容'; + + @override + String get the_box_is_empty => '箱子為空'; + + @override + String get top_artists => '熱門藝人'; + + @override + String get top_albums => '熱門專輯'; + + @override + String get this_week => '本週'; + + @override + String get this_month => '本月'; + + @override + String get last_6_months => '過去6個月'; + + @override + String get this_year => '今年'; + + @override + String get last_2_years => '過去2年'; + + @override + String get all_time => '所有時間'; + + @override + String powered_by_provider(Object providerName) { + return '由 $providerName 提供支援'; + } + + @override + String get email => '電子郵件'; + + @override + String get profile_followers => '追蹤者'; + + @override + String get birthday => '生日'; + + @override + String get subscription => '訂閱'; + + @override + String get not_born => '尚未建立'; + + @override + String get hacker => '駭客'; + + @override + String get profile => '個人資訊'; + + @override + String get no_name => '沒有名字'; + + @override + String get edit => '編輯'; + + @override + String get user_profile => '使用者資料'; + + @override + String count_plays(Object count) { + return '$count 次播放'; + } + + @override + String get streaming_fees_hypothetical => + '*基於 Spotify 每次播放的支付金額\n從 \$0.003 到 \$0.005 計算。這是一個假設性的\n計算,旨在讓用戶了解如果他們在 Spotify 上收聽\n這些歌曲,可能會付給作者的金額。'; + + @override + String get minutes_listened => '聽的分鐘數'; + + @override + String get streamed_songs => '已串流歌曲'; + + @override + String count_streams(Object count) { + return '$count 次串流'; + } + + @override + String get owned_by_you => '由您所有'; + + @override + String copied_shareurl_to_clipboard(Object shareUrl) { + return '$shareUrl 已複製到剪貼簿'; + } + + @override + String get hipotetical_calculation => + '*此為根據線上音樂串流平台平均每次播放 \$0.003 至 \$0.005 的收益所計算的假設值。此為一個假設性計算,旨在讓使用者了解若他們在不同的音樂串流平台上收聽同一首歌曲,他們將會支付給藝人多少費用。'; + + @override + String count_mins(Object minutes) { + return '$minutes 分鐘'; + } + + @override + String get summary_minutes => '分鐘'; + + @override + String get summary_listened_to_music => '聽音樂'; + + @override + String get summary_songs => '歌曲'; + + @override + String get summary_streamed_overall => '整體串流媒體'; + + @override + String get summary_owed_to_artists => '本月欠藝術家的'; + + @override + String get summary_artists => '藝術家的'; + + @override + String get summary_music_reached_you => '音樂接觸到你'; + + @override + String get summary_full_albums => '完整專輯'; + + @override + String get summary_got_your_love => '獲得了你的愛心'; + + @override + String get summary_playlists => '播放清單'; + + @override + String get summary_were_on_repeat => '已經重複播放'; + + @override + String total_money(Object money) { + return '總計 $money'; + } + + @override + String get webview_not_found => '未找到 Webview 框架'; + + @override + String get webview_not_found_description => + '您的裝置中未安裝 Webview Runtime。\n如果已安裝,請確保它的位置在系統環境變數(PATH)中\n\n安裝後,重新啟動應用程式'; + + @override + String get unsupported_platform => '不支援的平台'; + + @override + String get cache_music => '快取音樂'; + + @override + String get open => '開啟'; + + @override + String get cache_folder => '快取資料夾'; + + @override + String get export => '導出'; + + @override + String get clear_cache => '清除快取'; + + @override + String get clear_cache_confirmation => '您要清除快取嗎?'; + + @override + String get export_cache_files => '匯出快取檔案'; + + @override + String found_n_files(Object count) { + return '找到 $count 個檔案'; + } + + @override + String get export_cache_confirmation => '您要匯出這些檔案到'; + + @override + String exported_n_out_of_m_files(Object files, Object filesExported) { + return '匯出了 $filesExported / $files 個檔案'; + } + + @override + String get undo => '取消'; + + @override + String get download_all => '下載全部'; + + @override + String get add_all_to_playlist => '全部加入到播放清單'; + + @override + String get add_all_to_queue => '全部加入清單'; + + @override + String get play_all_next => '播放全部下一首'; + + @override + String get pause => '暫停'; + + @override + String get view_all => '檢視全部'; + + @override + String get no_tracks_added_yet => '看起來你還沒有加入任何歌曲'; + + @override + String get no_tracks => '看起來這裡沒有任何歌曲'; + + @override + String get no_tracks_listened_yet => '看起來你還沒聽任何歌曲'; + + @override + String get not_following_artists => '你沒有關注任何藝術家'; + + @override + String get no_favorite_albums_yet => '看起來你還沒有將任何專輯加入到收藏夾'; + + @override + String get no_logs_found => '未找到日誌'; + + @override + String get youtube_engine => 'YouTube 引擎'; + + @override + String youtube_engine_not_installed_title(Object engine) { + return '$engine 未安裝'; + } + + @override + String youtube_engine_not_installed_message(Object engine) { + return '$engine 未在您的系統中安裝。'; + } + + @override + String youtube_engine_set_path(Object engine) { + return '確保它可用在 PATH 變數中,或\n設定 $engine 執行檔的絕對路徑'; + } + + @override + String get youtube_engine_unix_issue_message => + '在類 Unix 作業系統(如 macOS/Linux/Unix)中,請在 .zshrc/.bashrc/.bash_profile 等檔案中設定路徑無效。\n您需要在 shell 設定檔中設定路徑'; + + @override + String get download => '下載'; + + @override + String get file_not_found => '找不到檔案'; + + @override + String get custom => '自訂'; + + @override + String get add_custom_url => '新增自訂 URL'; + + @override + String get edit_port => '編輯端口'; + + @override + String get port_helper_msg => '預設值為 -1,表示隨機數。如果您已配置防火牆,建議設定此項目。'; + + @override + String connect_request(Object client) { + return '允許 $client 連線嗎?'; + } + + @override + String get connection_request_denied => '連線被拒絕。請求被使用者拒絕。'; + + @override + String get an_error_occurred => '發生錯誤'; + + @override + String get copy_to_clipboard => '複製到剪貼簿'; + + @override + String get view_logs => '檢視日誌'; + + @override + String get retry => '重試'; + + @override + String get no_default_metadata_provider_selected => '您沒有設定預設的中繼資料供應商'; + + @override + String get manage_metadata_providers => '管理中繼資料供應商'; + + @override + String get open_link_in_browser => '要在瀏覽器中開啟連結嗎?'; + + @override + String get do_you_want_to_open_the_following_link => '您想開啟以下連結嗎'; + + @override + String get unsafe_url_warning => '從不受信任的來源開啟連結可能不安全。請務必小心!\n您也可以將連結複製到剪貼簿。'; + + @override + String get copy_link => '複製連結'; + + @override + String get building_your_timeline => '正在根據您的收聽記錄建立您的時間軸...'; + + @override + String get official => '官方'; + + @override + String author_name(Object author) { + return '作者:$author'; + } + + @override + String get third_party => '第三方'; + + @override + String get plugin_requires_authentication => '此外掛程式需要驗證'; + + @override + String get update_available => '有可用的更新'; + + @override + String get supports_scrobbling => '支援 Scrobbling'; + + @override + String get plugin_scrobbling_info => '此外掛程式會 Scrobble 您的音樂以產生您的收聽記錄。'; + + @override + String get default_metadata_source => '預設中繼資料來源'; + + @override + String get set_default_metadata_source => '設定預設中繼資料來源'; + + @override + String get default_audio_source => '預設音訊來源'; + + @override + String get set_default_audio_source => '設定預設音訊來源'; + + @override + String get set_default => '設為預設'; + + @override + String get support => '支援'; + + @override + String get support_plugin_development => '支援外掛程式開發'; + + @override + String can_access_name_api(Object name) { + return '- 可以存取 **$name** API'; + } + + @override + String get do_you_want_to_install_this_plugin => '您想安裝此外掛程式嗎?'; + + @override + String get third_party_plugin_warning => '此外掛程式來自第三方儲存庫。請在安裝前確認您信任該來源。'; + + @override + String get author => '作者'; + + @override + String get this_plugin_can_do_following => '此外掛程式可以執行以下操作'; + + @override + String get install => '安裝'; + + @override + String get install_a_metadata_provider => '安裝中繼資料供應商'; + + @override + String get no_tracks_playing => '目前沒有正在播放的曲目'; + + @override + String get synced_lyrics_not_available => '此歌曲沒有同步歌詞。請改用'; + + @override + String get plain_lyrics => '純歌詞'; + + @override + String get tab_instead => '分頁。'; + + @override + String get disclaimer => '免責聲明'; + + @override + String get third_party_plugin_dmca_notice => + 'Spotube 團隊對任何「第三方」外掛程式不負任何責任(包括法律責任)。\n請自行承擔使用風險。如有任何錯誤/問題,請向該外掛程式的儲存庫回報。\n\n若有任何「第三方」外掛程式違反任何服務/法律實體的服務條款/DMCA,請向「第三方」外掛程式作者或託管平台(如 GitHub/Codeberg)要求採取行動。以上列出的(標記為「第三方」)外掛程式均為公開/社群維護的外掛程式。我們沒有對其進行審核,因此無法對其採取任何行動。\n\n'; + + @override + String get input_does_not_match_format => '輸入不符合所需格式'; + + @override + String get plugins => '外掛程式'; + + @override + String get paste_plugin_download_url => + '貼上下載網址、GitHub/Codeberg 儲存庫網址或 .smplug 檔案的直接連結'; + + @override + String get download_and_install_plugin_from_url => '從網址下載並安裝外掛程式'; + + @override + String failed_to_add_plugin_error(Object error) { + return '新增外掛程式失敗:$error'; + } + + @override + String get upload_plugin_from_file => '從檔案上傳外掛程式'; + + @override + String get installed => '已安裝'; + + @override + String get available_plugins => '可用的外掛程式'; + + @override + String get configure_plugins => '配置您自己的中繼資料提供者和音訊來源外掛程式'; + + @override + String get audio_scrobblers => '音訊 Scrobblers'; + + @override + String get scrobbling => 'Scrobbling'; + + @override + String get source => '來源:'; + + @override + String get uncompressed => '未壓縮'; + + @override + String get dab_music_source_description => + '適合音響發燒友。提供高品質/無損音訊串流。精確的 ISRC 曲目比對。'; +} diff --git a/lib/l10n/l10n.dart b/lib/l10n/l10n.dart index ebdc4b61..d0aeccc4 100644 --- a/lib/l10n/l10n.dart +++ b/lib/l10n/l10n.dart @@ -12,11 +12,14 @@ /// doannc2212@github => Vietnamese /// sappho192@github => Korean /// watchakorn-18k@github => Thai +/// llama3, vishnumur777@github => Tamil /// Microsoft Copilot, Tutislav@github => Czech +/// 510208@github => Traditional Chinese library l10n; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +export 'package:spotube/l10n/generated/app_localizations.dart'; class L10n { static final all = [ @@ -41,10 +44,13 @@ class L10n { const Locale('pl', 'PL'), const Locale('pt', 'PT'), const Locale('ru', 'RU'), + const Locale('tl', 'PH'), const Locale('uk', 'UA'), const Locale('th', 'TH'), + const Locale('ta', 'IN'), const Locale('tr', 'TR'), const Locale('zh', 'CN'), + const Locale('zh', 'TW'), const Locale('vi', 'VN'), const Locale('eu', 'ES'), ]; diff --git a/lib/main.dart b/lib/main.dart index f13991e2..ecf7148d 100644 --- a/lib/main.dart +++ b/lib/main.dart @@ -1,23 +1,25 @@ import 'dart:async'; import 'dart:ui'; +import 'dart:io'; import 'package:desktop_webview_window/desktop_webview_window.dart'; import 'package:flutter/foundation.dart'; -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' as material; import 'package:flutter/services.dart'; import 'package:flutter_discord_rpc/flutter_discord_rpc.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:flutter_localizations/flutter_localizations.dart'; -import 'package:hive/hive.dart'; + +import 'package:home_widget/home_widget.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:local_notifier/local_notifier.dart'; import 'package:media_kit/media_kit.dart'; import 'package:metadata_god/metadata_god.dart'; import 'package:smtc_windows/smtc_windows.dart'; import 'package:spotube/collections/env.dart'; -import 'package:spotube/collections/initializers.dart'; -import 'package:spotube/collections/routes.dart'; +import 'package:spotube/collections/http-override.dart'; import 'package:spotube/collections/intents.dart'; +import 'package:spotube/collections/routes.dart'; import 'package:spotube/hooks/configurators/use_close_behavior.dart'; import 'package:spotube/hooks/configurators/use_deep_linking.dart'; import 'package:spotube/hooks/configurators/use_disable_battery_optimizations.dart'; @@ -25,14 +27,17 @@ import 'package:spotube/hooks/configurators/use_fix_window_stretching.dart'; import 'package:spotube/hooks/configurators/use_get_storage_perms.dart'; import 'package:spotube/hooks/configurators/use_has_touch.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/modules/settings/color_scheme_picker_dialog.dart'; import 'package:spotube/provider/audio_player/audio_player_streams.dart'; import 'package:spotube/provider/database/database.dart'; +import 'package:spotube/provider/glance/glance.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/updater/update_checker.dart'; import 'package:spotube/provider/server/bonsoir.dart'; import 'package:spotube/provider/server/server.dart'; import 'package:spotube/provider/tray_manager/tray_manager.dart'; import 'package:spotube/l10n/l10n.dart'; import 'package:spotube/provider/connect/clients.dart'; -import 'package:spotube/provider/palette_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/cli/cli.dart'; @@ -40,17 +45,15 @@ import 'package:spotube/services/kv_store/encrypted_kv_store.dart'; import 'package:spotube/services/kv_store/kv_store.dart'; import 'package:spotube/services/logger/logger.dart'; import 'package:spotube/services/wm_tools/wm_tools.dart'; -import 'package:spotube/themes/theme.dart'; -import 'package:spotube/utils/migrations/hive.dart'; import 'package:spotube/utils/migrations/sandbox.dart'; import 'package:spotube/utils/platform.dart'; -import 'package:system_theme/system_theme.dart'; -import 'package:path_provider/path_provider.dart'; -import 'package:flutter_gen/gen_l10n/app_localizations.dart'; import 'package:flutter_native_splash/flutter_native_splash.dart'; import 'package:flutter_displaymode/flutter_displaymode.dart'; import 'package:timezone/data/latest.dart' as tz; import 'package:window_manager/window_manager.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:yt_dlp_dart/yt_dlp_dart.dart'; +import 'package:flutter_new_pipe_extractor/flutter_new_pipe_extractor.dart'; Future main(List rawArgs) async { if (rawArgs.contains("web_view_title_bar")) { @@ -65,7 +68,9 @@ Future main(List rawArgs) async { AppLogger.runZoned(() async { final widgetsBinding = WidgetsFlutterBinding.ensureInitialized(); - await registerWindowsScheme("spotify"); + HttpOverrides.global = BadCertificateAllowlistOverrides(); + + // await registerWindowsScheme("spotify"); tz.initializeTimeZones(); @@ -79,18 +84,24 @@ Future main(List rawArgs) async { if (kIsAndroid) { await FlutterDisplayMode.setHighRefreshRate(); } - - if (kIsDesktop) { - await windowManager.setPreventClose(true); + if (kIsAndroid || kIsDesktop) { + await NewPipeExtractor.init(); } - await SystemTheme.accentColor.load(); - if (!kIsWeb) { MetadataGod.initialize(); } + await KVStoreService.initialize(); + if (kIsDesktop) { + await windowManager.setPreventClose(true); + await YtDlp.instance + .setBinaryLocation( + KVStoreService.getYoutubeEnginePath(YoutubeClientEngine.ytDlp) ?? + "yt-dlp${kIsWindows ? '.exe' : ''}", + ) + .catchError((e, stack) => null); await FlutterDiscordRPC.initialize(Env.discordAppId); } @@ -98,23 +109,19 @@ Future main(List rawArgs) async { await SMTCWindows.initialize(); } - await KVStoreService.initialize(); await EncryptedKvStoreService.initialize(); - final hiveCacheDir = - kIsWeb ? null : (await getApplicationSupportDirectory()).path; - - Hive.init(hiveCacheDir); - final database = AppDatabase(); - await migrateFromHiveToDrift(database); - if (kIsDesktop) { await localNotifier.setup(appName: "Spotube"); await WindowManagerTools.initialize(); } + if (kIsIOS) { + HomeWidget.setAppGroupId("group.spotube_home_player_widget"); + } + runApp( ProviderScope( overrides: [ @@ -136,14 +143,10 @@ class Spotube extends HookConsumerWidget { Widget build(BuildContext context, ref) { final themeMode = ref.watch(userPreferencesProvider.select((s) => s.themeMode)); + final locale = ref.watch(userPreferencesProvider.select((s) => s.locale)); final accentMaterialColor = ref.watch(userPreferencesProvider.select((s) => s.accentColorScheme)); - final isAmoledTheme = - ref.watch(userPreferencesProvider.select((s) => s.amoledDarkTheme)); - final locale = ref.watch(userPreferencesProvider.select((s) => s.locale)); - final paletteColor = - ref.watch(paletteProvider.select((s) => s?.dominantColor?.color)); - final router = ref.watch(routerProvider); + final router = useMemoized(() => AppRouter(ref), []); final hasTouchSupport = useHasTouch(); ref.listen(audioPlayerStreamListenersProvider, (_, __) {}); @@ -151,16 +154,25 @@ class Spotube extends HookConsumerWidget { ref.listen(connectClientsProvider, (_, __) {}); ref.listen(serverProvider, (_, __) {}); ref.listen(trayManagerProvider, (_, __) {}); + ref.listen(metadataPluginsProvider, (_, __) {}); + ref.listen(metadataPluginProvider, (_, __) {}); + ref.listen(audioSourcePluginProvider, (_, __) {}); + ref.listen(metadataPluginUpdateCheckerProvider, (_, __) {}); + ref.listen(audioSourcePluginUpdateCheckerProvider, (_, __) {}); useFixWindowStretching(); useDisableBatteryOptimizations(); - useDeepLinking(ref); + useDeepLinking(ref, router); useCloseBehavior(ref); useGetStoragePermissions(ref); useEffect(() { FlutterNativeSplash.remove(); + if (kIsMobile) { + HomeWidget.registerInteractivityCallback(glanceBackgroundCallback); + } + return () { /// For enabling hot reload for audio player if (!kDebugMode) return; @@ -168,20 +180,7 @@ class Spotube extends HookConsumerWidget { }; }, []); - final lightTheme = useMemoized( - () => theme(paletteColor ?? accentMaterialColor, Brightness.light, false), - [paletteColor, accentMaterialColor], - ); - final darkTheme = useMemoized( - () => theme( - paletteColor ?? accentMaterialColor, - Brightness.dark, - isAmoledTheme, - ), - [paletteColor, accentMaterialColor, isAmoledTheme], - ); - - return MaterialApp.router( + return ShadcnApp.router( supportedLocales: L10n.all, locale: locale.languageCode == "system" ? null : locale, localizationsDelegates: const [ @@ -190,7 +189,7 @@ class Spotube extends HookConsumerWidget { GlobalWidgetsLocalizations.delegate, GlobalCupertinoLocalizations.delegate, ], - routerConfig: router, + routerConfig: router.config(), debugShowCheckedModeBanner: false, title: 'Spotube', builder: (context, child) { @@ -207,13 +206,49 @@ class Spotube extends HookConsumerWidget { child: child!, ); - if (kIsDesktop && !kIsMacOS) child = DragToResizeArea(child: child); + if (kIsLinux) { + child = DragToResizeArea( + resizeEdgeSize: 2.5, + child: child, + ); + } return child; }, + scaling: const AdaptiveScaling(1), + theme: ThemeData( + radius: .5, + iconTheme: const IconThemeProperties(), + colorScheme: + colorSchemeMap[accentMaterialColor.name]?.call(ThemeMode.light) ?? + LegacyColorSchemes.lightSlate(), + surfaceOpacity: .8, + surfaceBlur: 10, + ), + darkTheme: ThemeData( + radius: .5, + iconTheme: const IconThemeProperties(), + colorScheme: + colorSchemeMap[accentMaterialColor.name]?.call(ThemeMode.dark) ?? + LegacyColorSchemes.darkSlate(), + surfaceOpacity: .8, + surfaceBlur: 10, + ), + materialTheme: material.ThemeData( + brightness: switch (themeMode) { + ThemeMode.system => MediaQuery.platformBrightnessOf(context), + ThemeMode.light => Brightness.light, + ThemeMode.dark => Brightness.dark, + }, + splashFactory: material.NoSplash.splashFactory, + appBarTheme: const material.AppBarTheme( + surfaceTintColor: Colors.transparent, + scrolledUnderElevation: 0, + shadowColor: Colors.transparent, + elevation: 0, + ), + ), themeMode: themeMode, - theme: lightTheme, - darkTheme: darkTheme, shortcuts: { ...WidgetsApp.defaultShortcuts.map((key, value) { return MapEntry( @@ -228,22 +263,42 @@ class Spotube extends HookConsumerWidget { LogicalKeyboardKey.digit1, LogicalKeyboardKey.control, LogicalKeyboardKey.shift, - ): HomeTabIntent(ref, tab: HomeTabs.browse), + ): HomeTabIntent(router, tab: HomeTabs.browse), LogicalKeySet( LogicalKeyboardKey.digit2, LogicalKeyboardKey.control, LogicalKeyboardKey.shift, - ): HomeTabIntent(ref, tab: HomeTabs.search), + ): HomeTabIntent(router, tab: HomeTabs.search), LogicalKeySet( LogicalKeyboardKey.digit3, LogicalKeyboardKey.control, LogicalKeyboardKey.shift, - ): HomeTabIntent(ref, tab: HomeTabs.library), + ): HomeTabIntent(router, tab: HomeTabs.lyrics), LogicalKeySet( LogicalKeyboardKey.digit4, LogicalKeyboardKey.control, LogicalKeyboardKey.shift, - ): HomeTabIntent(ref, tab: HomeTabs.lyrics), + ): HomeTabIntent(router, tab: HomeTabs.userPlaylists), + LogicalKeySet( + LogicalKeyboardKey.digit5, + LogicalKeyboardKey.control, + LogicalKeyboardKey.shift, + ): HomeTabIntent(router, tab: HomeTabs.userArtists), + LogicalKeySet( + LogicalKeyboardKey.digit6, + LogicalKeyboardKey.control, + LogicalKeyboardKey.shift, + ): HomeTabIntent(router, tab: HomeTabs.userAlbums), + LogicalKeySet( + LogicalKeyboardKey.digit7, + LogicalKeyboardKey.control, + LogicalKeyboardKey.shift, + ): HomeTabIntent(router, tab: HomeTabs.userLocalLibrary), + LogicalKeySet( + LogicalKeyboardKey.digit8, + LogicalKeyboardKey.control, + LogicalKeyboardKey.shift, + ): HomeTabIntent(router, tab: HomeTabs.userDownloads), LogicalKeySet( LogicalKeyboardKey.keyW, LogicalKeyboardKey.control, diff --git a/lib/models/connect/connect.dart b/lib/models/connect/connect.dart index a70520ad..11370dcb 100644 --- a/lib/models/connect/connect.dart +++ b/lib/models/connect/connect.dart @@ -5,7 +5,7 @@ import 'dart:convert'; import 'package:freezed_annotation/freezed_annotation.dart'; import 'package:media_kit/media_kit.dart' hide Track; -import 'package:spotify/spotify.dart' hide Playlist; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/state.dart'; part 'connect.freezed.dart'; diff --git a/lib/models/connect/connect.freezed.dart b/lib/models/connect/connect.freezed.dart index 9103dd2b..157d0911 100644 --- a/lib/models/connect/connect.freezed.dart +++ b/lib/models/connect/connect.freezed.dart @@ -33,20 +33,25 @@ WebSocketLoadEventData _$WebSocketLoadEventDataFromJson( /// @nodoc mixin _$WebSocketLoadEventData { - @JsonKey(name: 'tracks', toJson: _tracksJson) - List get tracks => throw _privateConstructorUsedError; + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List get tracks => throw _privateConstructorUsedError; Object? get collection => throw _privateConstructorUsedError; int? get initialIndex => throw _privateConstructorUsedError; @optionalTypeArgs TResult when({ required TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex) playlist, required TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex) album, }) => @@ -54,13 +59,17 @@ mixin _$WebSocketLoadEventData { @optionalTypeArgs TResult? whenOrNull({ TResult? Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex)? playlist, TResult? Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex)? album, }) => @@ -68,13 +77,17 @@ mixin _$WebSocketLoadEventData { @optionalTypeArgs TResult maybeWhen({ TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex)? playlist, TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex)? album, required TResult orElse(), @@ -117,7 +130,9 @@ abstract class $WebSocketLoadEventDataCopyWith<$Res> { _$WebSocketLoadEventDataCopyWithImpl<$Res, WebSocketLoadEventData>; @useResult $Res call( - {@JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, int? initialIndex}); } @@ -144,7 +159,7 @@ class _$WebSocketLoadEventDataCopyWithImpl<$Res, tracks: null == tracks ? _value.tracks : tracks // ignore: cast_nullable_to_non_nullable - as List, + as List, initialIndex: freezed == initialIndex ? _value.initialIndex : initialIndex // ignore: cast_nullable_to_non_nullable @@ -163,9 +178,13 @@ abstract class _$$WebSocketLoadEventDataPlaylistImplCopyWith<$Res> @override @useResult $Res call( - {@JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex}); + + $SpotubeSimplePlaylistObjectCopyWith<$Res>? get collection; } /// @nodoc @@ -191,17 +210,32 @@ class __$$WebSocketLoadEventDataPlaylistImplCopyWithImpl<$Res> tracks: null == tracks ? _value._tracks : tracks // ignore: cast_nullable_to_non_nullable - as List, + as List, collection: freezed == collection ? _value.collection : collection // ignore: cast_nullable_to_non_nullable - as PlaylistSimple?, + as SpotubeSimplePlaylistObject?, initialIndex: freezed == initialIndex ? _value.initialIndex : initialIndex // ignore: cast_nullable_to_non_nullable as int?, )); } + + /// Create a copy of WebSocketLoadEventData + /// with the given fields replaced by the non-null parameter values. + @override + @pragma('vm:prefer-inline') + $SpotubeSimplePlaylistObjectCopyWith<$Res>? get collection { + if (_value.collection == null) { + return null; + } + + return $SpotubeSimplePlaylistObjectCopyWith<$Res>(_value.collection!, + (value) { + return _then(_value.copyWith(collection: value)); + }); + } } /// @nodoc @@ -209,8 +243,9 @@ class __$$WebSocketLoadEventDataPlaylistImplCopyWithImpl<$Res> class _$WebSocketLoadEventDataPlaylistImpl extends WebSocketLoadEventDataPlaylist { _$WebSocketLoadEventDataPlaylistImpl( - {@JsonKey(name: 'tracks', toJson: _tracksJson) - required final List tracks, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + required final List tracks, this.collection, this.initialIndex, final String? $type}) @@ -222,17 +257,18 @@ class _$WebSocketLoadEventDataPlaylistImpl Map json) => _$$WebSocketLoadEventDataPlaylistImplFromJson(json); - final List _tracks; + final List _tracks; @override - @JsonKey(name: 'tracks', toJson: _tracksJson) - List get tracks { + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List get tracks { if (_tracks is EqualUnmodifiableListView) return _tracks; // ignore: implicit_dynamic_type return EqualUnmodifiableListView(_tracks); } @override - final PlaylistSimple? collection; + final SpotubeSimplePlaylistObject? collection; @override final int? initialIndex; @@ -275,13 +311,17 @@ class _$WebSocketLoadEventDataPlaylistImpl @optionalTypeArgs TResult when({ required TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex) playlist, required TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex) album, }) { @@ -292,13 +332,17 @@ class _$WebSocketLoadEventDataPlaylistImpl @optionalTypeArgs TResult? whenOrNull({ TResult? Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex)? playlist, TResult? Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex)? album, }) { @@ -309,13 +353,17 @@ class _$WebSocketLoadEventDataPlaylistImpl @optionalTypeArgs TResult maybeWhen({ TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex)? playlist, TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex)? album, required TResult orElse(), @@ -367,9 +415,10 @@ class _$WebSocketLoadEventDataPlaylistImpl abstract class WebSocketLoadEventDataPlaylist extends WebSocketLoadEventData { factory WebSocketLoadEventDataPlaylist( - {@JsonKey(name: 'tracks', toJson: _tracksJson) - required final List tracks, - final PlaylistSimple? collection, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + required final List tracks, + final SpotubeSimplePlaylistObject? collection, final int? initialIndex}) = _$WebSocketLoadEventDataPlaylistImpl; WebSocketLoadEventDataPlaylist._() : super._(); @@ -377,10 +426,11 @@ abstract class WebSocketLoadEventDataPlaylist extends WebSocketLoadEventData { _$WebSocketLoadEventDataPlaylistImpl.fromJson; @override - @JsonKey(name: 'tracks', toJson: _tracksJson) - List get tracks; + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List get tracks; @override - PlaylistSimple? get collection; + SpotubeSimplePlaylistObject? get collection; @override int? get initialIndex; @@ -403,9 +453,13 @@ abstract class _$$WebSocketLoadEventDataAlbumImplCopyWith<$Res> @override @useResult $Res call( - {@JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex}); + + $SpotubeSimpleAlbumObjectCopyWith<$Res>? get collection; } /// @nodoc @@ -431,25 +485,40 @@ class __$$WebSocketLoadEventDataAlbumImplCopyWithImpl<$Res> tracks: null == tracks ? _value._tracks : tracks // ignore: cast_nullable_to_non_nullable - as List, + as List, collection: freezed == collection ? _value.collection : collection // ignore: cast_nullable_to_non_nullable - as AlbumSimple?, + as SpotubeSimpleAlbumObject?, initialIndex: freezed == initialIndex ? _value.initialIndex : initialIndex // ignore: cast_nullable_to_non_nullable as int?, )); } + + /// Create a copy of WebSocketLoadEventData + /// with the given fields replaced by the non-null parameter values. + @override + @pragma('vm:prefer-inline') + $SpotubeSimpleAlbumObjectCopyWith<$Res>? get collection { + if (_value.collection == null) { + return null; + } + + return $SpotubeSimpleAlbumObjectCopyWith<$Res>(_value.collection!, (value) { + return _then(_value.copyWith(collection: value)); + }); + } } /// @nodoc @JsonSerializable() class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum { _$WebSocketLoadEventDataAlbumImpl( - {@JsonKey(name: 'tracks', toJson: _tracksJson) - required final List tracks, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + required final List tracks, this.collection, this.initialIndex, final String? $type}) @@ -461,17 +530,18 @@ class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum { Map json) => _$$WebSocketLoadEventDataAlbumImplFromJson(json); - final List _tracks; + final List _tracks; @override - @JsonKey(name: 'tracks', toJson: _tracksJson) - List get tracks { + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List get tracks { if (_tracks is EqualUnmodifiableListView) return _tracks; // ignore: implicit_dynamic_type return EqualUnmodifiableListView(_tracks); } @override - final AlbumSimple? collection; + final SpotubeSimpleAlbumObject? collection; @override final int? initialIndex; @@ -513,13 +583,17 @@ class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum { @optionalTypeArgs TResult when({ required TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex) playlist, required TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex) album, }) { @@ -530,13 +604,17 @@ class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum { @optionalTypeArgs TResult? whenOrNull({ TResult? Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex)? playlist, TResult? Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex)? album, }) { @@ -547,13 +625,17 @@ class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum { @optionalTypeArgs TResult maybeWhen({ TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - PlaylistSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex)? playlist, TResult Function( - @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks, - AlbumSimple? collection, + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex)? album, required TResult orElse(), @@ -605,9 +687,10 @@ class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum { abstract class WebSocketLoadEventDataAlbum extends WebSocketLoadEventData { factory WebSocketLoadEventDataAlbum( - {@JsonKey(name: 'tracks', toJson: _tracksJson) - required final List tracks, - final AlbumSimple? collection, + {@Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + required final List tracks, + final SpotubeSimpleAlbumObject? collection, final int? initialIndex}) = _$WebSocketLoadEventDataAlbumImpl; WebSocketLoadEventDataAlbum._() : super._(); @@ -615,10 +698,11 @@ abstract class WebSocketLoadEventDataAlbum extends WebSocketLoadEventData { _$WebSocketLoadEventDataAlbumImpl.fromJson; @override - @JsonKey(name: 'tracks', toJson: _tracksJson) - List get tracks; + @Assert("tracks is List", + "tracks must be a list of SpotubeFullTrackObject") + List get tracks; @override - AlbumSimple? get collection; + SpotubeSimpleAlbumObject? get collection; @override int? get initialIndex; diff --git a/lib/models/connect/connect.g.dart b/lib/models/connect/connect.g.dart index 10f46c65..2da8f9b0 100644 --- a/lib/models/connect/connect.g.dart +++ b/lib/models/connect/connect.g.dart @@ -10,11 +10,12 @@ _$WebSocketLoadEventDataPlaylistImpl _$$WebSocketLoadEventDataPlaylistImplFromJson(Map json) => _$WebSocketLoadEventDataPlaylistImpl( tracks: (json['tracks'] as List) - .map((e) => Track.fromJson(Map.from(e as Map))) + .map((e) => SpotubeTrackObject.fromJson( + Map.from(e as Map))) .toList(), collection: json['collection'] == null ? null - : PlaylistSimple.fromJson( + : SpotubeSimplePlaylistObject.fromJson( Map.from(json['collection'] as Map)), initialIndex: (json['initialIndex'] as num?)?.toInt(), $type: json['runtimeType'] as String?, @@ -23,7 +24,7 @@ _$WebSocketLoadEventDataPlaylistImpl Map _$$WebSocketLoadEventDataPlaylistImplToJson( _$WebSocketLoadEventDataPlaylistImpl instance) => { - 'tracks': _tracksJson(instance.tracks), + 'tracks': instance.tracks.map((e) => e.toJson()).toList(), 'collection': instance.collection?.toJson(), 'initialIndex': instance.initialIndex, 'runtimeType': instance.$type, @@ -33,11 +34,12 @@ _$WebSocketLoadEventDataAlbumImpl _$$WebSocketLoadEventDataAlbumImplFromJson( Map json) => _$WebSocketLoadEventDataAlbumImpl( tracks: (json['tracks'] as List) - .map((e) => Track.fromJson(Map.from(e as Map))) + .map((e) => + SpotubeTrackObject.fromJson(Map.from(e as Map))) .toList(), collection: json['collection'] == null ? null - : AlbumSimple.fromJson( + : SpotubeSimpleAlbumObject.fromJson( Map.from(json['collection'] as Map)), initialIndex: (json['initialIndex'] as num?)?.toInt(), $type: json['runtimeType'] as String?, @@ -46,7 +48,7 @@ _$WebSocketLoadEventDataAlbumImpl _$$WebSocketLoadEventDataAlbumImplFromJson( Map _$$WebSocketLoadEventDataAlbumImplToJson( _$WebSocketLoadEventDataAlbumImpl instance) => { - 'tracks': _tracksJson(instance.tracks), + 'tracks': instance.tracks.map((e) => e.toJson()).toList(), 'collection': instance.collection?.toJson(), 'initialIndex': instance.initialIndex, 'runtimeType': instance.$type, diff --git a/lib/models/connect/load.dart b/lib/models/connect/load.dart index bf0e164d..d61e0f1e 100644 --- a/lib/models/connect/load.dart +++ b/lib/models/connect/load.dart @@ -1,22 +1,26 @@ part of 'connect.dart'; -List> _tracksJson(List tracks) { - return tracks.map((e) => e.toJson()).toList(); -} - @freezed class WebSocketLoadEventData with _$WebSocketLoadEventData { const WebSocketLoadEventData._(); factory WebSocketLoadEventData.playlist({ - @JsonKey(name: 'tracks', toJson: _tracksJson) required List tracks, - PlaylistSimple? collection, + @Assert( + "tracks is List", + "tracks must be a list of SpotubeFullTrackObject", + ) + required List tracks, + SpotubeSimplePlaylistObject? collection, int? initialIndex, }) = WebSocketLoadEventDataPlaylist; factory WebSocketLoadEventData.album({ - @JsonKey(name: 'tracks', toJson: _tracksJson) required List tracks, - AlbumSimple? collection, + @Assert( + "tracks is List", + "tracks must be a list of SpotubeFullTrackObject", + ) + required List tracks, + SpotubeSimpleAlbumObject? collection, int? initialIndex, }) = WebSocketLoadEventDataAlbum; diff --git a/lib/models/connect/ws_event.dart b/lib/models/connect/ws_event.dart index d1047646..7867f686 100644 --- a/lib/models/connect/ws_event.dart +++ b/lib/models/connect/ws_event.dart @@ -338,13 +338,16 @@ class WebSocketRemoveTrackEvent extends WebSocketEvent { WebSocketRemoveTrackEvent(String data) : super(WsEvent.removeTrack, data); } -class WebSocketAddTrackEvent extends WebSocketEvent { - WebSocketAddTrackEvent(Track data) : super(WsEvent.addTrack, data); +class WebSocketAddTrackEvent extends WebSocketEvent { + WebSocketAddTrackEvent(SpotubeFullTrackObject data) + : super(WsEvent.addTrack, data); WebSocketAddTrackEvent.fromJson(Map json) : super( WsEvent.addTrack, - Track.fromJson(json["data"] as Map), + SpotubeFullTrackObject.fromJson( + json["data"] as Map, + ), ); } diff --git a/lib/models/current_playlist.dart b/lib/models/current_playlist.dart deleted file mode 100644 index 7e55e393..00000000 --- a/lib/models/current_playlist.dart +++ /dev/null @@ -1,75 +0,0 @@ -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; - -class CurrentPlaylist { - List? _tempTrack; - List tracks; - String id; - String name; - String thumbnail; - bool isLocal; - - CurrentPlaylist({ - required this.tracks, - required this.id, - required this.name, - required this.thumbnail, - this.isLocal = false, - }); - - static CurrentPlaylist fromJson(Map map, Ref ref) { - return CurrentPlaylist( - id: map["id"], - tracks: List.castFrom(map["tracks"] - .map( - (track) => map["isLocal"] == true - ? SourcedTrack.fromJson(track, ref: ref) - : Track.fromJson(track), - ) - .toList()), - name: map["name"], - thumbnail: map["thumbnail"], - isLocal: map["isLocal"], - ); - } - - List get trackIds => tracks.map((e) => e.id!).toList(); - - bool shuffle(Track? topTrack) { - // won't shuffle if already shuffled - if (_tempTrack == null) { - _tempTrack = [...tracks]; - tracks = List.from(tracks)..shuffle(); - if (topTrack != null) { - tracks.remove(topTrack); - tracks.insert(0, topTrack); - } - return true; - } - return false; - } - - bool unshuffle() { - // without _tempTracks unshuffling can't be done - if (_tempTrack != null) { - tracks = [..._tempTrack!]; - _tempTrack = null; - return true; - } - return false; - } - - Map toJson() { - return { - "id": id, - "name": name, - "tracks": tracks - .map((track) => - track is SourcedTrack ? track.toJson() : track.toJson()) - .toList(), - "thumbnail": thumbnail, - "isLocal": isLocal, - }; - } -} diff --git a/lib/models/database/database.dart b/lib/models/database/database.dart index 0f30df19..f1c66c1a 100644 --- a/lib/models/database/database.dart +++ b/lib/models/database/database.dart @@ -4,19 +4,26 @@ import 'dart:convert'; import 'dart:io'; import 'package:drift/drift.dart'; +import 'package:drift/remote.dart'; import 'package:encrypt/encrypt.dart'; import 'package:media_kit/media_kit.dart' hide Track; import 'package:path/path.dart'; import 'package:path_provider/path_provider.dart'; -import 'package:spotify/spotify.dart' hide Playlist; +import 'package:shadcn_flutter/shadcn_flutter.dart' show ThemeMode, Colors; import 'package:spotube/models/database/database.steps.dart'; import 'package:spotube/models/lyrics.dart'; +import 'package:spotube/models/metadata/market.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/services/kv_store/encrypted_kv_store.dart'; import 'package:spotube/services/kv_store/kv_store.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:flutter/material.dart' hide Table, Key, View; +import 'package:flutter/widgets.dart' hide Table, Key, View; import 'package:spotube/modules/settings/color_scheme_picker_dialog.dart'; import 'package:drift/native.dart'; +import 'package:spotube/services/logger/logger.dart'; +import 'package:spotube/services/youtube_engine/newpipe_engine.dart'; +import 'package:spotube/services/youtube_engine/youtube_explode_engine.dart'; +import 'package:spotube/services/youtube_engine/yt_dlp_engine.dart'; +import 'package:spotube/utils/platform.dart'; import 'package:sqlite3/sqlite3.dart'; import 'package:sqlite3_flutter_libs/sqlite3_flutter_libs.dart'; @@ -31,12 +38,14 @@ part 'tables/source_match.dart'; part 'tables/audio_player_state.dart'; part 'tables/history.dart'; part 'tables/lyrics.dart'; +part 'tables/metadata_plugins.dart'; part 'typeconverters/color.dart'; part 'typeconverters/locale.dart'; part 'typeconverters/string_list.dart'; part 'typeconverters/encrypted_text.dart'; part 'typeconverters/map.dart'; +part 'typeconverters/map_list.dart'; part 'typeconverters/subtitle.dart'; @DriftDatabase( @@ -48,17 +57,16 @@ part 'typeconverters/subtitle.dart'; SkipSegmentTable, SourceMatchTable, AudioPlayerStateTable, - PlaylistTable, - PlaylistMediaTable, HistoryTable, LyricsTable, + PluginsTable, ], ) class AppDatabase extends _$AppDatabase { AppDatabase() : super(_openConnection()); @override - int get schemaVersion => 3; + int get schemaVersion => 10; @override MigrationStrategy get migration { @@ -77,6 +85,158 @@ class AppDatabase extends _$AppDatabase { schema.preferencesTable.cacheMusic, ); }, + from3To4: (m, schema) async { + await m.addColumn( + schema.preferencesTable, + schema.preferencesTable.youtubeClientEngine, + ); + }, + from4To5: (m, schema) async { + final columnName = schema.preferencesTable.accentColorScheme + .escapedNameFor(SqlDialect.sqlite); + final columnNameOld = + '"${schema.preferencesTable.accentColorScheme.name}_old"'; + final tableName = schema.preferencesTable.actualTableName; + await customStatement( + "ALTER TABLE $tableName " + "RENAME COLUMN $columnName to $columnNameOld", + ); + await customStatement( + "ALTER TABLE $tableName " + "ADD COLUMN $columnName TEXT NOT NULL DEFAULT 'Slate:0xff64748b'", + ); + await customStatement( + "UPDATE $tableName " + "SET $columnName = $columnNameOld", + ); + await customStatement( + "ALTER TABLE $tableName " + "DROP COLUMN $columnNameOld", + ); + await customStatement( + "UPDATE $tableName " + "SET $columnName = 'Slate:0xff64748b' WHERE $columnName = 'Blue:0xFF2196F3'", + ); + }, + from5To6: (m, schema) async { + try { + await m.addColumn( + schema.preferencesTable, + schema.preferencesTable.connectPort, + ); + } on DriftRemoteException catch (e) { + // If the column already exists, ignore the error + if (e.remoteCause != + 'duplicate column name: ${schema.preferencesTable.connectPort.name}') { + rethrow; + } + } + }, + from6To7: (m, schema) async { + await m.createTable(schema.metadataPluginsTable); + await m.addColumn( + schema.audioPlayerStateTable, + schema.audioPlayerStateTable.currentIndex, + ); + await m.addColumn( + schema.audioPlayerStateTable, + schema.audioPlayerStateTable.tracks, + ); + }, + from7To8: (m, schema) async { + await m + .addColumn( + schema.metadataPluginsTable, + schema.metadataPluginsTable.entryPoint, + ) + .catchError((error, stackTrace) { + // If the column already exists, ignore the error + if (!error.toString().contains('duplicate column name')) { + throw error; + } + }); + await m + .addColumn( + schema.metadataPluginsTable, + schema.metadataPluginsTable.apis, + ) + .catchError((error, stackTrace) { + // If the column already exists, ignore the error + if (!error.toString().contains('duplicate column name')) { + throw error; + } + }); + await m + .addColumn( + schema.metadataPluginsTable, + schema.metadataPluginsTable.abilities, + ) + .catchError((error, stackTrace) { + // If the column already exists, ignore the error + if (!error.toString().contains('duplicate column name')) { + throw error; + } + }); + await m + .addColumn( + schema.metadataPluginsTable, + schema.metadataPluginsTable.repository, + ) + .catchError((error, stackTrace) { + // If the column already exists, ignore the error + if (!error.toString().contains('duplicate column name')) { + throw error; + } + }); + await m + .addColumn( + schema.metadataPluginsTable, + schema.metadataPluginsTable.pluginApiVersion, + ) + .catchError((error, stackTrace) { + // If the column already exists, ignore the error + if (!error.toString().contains('duplicate column name')) { + throw error; + } + }); + }, + from8To9: (m, schema) async { + await m + .renameTable(schema.pluginsTable, "metadata_plugins_table") + .catchError((e, stack) => AppLogger.reportError(e, stack)); + await m + .renameColumn( + schema.pluginsTable, + "selected", + pluginsTable.selectedForMetadata, + ) + .catchError((e, stack) => AppLogger.reportError(e, stack)); + await m + .addColumn( + schema.pluginsTable, + pluginsTable.selectedForAudioSource, + ) + .catchError((e, stack) => AppLogger.reportError(e, stack)); + }, + from9To10: (m, schema) async { + await m + .dropColumn(schema.preferencesTable, "piped_instance") + .catchError((e, stack) => AppLogger.reportError(e, stack)); + await m + .dropColumn(schema.preferencesTable, "invidious_instance") + .catchError((e, stack) => AppLogger.reportError(e, stack)); + await m + .addColumn( + schema.sourceMatchTable, + sourceMatchTable.sourceInfo, + ) + .catchError((e, stack) => AppLogger.reportError(e, stack)); + await customStatement("DROP INDEX IF EXISTS uniq_track_match;") + .catchError((e, stack) => AppLogger.reportError(e, stack)); + await m + .dropColumn(schema.sourceMatchTable, "source_id") + .catchError((e, stack) => AppLogger.reportError(e, stack)); + }, ), ); } diff --git a/lib/models/database/database.g.dart b/lib/models/database/database.g.dart index 951b2ed5..8aa14899 100644 --- a/lib/models/database/database.g.dart +++ b/lib/models/database/database.g.dart @@ -18,15 +18,12 @@ class $AuthenticationTableTable extends AuthenticationTable requiredDuringInsert: false, defaultConstraints: GeneratedColumn.constraintIsAlways('PRIMARY KEY AUTOINCREMENT')); - static const VerificationMeta _cookieMeta = const VerificationMeta('cookie'); @override late final GeneratedColumnWithTypeConverter cookie = GeneratedColumn('cookie', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: true) .withConverter( $AuthenticationTableTable.$convertercookie); - static const VerificationMeta _accessTokenMeta = - const VerificationMeta('accessToken'); @override late final GeneratedColumnWithTypeConverter accessToken = GeneratedColumn('access_token', aliasedName, false, @@ -55,8 +52,6 @@ class $AuthenticationTableTable extends AuthenticationTable if (data.containsKey('id')) { context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); } - context.handle(_cookieMeta, const VerificationResult.success()); - context.handle(_accessTokenMeta, const VerificationResult.success()); if (data.containsKey('expiration')) { context.handle( _expirationMeta, @@ -301,8 +296,6 @@ class $BlacklistTableTable extends BlacklistTable late final GeneratedColumn name = GeneratedColumn( 'name', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: true); - static const VerificationMeta _elementTypeMeta = - const VerificationMeta('elementType'); @override late final GeneratedColumnWithTypeConverter elementType = GeneratedColumn('element_type', aliasedName, false, @@ -336,7 +329,6 @@ class $BlacklistTableTable extends BlacklistTable } else if (isInserting) { context.missing(_nameMeta); } - context.handle(_elementTypeMeta, const VerificationResult.success()); if (data.containsKey('element_id')) { context.handle(_elementIdMeta, elementId.isAcceptableOrUnknown(data['element_id']!, _elementIdMeta)); @@ -566,17 +558,6 @@ class $PreferencesTableTable extends PreferencesTable requiredDuringInsert: false, defaultConstraints: GeneratedColumn.constraintIsAlways('PRIMARY KEY AUTOINCREMENT')); - static const VerificationMeta _audioQualityMeta = - const VerificationMeta('audioQuality'); - @override - late final GeneratedColumnWithTypeConverter - audioQuality = GeneratedColumn( - 'audio_quality', aliasedName, false, - type: DriftSqlType.string, - requiredDuringInsert: false, - defaultValue: Constant(SourceQualities.high.name)) - .withConverter( - $PreferencesTableTable.$converteraudioQuality); static const VerificationMeta _albumColorSyncMeta = const VerificationMeta('albumColorSync'); @override @@ -647,8 +628,6 @@ class $PreferencesTableTable extends PreferencesTable defaultConstraints: GeneratedColumn.constraintIsAlways( 'CHECK ("skip_non_music" IN (0, 1))'), defaultValue: const Constant(false)); - static const VerificationMeta _closeBehaviorMeta = - const VerificationMeta('closeBehavior'); @override late final GeneratedColumnWithTypeConverter closeBehavior = GeneratedColumn( @@ -658,19 +637,15 @@ class $PreferencesTableTable extends PreferencesTable defaultValue: Constant(CloseBehavior.close.name)) .withConverter( $PreferencesTableTable.$convertercloseBehavior); - static const VerificationMeta _accentColorSchemeMeta = - const VerificationMeta('accentColorScheme'); @override late final GeneratedColumnWithTypeConverter accentColorScheme = GeneratedColumn( 'accent_color_scheme', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: false, - defaultValue: const Constant("Blue:0xFF2196F3")) + defaultValue: const Constant("Slate:0xff64748b")) .withConverter( $PreferencesTableTable.$converteraccentColorScheme); - static const VerificationMeta _layoutModeMeta = - const VerificationMeta('layoutMode'); @override late final GeneratedColumnWithTypeConverter layoutMode = GeneratedColumn('layout_mode', aliasedName, false, @@ -679,7 +654,6 @@ class $PreferencesTableTable extends PreferencesTable defaultValue: Constant(LayoutMode.adaptive.name)) .withConverter( $PreferencesTableTable.$converterlayoutMode); - static const VerificationMeta _localeMeta = const VerificationMeta('locale'); @override late final GeneratedColumnWithTypeConverter locale = GeneratedColumn('locale', aliasedName, false, @@ -688,7 +662,6 @@ class $PreferencesTableTable extends PreferencesTable defaultValue: const Constant( '{"languageCode":"system","countryCode":"system"}')) .withConverter($PreferencesTableTable.$converterlocale); - static const VerificationMeta _marketMeta = const VerificationMeta('market'); @override late final GeneratedColumnWithTypeConverter market = GeneratedColumn('market', aliasedName, false, @@ -696,8 +669,6 @@ class $PreferencesTableTable extends PreferencesTable requiredDuringInsert: false, defaultValue: Constant(Market.US.name)) .withConverter($PreferencesTableTable.$convertermarket); - static const VerificationMeta _searchModeMeta = - const VerificationMeta('searchMode'); @override late final GeneratedColumnWithTypeConverter searchMode = GeneratedColumn('search_mode', aliasedName, false, @@ -714,8 +685,6 @@ class $PreferencesTableTable extends PreferencesTable type: DriftSqlType.string, requiredDuringInsert: false, defaultValue: const Constant("")); - static const VerificationMeta _localLibraryLocationMeta = - const VerificationMeta('localLibraryLocation'); @override late final GeneratedColumnWithTypeConverter, String> localLibraryLocation = GeneratedColumn( @@ -725,24 +694,6 @@ class $PreferencesTableTable extends PreferencesTable defaultValue: const Constant("")) .withConverter>( $PreferencesTableTable.$converterlocalLibraryLocation); - static const VerificationMeta _pipedInstanceMeta = - const VerificationMeta('pipedInstance'); - @override - late final GeneratedColumn pipedInstance = GeneratedColumn( - 'piped_instance', aliasedName, false, - type: DriftSqlType.string, - requiredDuringInsert: false, - defaultValue: const Constant("https://pipedapi.kavin.rocks")); - static const VerificationMeta _invidiousInstanceMeta = - const VerificationMeta('invidiousInstance'); - @override - late final GeneratedColumn invidiousInstance = - GeneratedColumn('invidious_instance', aliasedName, false, - type: DriftSqlType.string, - requiredDuringInsert: false, - defaultValue: const Constant("https://inv.nadeko.net")); - static const VerificationMeta _themeModeMeta = - const VerificationMeta('themeMode'); @override late final GeneratedColumnWithTypeConverter themeMode = GeneratedColumn('theme_mode', aliasedName, false, @@ -750,38 +701,21 @@ class $PreferencesTableTable extends PreferencesTable requiredDuringInsert: false, defaultValue: Constant(ThemeMode.system.name)) .withConverter($PreferencesTableTable.$converterthemeMode); - static const VerificationMeta _audioSourceMeta = - const VerificationMeta('audioSource'); + static const VerificationMeta _audioSourceIdMeta = + const VerificationMeta('audioSourceId'); @override - late final GeneratedColumnWithTypeConverter audioSource = - GeneratedColumn('audio_source', aliasedName, false, + late final GeneratedColumn audioSourceId = GeneratedColumn( + 'audio_source_id', aliasedName, true, + type: DriftSqlType.string, requiredDuringInsert: false); + @override + late final GeneratedColumnWithTypeConverter + youtubeClientEngine = GeneratedColumn( + 'youtube_client_engine', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: false, - defaultValue: Constant(AudioSource.youtube.name)) - .withConverter( - $PreferencesTableTable.$converteraudioSource); - static const VerificationMeta _streamMusicCodecMeta = - const VerificationMeta('streamMusicCodec'); - @override - late final GeneratedColumnWithTypeConverter - streamMusicCodec = GeneratedColumn( - 'stream_music_codec', aliasedName, false, - type: DriftSqlType.string, - requiredDuringInsert: false, - defaultValue: Constant(SourceCodecs.weba.name)) - .withConverter( - $PreferencesTableTable.$converterstreamMusicCodec); - static const VerificationMeta _downloadMusicCodecMeta = - const VerificationMeta('downloadMusicCodec'); - @override - late final GeneratedColumnWithTypeConverter - downloadMusicCodec = GeneratedColumn( - 'download_music_codec', aliasedName, false, - type: DriftSqlType.string, - requiredDuringInsert: false, - defaultValue: Constant(SourceCodecs.m4a.name)) - .withConverter( - $PreferencesTableTable.$converterdownloadMusicCodec); + defaultValue: Constant(YoutubeClientEngine.youtubeExplode.name)) + .withConverter( + $PreferencesTableTable.$converteryoutubeClientEngine); static const VerificationMeta _discordPresenceMeta = const VerificationMeta('discordPresence'); @override @@ -812,6 +746,14 @@ class $PreferencesTableTable extends PreferencesTable defaultConstraints: GeneratedColumn.constraintIsAlways( 'CHECK ("enable_connect" IN (0, 1))'), defaultValue: const Constant(false)); + static const VerificationMeta _connectPortMeta = + const VerificationMeta('connectPort'); + @override + late final GeneratedColumn connectPort = GeneratedColumn( + 'connect_port', aliasedName, false, + type: DriftSqlType.int, + requiredDuringInsert: false, + defaultValue: const Constant(-1)); static const VerificationMeta _cacheMusicMeta = const VerificationMeta('cacheMusic'); @override @@ -825,7 +767,6 @@ class $PreferencesTableTable extends PreferencesTable @override List get $columns => [ id, - audioQuality, albumColorSync, amoledDarkTheme, checkUpdate, @@ -841,15 +782,13 @@ class $PreferencesTableTable extends PreferencesTable searchMode, downloadLocation, localLibraryLocation, - pipedInstance, - invidiousInstance, themeMode, - audioSource, - streamMusicCodec, - downloadMusicCodec, + audioSourceId, + youtubeClientEngine, discordPresence, endlessPlayback, enableConnect, + connectPort, cacheMusic ]; @override @@ -866,7 +805,6 @@ class $PreferencesTableTable extends PreferencesTable if (data.containsKey('id')) { context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); } - context.handle(_audioQualityMeta, const VerificationResult.success()); if (data.containsKey('album_color_sync')) { context.handle( _albumColorSyncMeta, @@ -909,36 +847,18 @@ class $PreferencesTableTable extends PreferencesTable skipNonMusic.isAcceptableOrUnknown( data['skip_non_music']!, _skipNonMusicMeta)); } - context.handle(_closeBehaviorMeta, const VerificationResult.success()); - context.handle(_accentColorSchemeMeta, const VerificationResult.success()); - context.handle(_layoutModeMeta, const VerificationResult.success()); - context.handle(_localeMeta, const VerificationResult.success()); - context.handle(_marketMeta, const VerificationResult.success()); - context.handle(_searchModeMeta, const VerificationResult.success()); if (data.containsKey('download_location')) { context.handle( _downloadLocationMeta, downloadLocation.isAcceptableOrUnknown( data['download_location']!, _downloadLocationMeta)); } - context.handle( - _localLibraryLocationMeta, const VerificationResult.success()); - if (data.containsKey('piped_instance')) { + if (data.containsKey('audio_source_id')) { context.handle( - _pipedInstanceMeta, - pipedInstance.isAcceptableOrUnknown( - data['piped_instance']!, _pipedInstanceMeta)); + _audioSourceIdMeta, + audioSourceId.isAcceptableOrUnknown( + data['audio_source_id']!, _audioSourceIdMeta)); } - if (data.containsKey('invidious_instance')) { - context.handle( - _invidiousInstanceMeta, - invidiousInstance.isAcceptableOrUnknown( - data['invidious_instance']!, _invidiousInstanceMeta)); - } - context.handle(_themeModeMeta, const VerificationResult.success()); - context.handle(_audioSourceMeta, const VerificationResult.success()); - context.handle(_streamMusicCodecMeta, const VerificationResult.success()); - context.handle(_downloadMusicCodecMeta, const VerificationResult.success()); if (data.containsKey('discord_presence')) { context.handle( _discordPresenceMeta, @@ -957,6 +877,12 @@ class $PreferencesTableTable extends PreferencesTable enableConnect.isAcceptableOrUnknown( data['enable_connect']!, _enableConnectMeta)); } + if (data.containsKey('connect_port')) { + context.handle( + _connectPortMeta, + connectPort.isAcceptableOrUnknown( + data['connect_port']!, _connectPortMeta)); + } if (data.containsKey('cache_music')) { context.handle( _cacheMusicMeta, @@ -974,9 +900,6 @@ class $PreferencesTableTable extends PreferencesTable return PreferencesTableData( id: attachedDatabase.typeMapping .read(DriftSqlType.int, data['${effectivePrefix}id'])!, - audioQuality: $PreferencesTableTable.$converteraudioQuality.fromSql( - attachedDatabase.typeMapping.read( - DriftSqlType.string, data['${effectivePrefix}audio_quality'])!), albumColorSync: attachedDatabase.typeMapping .read(DriftSqlType.bool, data['${effectivePrefix}album_color_sync'])!, amoledDarkTheme: attachedDatabase.typeMapping.read( @@ -1015,28 +938,22 @@ class $PreferencesTableTable extends PreferencesTable .$converterlocalLibraryLocation .fromSql(attachedDatabase.typeMapping.read(DriftSqlType.string, data['${effectivePrefix}local_library_location'])!), - pipedInstance: attachedDatabase.typeMapping - .read(DriftSqlType.string, data['${effectivePrefix}piped_instance'])!, - invidiousInstance: attachedDatabase.typeMapping.read( - DriftSqlType.string, data['${effectivePrefix}invidious_instance'])!, themeMode: $PreferencesTableTable.$converterthemeMode.fromSql( attachedDatabase.typeMapping.read( DriftSqlType.string, data['${effectivePrefix}theme_mode'])!), - audioSource: $PreferencesTableTable.$converteraudioSource.fromSql( - attachedDatabase.typeMapping.read( - DriftSqlType.string, data['${effectivePrefix}audio_source'])!), - streamMusicCodec: $PreferencesTableTable.$converterstreamMusicCodec + audioSourceId: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}audio_source_id']), + youtubeClientEngine: $PreferencesTableTable.$converteryoutubeClientEngine .fromSql(attachedDatabase.typeMapping.read(DriftSqlType.string, - data['${effectivePrefix}stream_music_codec'])!), - downloadMusicCodec: $PreferencesTableTable.$converterdownloadMusicCodec - .fromSql(attachedDatabase.typeMapping.read(DriftSqlType.string, - data['${effectivePrefix}download_music_codec'])!), + data['${effectivePrefix}youtube_client_engine'])!), discordPresence: attachedDatabase.typeMapping .read(DriftSqlType.bool, data['${effectivePrefix}discord_presence'])!, endlessPlayback: attachedDatabase.typeMapping .read(DriftSqlType.bool, data['${effectivePrefix}endless_playback'])!, enableConnect: attachedDatabase.typeMapping .read(DriftSqlType.bool, data['${effectivePrefix}enable_connect'])!, + connectPort: attachedDatabase.typeMapping + .read(DriftSqlType.int, data['${effectivePrefix}connect_port'])!, cacheMusic: attachedDatabase.typeMapping .read(DriftSqlType.bool, data['${effectivePrefix}cache_music'])!, ); @@ -1047,9 +964,6 @@ class $PreferencesTableTable extends PreferencesTable return $PreferencesTableTable(attachedDatabase, alias); } - static JsonTypeConverter2 - $converteraudioQuality = - const EnumNameConverter(SourceQualities.values); static JsonTypeConverter2 $convertercloseBehavior = const EnumNameConverter(CloseBehavior.values); @@ -1067,20 +981,14 @@ class $PreferencesTableTable extends PreferencesTable const StringListConverter(); static JsonTypeConverter2 $converterthemeMode = const EnumNameConverter(ThemeMode.values); - static JsonTypeConverter2 $converteraudioSource = - const EnumNameConverter(AudioSource.values); - static JsonTypeConverter2 - $converterstreamMusicCodec = - const EnumNameConverter(SourceCodecs.values); - static JsonTypeConverter2 - $converterdownloadMusicCodec = - const EnumNameConverter(SourceCodecs.values); + static JsonTypeConverter2 + $converteryoutubeClientEngine = + const EnumNameConverter(YoutubeClientEngine.values); } class PreferencesTableData extends DataClass implements Insertable { final int id; - final SourceQualities audioQuality; final bool albumColorSync; final bool amoledDarkTheme; final bool checkUpdate; @@ -1096,19 +1004,16 @@ class PreferencesTableData extends DataClass final SearchMode searchMode; final String downloadLocation; final List localLibraryLocation; - final String pipedInstance; - final String invidiousInstance; final ThemeMode themeMode; - final AudioSource audioSource; - final SourceCodecs streamMusicCodec; - final SourceCodecs downloadMusicCodec; + final String? audioSourceId; + final YoutubeClientEngine youtubeClientEngine; final bool discordPresence; final bool endlessPlayback; final bool enableConnect; + final int connectPort; final bool cacheMusic; const PreferencesTableData( {required this.id, - required this.audioQuality, required this.albumColorSync, required this.amoledDarkTheme, required this.checkUpdate, @@ -1124,24 +1029,18 @@ class PreferencesTableData extends DataClass required this.searchMode, required this.downloadLocation, required this.localLibraryLocation, - required this.pipedInstance, - required this.invidiousInstance, required this.themeMode, - required this.audioSource, - required this.streamMusicCodec, - required this.downloadMusicCodec, + this.audioSourceId, + required this.youtubeClientEngine, required this.discordPresence, required this.endlessPlayback, required this.enableConnect, + required this.connectPort, required this.cacheMusic}); @override Map toColumns(bool nullToAbsent) { final map = {}; map['id'] = Variable(id); - { - map['audio_quality'] = Variable( - $PreferencesTableTable.$converteraudioQuality.toSql(audioQuality)); - } map['album_color_sync'] = Variable(albumColorSync); map['amoled_dark_theme'] = Variable(amoledDarkTheme); map['check_update'] = Variable(checkUpdate); @@ -1180,29 +1079,22 @@ class PreferencesTableData extends DataClass .$converterlocalLibraryLocation .toSql(localLibraryLocation)); } - map['piped_instance'] = Variable(pipedInstance); - map['invidious_instance'] = Variable(invidiousInstance); { map['theme_mode'] = Variable( $PreferencesTableTable.$converterthemeMode.toSql(themeMode)); } - { - map['audio_source'] = Variable( - $PreferencesTableTable.$converteraudioSource.toSql(audioSource)); + if (!nullToAbsent || audioSourceId != null) { + map['audio_source_id'] = Variable(audioSourceId); } { - map['stream_music_codec'] = Variable($PreferencesTableTable - .$converterstreamMusicCodec - .toSql(streamMusicCodec)); - } - { - map['download_music_codec'] = Variable($PreferencesTableTable - .$converterdownloadMusicCodec - .toSql(downloadMusicCodec)); + map['youtube_client_engine'] = Variable($PreferencesTableTable + .$converteryoutubeClientEngine + .toSql(youtubeClientEngine)); } map['discord_presence'] = Variable(discordPresence); map['endless_playback'] = Variable(endlessPlayback); map['enable_connect'] = Variable(enableConnect); + map['connect_port'] = Variable(connectPort); map['cache_music'] = Variable(cacheMusic); return map; } @@ -1210,7 +1102,6 @@ class PreferencesTableData extends DataClass PreferencesTableCompanion toCompanion(bool nullToAbsent) { return PreferencesTableCompanion( id: Value(id), - audioQuality: Value(audioQuality), albumColorSync: Value(albumColorSync), amoledDarkTheme: Value(amoledDarkTheme), checkUpdate: Value(checkUpdate), @@ -1226,15 +1117,15 @@ class PreferencesTableData extends DataClass searchMode: Value(searchMode), downloadLocation: Value(downloadLocation), localLibraryLocation: Value(localLibraryLocation), - pipedInstance: Value(pipedInstance), - invidiousInstance: Value(invidiousInstance), themeMode: Value(themeMode), - audioSource: Value(audioSource), - streamMusicCodec: Value(streamMusicCodec), - downloadMusicCodec: Value(downloadMusicCodec), + audioSourceId: audioSourceId == null && nullToAbsent + ? const Value.absent() + : Value(audioSourceId), + youtubeClientEngine: Value(youtubeClientEngine), discordPresence: Value(discordPresence), endlessPlayback: Value(endlessPlayback), enableConnect: Value(enableConnect), + connectPort: Value(connectPort), cacheMusic: Value(cacheMusic), ); } @@ -1244,8 +1135,6 @@ class PreferencesTableData extends DataClass serializer ??= driftRuntimeOptions.defaultSerializer; return PreferencesTableData( id: serializer.fromJson(json['id']), - audioQuality: $PreferencesTableTable.$converteraudioQuality - .fromJson(serializer.fromJson(json['audioQuality'])), albumColorSync: serializer.fromJson(json['albumColorSync']), amoledDarkTheme: serializer.fromJson(json['amoledDarkTheme']), checkUpdate: serializer.fromJson(json['checkUpdate']), @@ -1267,19 +1156,15 @@ class PreferencesTableData extends DataClass downloadLocation: serializer.fromJson(json['downloadLocation']), localLibraryLocation: serializer.fromJson>(json['localLibraryLocation']), - pipedInstance: serializer.fromJson(json['pipedInstance']), - invidiousInstance: serializer.fromJson(json['invidiousInstance']), themeMode: $PreferencesTableTable.$converterthemeMode .fromJson(serializer.fromJson(json['themeMode'])), - audioSource: $PreferencesTableTable.$converteraudioSource - .fromJson(serializer.fromJson(json['audioSource'])), - streamMusicCodec: $PreferencesTableTable.$converterstreamMusicCodec - .fromJson(serializer.fromJson(json['streamMusicCodec'])), - downloadMusicCodec: $PreferencesTableTable.$converterdownloadMusicCodec - .fromJson(serializer.fromJson(json['downloadMusicCodec'])), + audioSourceId: serializer.fromJson(json['audioSourceId']), + youtubeClientEngine: $PreferencesTableTable.$converteryoutubeClientEngine + .fromJson(serializer.fromJson(json['youtubeClientEngine'])), discordPresence: serializer.fromJson(json['discordPresence']), endlessPlayback: serializer.fromJson(json['endlessPlayback']), enableConnect: serializer.fromJson(json['enableConnect']), + connectPort: serializer.fromJson(json['connectPort']), cacheMusic: serializer.fromJson(json['cacheMusic']), ); } @@ -1288,8 +1173,6 @@ class PreferencesTableData extends DataClass serializer ??= driftRuntimeOptions.defaultSerializer; return { 'id': serializer.toJson(id), - 'audioQuality': serializer.toJson( - $PreferencesTableTable.$converteraudioQuality.toJson(audioQuality)), 'albumColorSync': serializer.toJson(albumColorSync), 'amoledDarkTheme': serializer.toJson(amoledDarkTheme), 'checkUpdate': serializer.toJson(checkUpdate), @@ -1310,28 +1193,22 @@ class PreferencesTableData extends DataClass 'downloadLocation': serializer.toJson(downloadLocation), 'localLibraryLocation': serializer.toJson>(localLibraryLocation), - 'pipedInstance': serializer.toJson(pipedInstance), - 'invidiousInstance': serializer.toJson(invidiousInstance), 'themeMode': serializer.toJson( $PreferencesTableTable.$converterthemeMode.toJson(themeMode)), - 'audioSource': serializer.toJson( - $PreferencesTableTable.$converteraudioSource.toJson(audioSource)), - 'streamMusicCodec': serializer.toJson($PreferencesTableTable - .$converterstreamMusicCodec - .toJson(streamMusicCodec)), - 'downloadMusicCodec': serializer.toJson($PreferencesTableTable - .$converterdownloadMusicCodec - .toJson(downloadMusicCodec)), + 'audioSourceId': serializer.toJson(audioSourceId), + 'youtubeClientEngine': serializer.toJson($PreferencesTableTable + .$converteryoutubeClientEngine + .toJson(youtubeClientEngine)), 'discordPresence': serializer.toJson(discordPresence), 'endlessPlayback': serializer.toJson(endlessPlayback), 'enableConnect': serializer.toJson(enableConnect), + 'connectPort': serializer.toJson(connectPort), 'cacheMusic': serializer.toJson(cacheMusic), }; } PreferencesTableData copyWith( {int? id, - SourceQualities? audioQuality, bool? albumColorSync, bool? amoledDarkTheme, bool? checkUpdate, @@ -1347,19 +1224,16 @@ class PreferencesTableData extends DataClass SearchMode? searchMode, String? downloadLocation, List? localLibraryLocation, - String? pipedInstance, - String? invidiousInstance, ThemeMode? themeMode, - AudioSource? audioSource, - SourceCodecs? streamMusicCodec, - SourceCodecs? downloadMusicCodec, + Value audioSourceId = const Value.absent(), + YoutubeClientEngine? youtubeClientEngine, bool? discordPresence, bool? endlessPlayback, bool? enableConnect, + int? connectPort, bool? cacheMusic}) => PreferencesTableData( id: id ?? this.id, - audioQuality: audioQuality ?? this.audioQuality, albumColorSync: albumColorSync ?? this.albumColorSync, amoledDarkTheme: amoledDarkTheme ?? this.amoledDarkTheme, checkUpdate: checkUpdate ?? this.checkUpdate, @@ -1375,23 +1249,19 @@ class PreferencesTableData extends DataClass searchMode: searchMode ?? this.searchMode, downloadLocation: downloadLocation ?? this.downloadLocation, localLibraryLocation: localLibraryLocation ?? this.localLibraryLocation, - pipedInstance: pipedInstance ?? this.pipedInstance, - invidiousInstance: invidiousInstance ?? this.invidiousInstance, themeMode: themeMode ?? this.themeMode, - audioSource: audioSource ?? this.audioSource, - streamMusicCodec: streamMusicCodec ?? this.streamMusicCodec, - downloadMusicCodec: downloadMusicCodec ?? this.downloadMusicCodec, + audioSourceId: + audioSourceId.present ? audioSourceId.value : this.audioSourceId, + youtubeClientEngine: youtubeClientEngine ?? this.youtubeClientEngine, discordPresence: discordPresence ?? this.discordPresence, endlessPlayback: endlessPlayback ?? this.endlessPlayback, enableConnect: enableConnect ?? this.enableConnect, + connectPort: connectPort ?? this.connectPort, cacheMusic: cacheMusic ?? this.cacheMusic, ); PreferencesTableData copyWithCompanion(PreferencesTableCompanion data) { return PreferencesTableData( id: data.id.present ? data.id.value : this.id, - audioQuality: data.audioQuality.present - ? data.audioQuality.value - : this.audioQuality, albumColorSync: data.albumColorSync.present ? data.albumColorSync.value : this.albumColorSync, @@ -1430,21 +1300,13 @@ class PreferencesTableData extends DataClass localLibraryLocation: data.localLibraryLocation.present ? data.localLibraryLocation.value : this.localLibraryLocation, - pipedInstance: data.pipedInstance.present - ? data.pipedInstance.value - : this.pipedInstance, - invidiousInstance: data.invidiousInstance.present - ? data.invidiousInstance.value - : this.invidiousInstance, themeMode: data.themeMode.present ? data.themeMode.value : this.themeMode, - audioSource: - data.audioSource.present ? data.audioSource.value : this.audioSource, - streamMusicCodec: data.streamMusicCodec.present - ? data.streamMusicCodec.value - : this.streamMusicCodec, - downloadMusicCodec: data.downloadMusicCodec.present - ? data.downloadMusicCodec.value - : this.downloadMusicCodec, + audioSourceId: data.audioSourceId.present + ? data.audioSourceId.value + : this.audioSourceId, + youtubeClientEngine: data.youtubeClientEngine.present + ? data.youtubeClientEngine.value + : this.youtubeClientEngine, discordPresence: data.discordPresence.present ? data.discordPresence.value : this.discordPresence, @@ -1454,6 +1316,8 @@ class PreferencesTableData extends DataClass enableConnect: data.enableConnect.present ? data.enableConnect.value : this.enableConnect, + connectPort: + data.connectPort.present ? data.connectPort.value : this.connectPort, cacheMusic: data.cacheMusic.present ? data.cacheMusic.value : this.cacheMusic, ); @@ -1463,7 +1327,6 @@ class PreferencesTableData extends DataClass String toString() { return (StringBuffer('PreferencesTableData(') ..write('id: $id, ') - ..write('audioQuality: $audioQuality, ') ..write('albumColorSync: $albumColorSync, ') ..write('amoledDarkTheme: $amoledDarkTheme, ') ..write('checkUpdate: $checkUpdate, ') @@ -1479,15 +1342,13 @@ class PreferencesTableData extends DataClass ..write('searchMode: $searchMode, ') ..write('downloadLocation: $downloadLocation, ') ..write('localLibraryLocation: $localLibraryLocation, ') - ..write('pipedInstance: $pipedInstance, ') - ..write('invidiousInstance: $invidiousInstance, ') ..write('themeMode: $themeMode, ') - ..write('audioSource: $audioSource, ') - ..write('streamMusicCodec: $streamMusicCodec, ') - ..write('downloadMusicCodec: $downloadMusicCodec, ') + ..write('audioSourceId: $audioSourceId, ') + ..write('youtubeClientEngine: $youtubeClientEngine, ') ..write('discordPresence: $discordPresence, ') ..write('endlessPlayback: $endlessPlayback, ') ..write('enableConnect: $enableConnect, ') + ..write('connectPort: $connectPort, ') ..write('cacheMusic: $cacheMusic') ..write(')')) .toString(); @@ -1496,7 +1357,6 @@ class PreferencesTableData extends DataClass @override int get hashCode => Object.hashAll([ id, - audioQuality, albumColorSync, amoledDarkTheme, checkUpdate, @@ -1512,15 +1372,13 @@ class PreferencesTableData extends DataClass searchMode, downloadLocation, localLibraryLocation, - pipedInstance, - invidiousInstance, themeMode, - audioSource, - streamMusicCodec, - downloadMusicCodec, + audioSourceId, + youtubeClientEngine, discordPresence, endlessPlayback, enableConnect, + connectPort, cacheMusic ]); @override @@ -1528,7 +1386,6 @@ class PreferencesTableData extends DataClass identical(this, other) || (other is PreferencesTableData && other.id == this.id && - other.audioQuality == this.audioQuality && other.albumColorSync == this.albumColorSync && other.amoledDarkTheme == this.amoledDarkTheme && other.checkUpdate == this.checkUpdate && @@ -1544,21 +1401,18 @@ class PreferencesTableData extends DataClass other.searchMode == this.searchMode && other.downloadLocation == this.downloadLocation && other.localLibraryLocation == this.localLibraryLocation && - other.pipedInstance == this.pipedInstance && - other.invidiousInstance == this.invidiousInstance && other.themeMode == this.themeMode && - other.audioSource == this.audioSource && - other.streamMusicCodec == this.streamMusicCodec && - other.downloadMusicCodec == this.downloadMusicCodec && + other.audioSourceId == this.audioSourceId && + other.youtubeClientEngine == this.youtubeClientEngine && other.discordPresence == this.discordPresence && other.endlessPlayback == this.endlessPlayback && other.enableConnect == this.enableConnect && + other.connectPort == this.connectPort && other.cacheMusic == this.cacheMusic); } class PreferencesTableCompanion extends UpdateCompanion { final Value id; - final Value audioQuality; final Value albumColorSync; final Value amoledDarkTheme; final Value checkUpdate; @@ -1574,19 +1428,16 @@ class PreferencesTableCompanion extends UpdateCompanion { final Value searchMode; final Value downloadLocation; final Value> localLibraryLocation; - final Value pipedInstance; - final Value invidiousInstance; final Value themeMode; - final Value audioSource; - final Value streamMusicCodec; - final Value downloadMusicCodec; + final Value audioSourceId; + final Value youtubeClientEngine; final Value discordPresence; final Value endlessPlayback; final Value enableConnect; + final Value connectPort; final Value cacheMusic; const PreferencesTableCompanion({ this.id = const Value.absent(), - this.audioQuality = const Value.absent(), this.albumColorSync = const Value.absent(), this.amoledDarkTheme = const Value.absent(), this.checkUpdate = const Value.absent(), @@ -1602,20 +1453,17 @@ class PreferencesTableCompanion extends UpdateCompanion { this.searchMode = const Value.absent(), this.downloadLocation = const Value.absent(), this.localLibraryLocation = const Value.absent(), - this.pipedInstance = const Value.absent(), - this.invidiousInstance = const Value.absent(), this.themeMode = const Value.absent(), - this.audioSource = const Value.absent(), - this.streamMusicCodec = const Value.absent(), - this.downloadMusicCodec = const Value.absent(), + this.audioSourceId = const Value.absent(), + this.youtubeClientEngine = const Value.absent(), this.discordPresence = const Value.absent(), this.endlessPlayback = const Value.absent(), this.enableConnect = const Value.absent(), + this.connectPort = const Value.absent(), this.cacheMusic = const Value.absent(), }); PreferencesTableCompanion.insert({ this.id = const Value.absent(), - this.audioQuality = const Value.absent(), this.albumColorSync = const Value.absent(), this.amoledDarkTheme = const Value.absent(), this.checkUpdate = const Value.absent(), @@ -1631,20 +1479,17 @@ class PreferencesTableCompanion extends UpdateCompanion { this.searchMode = const Value.absent(), this.downloadLocation = const Value.absent(), this.localLibraryLocation = const Value.absent(), - this.pipedInstance = const Value.absent(), - this.invidiousInstance = const Value.absent(), this.themeMode = const Value.absent(), - this.audioSource = const Value.absent(), - this.streamMusicCodec = const Value.absent(), - this.downloadMusicCodec = const Value.absent(), + this.audioSourceId = const Value.absent(), + this.youtubeClientEngine = const Value.absent(), this.discordPresence = const Value.absent(), this.endlessPlayback = const Value.absent(), this.enableConnect = const Value.absent(), + this.connectPort = const Value.absent(), this.cacheMusic = const Value.absent(), }); static Insertable custom({ Expression? id, - Expression? audioQuality, Expression? albumColorSync, Expression? amoledDarkTheme, Expression? checkUpdate, @@ -1660,20 +1505,17 @@ class PreferencesTableCompanion extends UpdateCompanion { Expression? searchMode, Expression? downloadLocation, Expression? localLibraryLocation, - Expression? pipedInstance, - Expression? invidiousInstance, Expression? themeMode, - Expression? audioSource, - Expression? streamMusicCodec, - Expression? downloadMusicCodec, + Expression? audioSourceId, + Expression? youtubeClientEngine, Expression? discordPresence, Expression? endlessPlayback, Expression? enableConnect, + Expression? connectPort, Expression? cacheMusic, }) { return RawValuesInsertable({ if (id != null) 'id': id, - if (audioQuality != null) 'audio_quality': audioQuality, if (albumColorSync != null) 'album_color_sync': albumColorSync, if (amoledDarkTheme != null) 'amoled_dark_theme': amoledDarkTheme, if (checkUpdate != null) 'check_update': checkUpdate, @@ -1691,23 +1533,20 @@ class PreferencesTableCompanion extends UpdateCompanion { if (downloadLocation != null) 'download_location': downloadLocation, if (localLibraryLocation != null) 'local_library_location': localLibraryLocation, - if (pipedInstance != null) 'piped_instance': pipedInstance, - if (invidiousInstance != null) 'invidious_instance': invidiousInstance, if (themeMode != null) 'theme_mode': themeMode, - if (audioSource != null) 'audio_source': audioSource, - if (streamMusicCodec != null) 'stream_music_codec': streamMusicCodec, - if (downloadMusicCodec != null) - 'download_music_codec': downloadMusicCodec, + if (audioSourceId != null) 'audio_source_id': audioSourceId, + if (youtubeClientEngine != null) + 'youtube_client_engine': youtubeClientEngine, if (discordPresence != null) 'discord_presence': discordPresence, if (endlessPlayback != null) 'endless_playback': endlessPlayback, if (enableConnect != null) 'enable_connect': enableConnect, + if (connectPort != null) 'connect_port': connectPort, if (cacheMusic != null) 'cache_music': cacheMusic, }); } PreferencesTableCompanion copyWith( {Value? id, - Value? audioQuality, Value? albumColorSync, Value? amoledDarkTheme, Value? checkUpdate, @@ -1723,19 +1562,16 @@ class PreferencesTableCompanion extends UpdateCompanion { Value? searchMode, Value? downloadLocation, Value>? localLibraryLocation, - Value? pipedInstance, - Value? invidiousInstance, Value? themeMode, - Value? audioSource, - Value? streamMusicCodec, - Value? downloadMusicCodec, + Value? audioSourceId, + Value? youtubeClientEngine, Value? discordPresence, Value? endlessPlayback, Value? enableConnect, + Value? connectPort, Value? cacheMusic}) { return PreferencesTableCompanion( id: id ?? this.id, - audioQuality: audioQuality ?? this.audioQuality, albumColorSync: albumColorSync ?? this.albumColorSync, amoledDarkTheme: amoledDarkTheme ?? this.amoledDarkTheme, checkUpdate: checkUpdate ?? this.checkUpdate, @@ -1751,15 +1587,13 @@ class PreferencesTableCompanion extends UpdateCompanion { searchMode: searchMode ?? this.searchMode, downloadLocation: downloadLocation ?? this.downloadLocation, localLibraryLocation: localLibraryLocation ?? this.localLibraryLocation, - pipedInstance: pipedInstance ?? this.pipedInstance, - invidiousInstance: invidiousInstance ?? this.invidiousInstance, themeMode: themeMode ?? this.themeMode, - audioSource: audioSource ?? this.audioSource, - streamMusicCodec: streamMusicCodec ?? this.streamMusicCodec, - downloadMusicCodec: downloadMusicCodec ?? this.downloadMusicCodec, + audioSourceId: audioSourceId ?? this.audioSourceId, + youtubeClientEngine: youtubeClientEngine ?? this.youtubeClientEngine, discordPresence: discordPresence ?? this.discordPresence, endlessPlayback: endlessPlayback ?? this.endlessPlayback, enableConnect: enableConnect ?? this.enableConnect, + connectPort: connectPort ?? this.connectPort, cacheMusic: cacheMusic ?? this.cacheMusic, ); } @@ -1770,11 +1604,6 @@ class PreferencesTableCompanion extends UpdateCompanion { if (id.present) { map['id'] = Variable(id.value); } - if (audioQuality.present) { - map['audio_quality'] = Variable($PreferencesTableTable - .$converteraudioQuality - .toSql(audioQuality.value)); - } if (albumColorSync.present) { map['album_color_sync'] = Variable(albumColorSync.value); } @@ -1830,30 +1659,17 @@ class PreferencesTableCompanion extends UpdateCompanion { .$converterlocalLibraryLocation .toSql(localLibraryLocation.value)); } - if (pipedInstance.present) { - map['piped_instance'] = Variable(pipedInstance.value); - } - if (invidiousInstance.present) { - map['invidious_instance'] = Variable(invidiousInstance.value); - } if (themeMode.present) { map['theme_mode'] = Variable( $PreferencesTableTable.$converterthemeMode.toSql(themeMode.value)); } - if (audioSource.present) { - map['audio_source'] = Variable($PreferencesTableTable - .$converteraudioSource - .toSql(audioSource.value)); + if (audioSourceId.present) { + map['audio_source_id'] = Variable(audioSourceId.value); } - if (streamMusicCodec.present) { - map['stream_music_codec'] = Variable($PreferencesTableTable - .$converterstreamMusicCodec - .toSql(streamMusicCodec.value)); - } - if (downloadMusicCodec.present) { - map['download_music_codec'] = Variable($PreferencesTableTable - .$converterdownloadMusicCodec - .toSql(downloadMusicCodec.value)); + if (youtubeClientEngine.present) { + map['youtube_client_engine'] = Variable($PreferencesTableTable + .$converteryoutubeClientEngine + .toSql(youtubeClientEngine.value)); } if (discordPresence.present) { map['discord_presence'] = Variable(discordPresence.value); @@ -1864,6 +1680,9 @@ class PreferencesTableCompanion extends UpdateCompanion { if (enableConnect.present) { map['enable_connect'] = Variable(enableConnect.value); } + if (connectPort.present) { + map['connect_port'] = Variable(connectPort.value); + } if (cacheMusic.present) { map['cache_music'] = Variable(cacheMusic.value); } @@ -1874,7 +1693,6 @@ class PreferencesTableCompanion extends UpdateCompanion { String toString() { return (StringBuffer('PreferencesTableCompanion(') ..write('id: $id, ') - ..write('audioQuality: $audioQuality, ') ..write('albumColorSync: $albumColorSync, ') ..write('amoledDarkTheme: $amoledDarkTheme, ') ..write('checkUpdate: $checkUpdate, ') @@ -1890,15 +1708,13 @@ class PreferencesTableCompanion extends UpdateCompanion { ..write('searchMode: $searchMode, ') ..write('downloadLocation: $downloadLocation, ') ..write('localLibraryLocation: $localLibraryLocation, ') - ..write('pipedInstance: $pipedInstance, ') - ..write('invidiousInstance: $invidiousInstance, ') ..write('themeMode: $themeMode, ') - ..write('audioSource: $audioSource, ') - ..write('streamMusicCodec: $streamMusicCodec, ') - ..write('downloadMusicCodec: $downloadMusicCodec, ') + ..write('audioSourceId: $audioSourceId, ') + ..write('youtubeClientEngine: $youtubeClientEngine, ') ..write('discordPresence: $discordPresence, ') ..write('endlessPlayback: $endlessPlayback, ') ..write('enableConnect: $enableConnect, ') + ..write('connectPort: $connectPort, ') ..write('cacheMusic: $cacheMusic') ..write(')')) .toString(); @@ -1934,8 +1750,6 @@ class $ScrobblerTableTable extends ScrobblerTable late final GeneratedColumn username = GeneratedColumn( 'username', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: true); - static const VerificationMeta _passwordHashMeta = - const VerificationMeta('passwordHash'); @override late final GeneratedColumnWithTypeConverter passwordHash = GeneratedColumn( @@ -1968,7 +1782,6 @@ class $ScrobblerTableTable extends ScrobblerTable } else if (isInserting) { context.missing(_usernameMeta); } - context.handle(_passwordHashMeta, const VerificationResult.success()); return context; } @@ -2493,22 +2306,20 @@ class $SourceMatchTableTable extends SourceMatchTable late final GeneratedColumn trackId = GeneratedColumn( 'track_id', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: true); - static const VerificationMeta _sourceIdMeta = - const VerificationMeta('sourceId'); + static const VerificationMeta _sourceInfoMeta = + const VerificationMeta('sourceInfo'); @override - late final GeneratedColumn sourceId = GeneratedColumn( - 'source_id', aliasedName, false, - type: DriftSqlType.string, requiredDuringInsert: true); + late final GeneratedColumn sourceInfo = GeneratedColumn( + 'source_info', aliasedName, false, + type: DriftSqlType.string, + requiredDuringInsert: false, + defaultValue: const Constant("{}")); static const VerificationMeta _sourceTypeMeta = const VerificationMeta('sourceType'); @override - late final GeneratedColumnWithTypeConverter sourceType = - GeneratedColumn('source_type', aliasedName, false, - type: DriftSqlType.string, - requiredDuringInsert: false, - defaultValue: Constant(SourceType.youtube.name)) - .withConverter( - $SourceMatchTableTable.$convertersourceType); + late final GeneratedColumn sourceType = GeneratedColumn( + 'source_type', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true); static const VerificationMeta _createdAtMeta = const VerificationMeta('createdAt'); @override @@ -2519,7 +2330,7 @@ class $SourceMatchTableTable extends SourceMatchTable defaultValue: currentDateAndTime); @override List get $columns => - [id, trackId, sourceId, sourceType, createdAt]; + [id, trackId, sourceInfo, sourceType, createdAt]; @override String get aliasedName => _alias ?? actualTableName; @override @@ -2540,13 +2351,20 @@ class $SourceMatchTableTable extends SourceMatchTable } else if (isInserting) { context.missing(_trackIdMeta); } - if (data.containsKey('source_id')) { - context.handle(_sourceIdMeta, - sourceId.isAcceptableOrUnknown(data['source_id']!, _sourceIdMeta)); - } else if (isInserting) { - context.missing(_sourceIdMeta); + if (data.containsKey('source_info')) { + context.handle( + _sourceInfoMeta, + sourceInfo.isAcceptableOrUnknown( + data['source_info']!, _sourceInfoMeta)); + } + if (data.containsKey('source_type')) { + context.handle( + _sourceTypeMeta, + sourceType.isAcceptableOrUnknown( + data['source_type']!, _sourceTypeMeta)); + } else if (isInserting) { + context.missing(_sourceTypeMeta); } - context.handle(_sourceTypeMeta, const VerificationResult.success()); if (data.containsKey('created_at')) { context.handle(_createdAtMeta, createdAt.isAcceptableOrUnknown(data['created_at']!, _createdAtMeta)); @@ -2564,11 +2382,10 @@ class $SourceMatchTableTable extends SourceMatchTable .read(DriftSqlType.int, data['${effectivePrefix}id'])!, trackId: attachedDatabase.typeMapping .read(DriftSqlType.string, data['${effectivePrefix}track_id'])!, - sourceId: attachedDatabase.typeMapping - .read(DriftSqlType.string, data['${effectivePrefix}source_id'])!, - sourceType: $SourceMatchTableTable.$convertersourceType.fromSql( - attachedDatabase.typeMapping.read( - DriftSqlType.string, data['${effectivePrefix}source_type'])!), + sourceInfo: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}source_info'])!, + sourceType: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}source_type'])!, createdAt: attachedDatabase.typeMapping .read(DriftSqlType.dateTime, data['${effectivePrefix}created_at'])!, ); @@ -2578,22 +2395,19 @@ class $SourceMatchTableTable extends SourceMatchTable $SourceMatchTableTable createAlias(String alias) { return $SourceMatchTableTable(attachedDatabase, alias); } - - static JsonTypeConverter2 $convertersourceType = - const EnumNameConverter(SourceType.values); } class SourceMatchTableData extends DataClass implements Insertable { final int id; final String trackId; - final String sourceId; - final SourceType sourceType; + final String sourceInfo; + final String sourceType; final DateTime createdAt; const SourceMatchTableData( {required this.id, required this.trackId, - required this.sourceId, + required this.sourceInfo, required this.sourceType, required this.createdAt}); @override @@ -2601,11 +2415,8 @@ class SourceMatchTableData extends DataClass final map = {}; map['id'] = Variable(id); map['track_id'] = Variable(trackId); - map['source_id'] = Variable(sourceId); - { - map['source_type'] = Variable( - $SourceMatchTableTable.$convertersourceType.toSql(sourceType)); - } + map['source_info'] = Variable(sourceInfo); + map['source_type'] = Variable(sourceType); map['created_at'] = Variable(createdAt); return map; } @@ -2614,7 +2425,7 @@ class SourceMatchTableData extends DataClass return SourceMatchTableCompanion( id: Value(id), trackId: Value(trackId), - sourceId: Value(sourceId), + sourceInfo: Value(sourceInfo), sourceType: Value(sourceType), createdAt: Value(createdAt), ); @@ -2626,9 +2437,8 @@ class SourceMatchTableData extends DataClass return SourceMatchTableData( id: serializer.fromJson(json['id']), trackId: serializer.fromJson(json['trackId']), - sourceId: serializer.fromJson(json['sourceId']), - sourceType: $SourceMatchTableTable.$convertersourceType - .fromJson(serializer.fromJson(json['sourceType'])), + sourceInfo: serializer.fromJson(json['sourceInfo']), + sourceType: serializer.fromJson(json['sourceType']), createdAt: serializer.fromJson(json['createdAt']), ); } @@ -2638,9 +2448,8 @@ class SourceMatchTableData extends DataClass return { 'id': serializer.toJson(id), 'trackId': serializer.toJson(trackId), - 'sourceId': serializer.toJson(sourceId), - 'sourceType': serializer.toJson( - $SourceMatchTableTable.$convertersourceType.toJson(sourceType)), + 'sourceInfo': serializer.toJson(sourceInfo), + 'sourceType': serializer.toJson(sourceType), 'createdAt': serializer.toJson(createdAt), }; } @@ -2648,13 +2457,13 @@ class SourceMatchTableData extends DataClass SourceMatchTableData copyWith( {int? id, String? trackId, - String? sourceId, - SourceType? sourceType, + String? sourceInfo, + String? sourceType, DateTime? createdAt}) => SourceMatchTableData( id: id ?? this.id, trackId: trackId ?? this.trackId, - sourceId: sourceId ?? this.sourceId, + sourceInfo: sourceInfo ?? this.sourceInfo, sourceType: sourceType ?? this.sourceType, createdAt: createdAt ?? this.createdAt, ); @@ -2662,7 +2471,8 @@ class SourceMatchTableData extends DataClass return SourceMatchTableData( id: data.id.present ? data.id.value : this.id, trackId: data.trackId.present ? data.trackId.value : this.trackId, - sourceId: data.sourceId.present ? data.sourceId.value : this.sourceId, + sourceInfo: + data.sourceInfo.present ? data.sourceInfo.value : this.sourceInfo, sourceType: data.sourceType.present ? data.sourceType.value : this.sourceType, createdAt: data.createdAt.present ? data.createdAt.value : this.createdAt, @@ -2674,7 +2484,7 @@ class SourceMatchTableData extends DataClass return (StringBuffer('SourceMatchTableData(') ..write('id: $id, ') ..write('trackId: $trackId, ') - ..write('sourceId: $sourceId, ') + ..write('sourceInfo: $sourceInfo, ') ..write('sourceType: $sourceType, ') ..write('createdAt: $createdAt') ..write(')')) @@ -2682,14 +2492,15 @@ class SourceMatchTableData extends DataClass } @override - int get hashCode => Object.hash(id, trackId, sourceId, sourceType, createdAt); + int get hashCode => + Object.hash(id, trackId, sourceInfo, sourceType, createdAt); @override bool operator ==(Object other) => identical(this, other) || (other is SourceMatchTableData && other.id == this.id && other.trackId == this.trackId && - other.sourceId == this.sourceId && + other.sourceInfo == this.sourceInfo && other.sourceType == this.sourceType && other.createdAt == this.createdAt); } @@ -2697,35 +2508,35 @@ class SourceMatchTableData extends DataClass class SourceMatchTableCompanion extends UpdateCompanion { final Value id; final Value trackId; - final Value sourceId; - final Value sourceType; + final Value sourceInfo; + final Value sourceType; final Value createdAt; const SourceMatchTableCompanion({ this.id = const Value.absent(), this.trackId = const Value.absent(), - this.sourceId = const Value.absent(), + this.sourceInfo = const Value.absent(), this.sourceType = const Value.absent(), this.createdAt = const Value.absent(), }); SourceMatchTableCompanion.insert({ this.id = const Value.absent(), required String trackId, - required String sourceId, - this.sourceType = const Value.absent(), + this.sourceInfo = const Value.absent(), + required String sourceType, this.createdAt = const Value.absent(), }) : trackId = Value(trackId), - sourceId = Value(sourceId); + sourceType = Value(sourceType); static Insertable custom({ Expression? id, Expression? trackId, - Expression? sourceId, + Expression? sourceInfo, Expression? sourceType, Expression? createdAt, }) { return RawValuesInsertable({ if (id != null) 'id': id, if (trackId != null) 'track_id': trackId, - if (sourceId != null) 'source_id': sourceId, + if (sourceInfo != null) 'source_info': sourceInfo, if (sourceType != null) 'source_type': sourceType, if (createdAt != null) 'created_at': createdAt, }); @@ -2734,13 +2545,13 @@ class SourceMatchTableCompanion extends UpdateCompanion { SourceMatchTableCompanion copyWith( {Value? id, Value? trackId, - Value? sourceId, - Value? sourceType, + Value? sourceInfo, + Value? sourceType, Value? createdAt}) { return SourceMatchTableCompanion( id: id ?? this.id, trackId: trackId ?? this.trackId, - sourceId: sourceId ?? this.sourceId, + sourceInfo: sourceInfo ?? this.sourceInfo, sourceType: sourceType ?? this.sourceType, createdAt: createdAt ?? this.createdAt, ); @@ -2755,12 +2566,11 @@ class SourceMatchTableCompanion extends UpdateCompanion { if (trackId.present) { map['track_id'] = Variable(trackId.value); } - if (sourceId.present) { - map['source_id'] = Variable(sourceId.value); + if (sourceInfo.present) { + map['source_info'] = Variable(sourceInfo.value); } if (sourceType.present) { - map['source_type'] = Variable( - $SourceMatchTableTable.$convertersourceType.toSql(sourceType.value)); + map['source_type'] = Variable(sourceType.value); } if (createdAt.present) { map['created_at'] = Variable(createdAt.value); @@ -2773,7 +2583,7 @@ class SourceMatchTableCompanion extends UpdateCompanion { return (StringBuffer('SourceMatchTableCompanion(') ..write('id: $id, ') ..write('trackId: $trackId, ') - ..write('sourceId: $sourceId, ') + ..write('sourceInfo: $sourceInfo, ') ..write('sourceType: $sourceType, ') ..write('createdAt: $createdAt') ..write(')')) @@ -2805,8 +2615,6 @@ class $AudioPlayerStateTableTable extends AudioPlayerStateTable requiredDuringInsert: true, defaultConstraints: GeneratedColumn.constraintIsAlways('CHECK ("playing" IN (0, 1))')); - static const VerificationMeta _loopModeMeta = - const VerificationMeta('loopMode'); @override late final GeneratedColumnWithTypeConverter loopMode = GeneratedColumn('loop_mode', aliasedName, false, @@ -2822,8 +2630,6 @@ class $AudioPlayerStateTableTable extends AudioPlayerStateTable requiredDuringInsert: true, defaultConstraints: GeneratedColumn.constraintIsAlways('CHECK ("shuffled" IN (0, 1))')); - static const VerificationMeta _collectionsMeta = - const VerificationMeta('collections'); @override late final GeneratedColumnWithTypeConverter, String> collections = GeneratedColumn('collections', aliasedName, false, @@ -2831,8 +2637,24 @@ class $AudioPlayerStateTableTable extends AudioPlayerStateTable .withConverter>( $AudioPlayerStateTableTable.$convertercollections); @override + late final GeneratedColumnWithTypeConverter, String> + tracks = GeneratedColumn('tracks', aliasedName, false, + type: DriftSqlType.string, + requiredDuringInsert: false, + defaultValue: const Constant("[]")) + .withConverter>( + $AudioPlayerStateTableTable.$convertertracks); + static const VerificationMeta _currentIndexMeta = + const VerificationMeta('currentIndex'); + @override + late final GeneratedColumn currentIndex = GeneratedColumn( + 'current_index', aliasedName, false, + type: DriftSqlType.int, + requiredDuringInsert: false, + defaultValue: const Constant(0)); + @override List get $columns => - [id, playing, loopMode, shuffled, collections]; + [id, playing, loopMode, shuffled, collections, tracks, currentIndex]; @override String get aliasedName => _alias ?? actualTableName; @override @@ -2853,14 +2675,18 @@ class $AudioPlayerStateTableTable extends AudioPlayerStateTable } else if (isInserting) { context.missing(_playingMeta); } - context.handle(_loopModeMeta, const VerificationResult.success()); if (data.containsKey('shuffled')) { context.handle(_shuffledMeta, shuffled.isAcceptableOrUnknown(data['shuffled']!, _shuffledMeta)); } else if (isInserting) { context.missing(_shuffledMeta); } - context.handle(_collectionsMeta, const VerificationResult.success()); + if (data.containsKey('current_index')) { + context.handle( + _currentIndexMeta, + currentIndex.isAcceptableOrUnknown( + data['current_index']!, _currentIndexMeta)); + } return context; } @@ -2883,6 +2709,11 @@ class $AudioPlayerStateTableTable extends AudioPlayerStateTable collections: $AudioPlayerStateTableTable.$convertercollections.fromSql( attachedDatabase.typeMapping.read( DriftSqlType.string, data['${effectivePrefix}collections'])!), + tracks: $AudioPlayerStateTableTable.$convertertracks.fromSql( + attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}tracks'])!), + currentIndex: attachedDatabase.typeMapping + .read(DriftSqlType.int, data['${effectivePrefix}current_index'])!, ); } @@ -2895,6 +2726,8 @@ class $AudioPlayerStateTableTable extends AudioPlayerStateTable const EnumNameConverter(PlaylistMode.values); static TypeConverter, String> $convertercollections = const StringListConverter(); + static TypeConverter, String> $convertertracks = + const SpotubeTrackObjectListConverter(); } class AudioPlayerStateTableData extends DataClass @@ -2904,12 +2737,16 @@ class AudioPlayerStateTableData extends DataClass final PlaylistMode loopMode; final bool shuffled; final List collections; + final List tracks; + final int currentIndex; const AudioPlayerStateTableData( {required this.id, required this.playing, required this.loopMode, required this.shuffled, - required this.collections}); + required this.collections, + required this.tracks, + required this.currentIndex}); @override Map toColumns(bool nullToAbsent) { final map = {}; @@ -2924,6 +2761,11 @@ class AudioPlayerStateTableData extends DataClass map['collections'] = Variable( $AudioPlayerStateTableTable.$convertercollections.toSql(collections)); } + { + map['tracks'] = Variable( + $AudioPlayerStateTableTable.$convertertracks.toSql(tracks)); + } + map['current_index'] = Variable(currentIndex); return map; } @@ -2934,6 +2776,8 @@ class AudioPlayerStateTableData extends DataClass loopMode: Value(loopMode), shuffled: Value(shuffled), collections: Value(collections), + tracks: Value(tracks), + currentIndex: Value(currentIndex), ); } @@ -2947,6 +2791,8 @@ class AudioPlayerStateTableData extends DataClass .fromJson(serializer.fromJson(json['loopMode'])), shuffled: serializer.fromJson(json['shuffled']), collections: serializer.fromJson>(json['collections']), + tracks: serializer.fromJson>(json['tracks']), + currentIndex: serializer.fromJson(json['currentIndex']), ); } @override @@ -2959,6 +2805,8 @@ class AudioPlayerStateTableData extends DataClass $AudioPlayerStateTableTable.$converterloopMode.toJson(loopMode)), 'shuffled': serializer.toJson(shuffled), 'collections': serializer.toJson>(collections), + 'tracks': serializer.toJson>(tracks), + 'currentIndex': serializer.toJson(currentIndex), }; } @@ -2967,13 +2815,17 @@ class AudioPlayerStateTableData extends DataClass bool? playing, PlaylistMode? loopMode, bool? shuffled, - List? collections}) => + List? collections, + List? tracks, + int? currentIndex}) => AudioPlayerStateTableData( id: id ?? this.id, playing: playing ?? this.playing, loopMode: loopMode ?? this.loopMode, shuffled: shuffled ?? this.shuffled, collections: collections ?? this.collections, + tracks: tracks ?? this.tracks, + currentIndex: currentIndex ?? this.currentIndex, ); AudioPlayerStateTableData copyWithCompanion( AudioPlayerStateTableCompanion data) { @@ -2984,6 +2836,10 @@ class AudioPlayerStateTableData extends DataClass shuffled: data.shuffled.present ? data.shuffled.value : this.shuffled, collections: data.collections.present ? data.collections.value : this.collections, + tracks: data.tracks.present ? data.tracks.value : this.tracks, + currentIndex: data.currentIndex.present + ? data.currentIndex.value + : this.currentIndex, ); } @@ -2994,13 +2850,16 @@ class AudioPlayerStateTableData extends DataClass ..write('playing: $playing, ') ..write('loopMode: $loopMode, ') ..write('shuffled: $shuffled, ') - ..write('collections: $collections') + ..write('collections: $collections, ') + ..write('tracks: $tracks, ') + ..write('currentIndex: $currentIndex') ..write(')')) .toString(); } @override - int get hashCode => Object.hash(id, playing, loopMode, shuffled, collections); + int get hashCode => Object.hash( + id, playing, loopMode, shuffled, collections, tracks, currentIndex); @override bool operator ==(Object other) => identical(this, other) || @@ -3009,7 +2868,9 @@ class AudioPlayerStateTableData extends DataClass other.playing == this.playing && other.loopMode == this.loopMode && other.shuffled == this.shuffled && - other.collections == this.collections); + other.collections == this.collections && + other.tracks == this.tracks && + other.currentIndex == this.currentIndex); } class AudioPlayerStateTableCompanion @@ -3019,12 +2880,16 @@ class AudioPlayerStateTableCompanion final Value loopMode; final Value shuffled; final Value> collections; + final Value> tracks; + final Value currentIndex; const AudioPlayerStateTableCompanion({ this.id = const Value.absent(), this.playing = const Value.absent(), this.loopMode = const Value.absent(), this.shuffled = const Value.absent(), this.collections = const Value.absent(), + this.tracks = const Value.absent(), + this.currentIndex = const Value.absent(), }); AudioPlayerStateTableCompanion.insert({ this.id = const Value.absent(), @@ -3032,6 +2897,8 @@ class AudioPlayerStateTableCompanion required PlaylistMode loopMode, required bool shuffled, required List collections, + this.tracks = const Value.absent(), + this.currentIndex = const Value.absent(), }) : playing = Value(playing), loopMode = Value(loopMode), shuffled = Value(shuffled), @@ -3042,6 +2909,8 @@ class AudioPlayerStateTableCompanion Expression? loopMode, Expression? shuffled, Expression? collections, + Expression? tracks, + Expression? currentIndex, }) { return RawValuesInsertable({ if (id != null) 'id': id, @@ -3049,6 +2918,8 @@ class AudioPlayerStateTableCompanion if (loopMode != null) 'loop_mode': loopMode, if (shuffled != null) 'shuffled': shuffled, if (collections != null) 'collections': collections, + if (tracks != null) 'tracks': tracks, + if (currentIndex != null) 'current_index': currentIndex, }); } @@ -3057,13 +2928,17 @@ class AudioPlayerStateTableCompanion Value? playing, Value? loopMode, Value? shuffled, - Value>? collections}) { + Value>? collections, + Value>? tracks, + Value? currentIndex}) { return AudioPlayerStateTableCompanion( id: id ?? this.id, playing: playing ?? this.playing, loopMode: loopMode ?? this.loopMode, shuffled: shuffled ?? this.shuffled, collections: collections ?? this.collections, + tracks: tracks ?? this.tracks, + currentIndex: currentIndex ?? this.currentIndex, ); } @@ -3088,6 +2963,13 @@ class AudioPlayerStateTableCompanion .$convertercollections .toSql(collections.value)); } + if (tracks.present) { + map['tracks'] = Variable( + $AudioPlayerStateTableTable.$convertertracks.toSql(tracks.value)); + } + if (currentIndex.present) { + map['current_index'] = Variable(currentIndex.value); + } return map; } @@ -3098,557 +2980,9 @@ class AudioPlayerStateTableCompanion ..write('playing: $playing, ') ..write('loopMode: $loopMode, ') ..write('shuffled: $shuffled, ') - ..write('collections: $collections') - ..write(')')) - .toString(); - } -} - -class $PlaylistTableTable extends PlaylistTable - with TableInfo<$PlaylistTableTable, PlaylistTableData> { - @override - final GeneratedDatabase attachedDatabase; - final String? _alias; - $PlaylistTableTable(this.attachedDatabase, [this._alias]); - static const VerificationMeta _idMeta = const VerificationMeta('id'); - @override - late final GeneratedColumn id = GeneratedColumn( - 'id', aliasedName, false, - hasAutoIncrement: true, - type: DriftSqlType.int, - requiredDuringInsert: false, - defaultConstraints: - GeneratedColumn.constraintIsAlways('PRIMARY KEY AUTOINCREMENT')); - static const VerificationMeta _audioPlayerStateIdMeta = - const VerificationMeta('audioPlayerStateId'); - @override - late final GeneratedColumn audioPlayerStateId = GeneratedColumn( - 'audio_player_state_id', aliasedName, false, - type: DriftSqlType.int, - requiredDuringInsert: true, - defaultConstraints: GeneratedColumn.constraintIsAlways( - 'REFERENCES audio_player_state_table (id)')); - static const VerificationMeta _indexMeta = const VerificationMeta('index'); - @override - late final GeneratedColumn index = GeneratedColumn( - 'index', aliasedName, false, - type: DriftSqlType.int, requiredDuringInsert: true); - @override - List get $columns => [id, audioPlayerStateId, index]; - @override - String get aliasedName => _alias ?? actualTableName; - @override - String get actualTableName => $name; - static const String $name = 'playlist_table'; - @override - VerificationContext validateIntegrity(Insertable instance, - {bool isInserting = false}) { - final context = VerificationContext(); - final data = instance.toColumns(true); - if (data.containsKey('id')) { - context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); - } - if (data.containsKey('audio_player_state_id')) { - context.handle( - _audioPlayerStateIdMeta, - audioPlayerStateId.isAcceptableOrUnknown( - data['audio_player_state_id']!, _audioPlayerStateIdMeta)); - } else if (isInserting) { - context.missing(_audioPlayerStateIdMeta); - } - if (data.containsKey('index')) { - context.handle( - _indexMeta, index.isAcceptableOrUnknown(data['index']!, _indexMeta)); - } else if (isInserting) { - context.missing(_indexMeta); - } - return context; - } - - @override - Set get $primaryKey => {id}; - @override - PlaylistTableData map(Map data, {String? tablePrefix}) { - final effectivePrefix = tablePrefix != null ? '$tablePrefix.' : ''; - return PlaylistTableData( - id: attachedDatabase.typeMapping - .read(DriftSqlType.int, data['${effectivePrefix}id'])!, - audioPlayerStateId: attachedDatabase.typeMapping.read( - DriftSqlType.int, data['${effectivePrefix}audio_player_state_id'])!, - index: attachedDatabase.typeMapping - .read(DriftSqlType.int, data['${effectivePrefix}index'])!, - ); - } - - @override - $PlaylistTableTable createAlias(String alias) { - return $PlaylistTableTable(attachedDatabase, alias); - } -} - -class PlaylistTableData extends DataClass - implements Insertable { - final int id; - final int audioPlayerStateId; - final int index; - const PlaylistTableData( - {required this.id, - required this.audioPlayerStateId, - required this.index}); - @override - Map toColumns(bool nullToAbsent) { - final map = {}; - map['id'] = Variable(id); - map['audio_player_state_id'] = Variable(audioPlayerStateId); - map['index'] = Variable(index); - return map; - } - - PlaylistTableCompanion toCompanion(bool nullToAbsent) { - return PlaylistTableCompanion( - id: Value(id), - audioPlayerStateId: Value(audioPlayerStateId), - index: Value(index), - ); - } - - factory PlaylistTableData.fromJson(Map json, - {ValueSerializer? serializer}) { - serializer ??= driftRuntimeOptions.defaultSerializer; - return PlaylistTableData( - id: serializer.fromJson(json['id']), - audioPlayerStateId: serializer.fromJson(json['audioPlayerStateId']), - index: serializer.fromJson(json['index']), - ); - } - @override - Map toJson({ValueSerializer? serializer}) { - serializer ??= driftRuntimeOptions.defaultSerializer; - return { - 'id': serializer.toJson(id), - 'audioPlayerStateId': serializer.toJson(audioPlayerStateId), - 'index': serializer.toJson(index), - }; - } - - PlaylistTableData copyWith({int? id, int? audioPlayerStateId, int? index}) => - PlaylistTableData( - id: id ?? this.id, - audioPlayerStateId: audioPlayerStateId ?? this.audioPlayerStateId, - index: index ?? this.index, - ); - PlaylistTableData copyWithCompanion(PlaylistTableCompanion data) { - return PlaylistTableData( - id: data.id.present ? data.id.value : this.id, - audioPlayerStateId: data.audioPlayerStateId.present - ? data.audioPlayerStateId.value - : this.audioPlayerStateId, - index: data.index.present ? data.index.value : this.index, - ); - } - - @override - String toString() { - return (StringBuffer('PlaylistTableData(') - ..write('id: $id, ') - ..write('audioPlayerStateId: $audioPlayerStateId, ') - ..write('index: $index') - ..write(')')) - .toString(); - } - - @override - int get hashCode => Object.hash(id, audioPlayerStateId, index); - @override - bool operator ==(Object other) => - identical(this, other) || - (other is PlaylistTableData && - other.id == this.id && - other.audioPlayerStateId == this.audioPlayerStateId && - other.index == this.index); -} - -class PlaylistTableCompanion extends UpdateCompanion { - final Value id; - final Value audioPlayerStateId; - final Value index; - const PlaylistTableCompanion({ - this.id = const Value.absent(), - this.audioPlayerStateId = const Value.absent(), - this.index = const Value.absent(), - }); - PlaylistTableCompanion.insert({ - this.id = const Value.absent(), - required int audioPlayerStateId, - required int index, - }) : audioPlayerStateId = Value(audioPlayerStateId), - index = Value(index); - static Insertable custom({ - Expression? id, - Expression? audioPlayerStateId, - Expression? index, - }) { - return RawValuesInsertable({ - if (id != null) 'id': id, - if (audioPlayerStateId != null) - 'audio_player_state_id': audioPlayerStateId, - if (index != null) 'index': index, - }); - } - - PlaylistTableCompanion copyWith( - {Value? id, Value? audioPlayerStateId, Value? index}) { - return PlaylistTableCompanion( - id: id ?? this.id, - audioPlayerStateId: audioPlayerStateId ?? this.audioPlayerStateId, - index: index ?? this.index, - ); - } - - @override - Map toColumns(bool nullToAbsent) { - final map = {}; - if (id.present) { - map['id'] = Variable(id.value); - } - if (audioPlayerStateId.present) { - map['audio_player_state_id'] = Variable(audioPlayerStateId.value); - } - if (index.present) { - map['index'] = Variable(index.value); - } - return map; - } - - @override - String toString() { - return (StringBuffer('PlaylistTableCompanion(') - ..write('id: $id, ') - ..write('audioPlayerStateId: $audioPlayerStateId, ') - ..write('index: $index') - ..write(')')) - .toString(); - } -} - -class $PlaylistMediaTableTable extends PlaylistMediaTable - with TableInfo<$PlaylistMediaTableTable, PlaylistMediaTableData> { - @override - final GeneratedDatabase attachedDatabase; - final String? _alias; - $PlaylistMediaTableTable(this.attachedDatabase, [this._alias]); - static const VerificationMeta _idMeta = const VerificationMeta('id'); - @override - late final GeneratedColumn id = GeneratedColumn( - 'id', aliasedName, false, - hasAutoIncrement: true, - type: DriftSqlType.int, - requiredDuringInsert: false, - defaultConstraints: - GeneratedColumn.constraintIsAlways('PRIMARY KEY AUTOINCREMENT')); - static const VerificationMeta _playlistIdMeta = - const VerificationMeta('playlistId'); - @override - late final GeneratedColumn playlistId = GeneratedColumn( - 'playlist_id', aliasedName, false, - type: DriftSqlType.int, - requiredDuringInsert: true, - defaultConstraints: - GeneratedColumn.constraintIsAlways('REFERENCES playlist_table (id)')); - static const VerificationMeta _uriMeta = const VerificationMeta('uri'); - @override - late final GeneratedColumn uri = GeneratedColumn( - 'uri', aliasedName, false, - type: DriftSqlType.string, requiredDuringInsert: true); - static const VerificationMeta _extrasMeta = const VerificationMeta('extras'); - @override - late final GeneratedColumnWithTypeConverter?, String> - extras = GeneratedColumn('extras', aliasedName, true, - type: DriftSqlType.string, requiredDuringInsert: false) - .withConverter?>( - $PlaylistMediaTableTable.$converterextrasn); - static const VerificationMeta _httpHeadersMeta = - const VerificationMeta('httpHeaders'); - @override - late final GeneratedColumnWithTypeConverter?, String> - httpHeaders = GeneratedColumn('http_headers', aliasedName, true, - type: DriftSqlType.string, requiredDuringInsert: false) - .withConverter?>( - $PlaylistMediaTableTable.$converterhttpHeadersn); - @override - List get $columns => - [id, playlistId, uri, extras, httpHeaders]; - @override - String get aliasedName => _alias ?? actualTableName; - @override - String get actualTableName => $name; - static const String $name = 'playlist_media_table'; - @override - VerificationContext validateIntegrity( - Insertable instance, - {bool isInserting = false}) { - final context = VerificationContext(); - final data = instance.toColumns(true); - if (data.containsKey('id')) { - context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); - } - if (data.containsKey('playlist_id')) { - context.handle( - _playlistIdMeta, - playlistId.isAcceptableOrUnknown( - data['playlist_id']!, _playlistIdMeta)); - } else if (isInserting) { - context.missing(_playlistIdMeta); - } - if (data.containsKey('uri')) { - context.handle( - _uriMeta, uri.isAcceptableOrUnknown(data['uri']!, _uriMeta)); - } else if (isInserting) { - context.missing(_uriMeta); - } - context.handle(_extrasMeta, const VerificationResult.success()); - context.handle(_httpHeadersMeta, const VerificationResult.success()); - return context; - } - - @override - Set get $primaryKey => {id}; - @override - PlaylistMediaTableData map(Map data, {String? tablePrefix}) { - final effectivePrefix = tablePrefix != null ? '$tablePrefix.' : ''; - return PlaylistMediaTableData( - id: attachedDatabase.typeMapping - .read(DriftSqlType.int, data['${effectivePrefix}id'])!, - playlistId: attachedDatabase.typeMapping - .read(DriftSqlType.int, data['${effectivePrefix}playlist_id'])!, - uri: attachedDatabase.typeMapping - .read(DriftSqlType.string, data['${effectivePrefix}uri'])!, - extras: $PlaylistMediaTableTable.$converterextrasn.fromSql( - attachedDatabase.typeMapping - .read(DriftSqlType.string, data['${effectivePrefix}extras'])), - httpHeaders: $PlaylistMediaTableTable.$converterhttpHeadersn.fromSql( - attachedDatabase.typeMapping.read( - DriftSqlType.string, data['${effectivePrefix}http_headers'])), - ); - } - - @override - $PlaylistMediaTableTable createAlias(String alias) { - return $PlaylistMediaTableTable(attachedDatabase, alias); - } - - static TypeConverter, String> $converterextras = - const MapTypeConverter(); - static TypeConverter?, String?> $converterextrasn = - NullAwareTypeConverter.wrap($converterextras); - static TypeConverter, String> $converterhttpHeaders = - const MapTypeConverter(); - static TypeConverter?, String?> $converterhttpHeadersn = - NullAwareTypeConverter.wrap($converterhttpHeaders); -} - -class PlaylistMediaTableData extends DataClass - implements Insertable { - final int id; - final int playlistId; - final String uri; - final Map? extras; - final Map? httpHeaders; - const PlaylistMediaTableData( - {required this.id, - required this.playlistId, - required this.uri, - this.extras, - this.httpHeaders}); - @override - Map toColumns(bool nullToAbsent) { - final map = {}; - map['id'] = Variable(id); - map['playlist_id'] = Variable(playlistId); - map['uri'] = Variable(uri); - if (!nullToAbsent || extras != null) { - map['extras'] = Variable( - $PlaylistMediaTableTable.$converterextrasn.toSql(extras)); - } - if (!nullToAbsent || httpHeaders != null) { - map['http_headers'] = Variable( - $PlaylistMediaTableTable.$converterhttpHeadersn.toSql(httpHeaders)); - } - return map; - } - - PlaylistMediaTableCompanion toCompanion(bool nullToAbsent) { - return PlaylistMediaTableCompanion( - id: Value(id), - playlistId: Value(playlistId), - uri: Value(uri), - extras: - extras == null && nullToAbsent ? const Value.absent() : Value(extras), - httpHeaders: httpHeaders == null && nullToAbsent - ? const Value.absent() - : Value(httpHeaders), - ); - } - - factory PlaylistMediaTableData.fromJson(Map json, - {ValueSerializer? serializer}) { - serializer ??= driftRuntimeOptions.defaultSerializer; - return PlaylistMediaTableData( - id: serializer.fromJson(json['id']), - playlistId: serializer.fromJson(json['playlistId']), - uri: serializer.fromJson(json['uri']), - extras: serializer.fromJson?>(json['extras']), - httpHeaders: - serializer.fromJson?>(json['httpHeaders']), - ); - } - @override - Map toJson({ValueSerializer? serializer}) { - serializer ??= driftRuntimeOptions.defaultSerializer; - return { - 'id': serializer.toJson(id), - 'playlistId': serializer.toJson(playlistId), - 'uri': serializer.toJson(uri), - 'extras': serializer.toJson?>(extras), - 'httpHeaders': serializer.toJson?>(httpHeaders), - }; - } - - PlaylistMediaTableData copyWith( - {int? id, - int? playlistId, - String? uri, - Value?> extras = const Value.absent(), - Value?> httpHeaders = const Value.absent()}) => - PlaylistMediaTableData( - id: id ?? this.id, - playlistId: playlistId ?? this.playlistId, - uri: uri ?? this.uri, - extras: extras.present ? extras.value : this.extras, - httpHeaders: httpHeaders.present ? httpHeaders.value : this.httpHeaders, - ); - PlaylistMediaTableData copyWithCompanion(PlaylistMediaTableCompanion data) { - return PlaylistMediaTableData( - id: data.id.present ? data.id.value : this.id, - playlistId: - data.playlistId.present ? data.playlistId.value : this.playlistId, - uri: data.uri.present ? data.uri.value : this.uri, - extras: data.extras.present ? data.extras.value : this.extras, - httpHeaders: - data.httpHeaders.present ? data.httpHeaders.value : this.httpHeaders, - ); - } - - @override - String toString() { - return (StringBuffer('PlaylistMediaTableData(') - ..write('id: $id, ') - ..write('playlistId: $playlistId, ') - ..write('uri: $uri, ') - ..write('extras: $extras, ') - ..write('httpHeaders: $httpHeaders') - ..write(')')) - .toString(); - } - - @override - int get hashCode => Object.hash(id, playlistId, uri, extras, httpHeaders); - @override - bool operator ==(Object other) => - identical(this, other) || - (other is PlaylistMediaTableData && - other.id == this.id && - other.playlistId == this.playlistId && - other.uri == this.uri && - other.extras == this.extras && - other.httpHeaders == this.httpHeaders); -} - -class PlaylistMediaTableCompanion - extends UpdateCompanion { - final Value id; - final Value playlistId; - final Value uri; - final Value?> extras; - final Value?> httpHeaders; - const PlaylistMediaTableCompanion({ - this.id = const Value.absent(), - this.playlistId = const Value.absent(), - this.uri = const Value.absent(), - this.extras = const Value.absent(), - this.httpHeaders = const Value.absent(), - }); - PlaylistMediaTableCompanion.insert({ - this.id = const Value.absent(), - required int playlistId, - required String uri, - this.extras = const Value.absent(), - this.httpHeaders = const Value.absent(), - }) : playlistId = Value(playlistId), - uri = Value(uri); - static Insertable custom({ - Expression? id, - Expression? playlistId, - Expression? uri, - Expression? extras, - Expression? httpHeaders, - }) { - return RawValuesInsertable({ - if (id != null) 'id': id, - if (playlistId != null) 'playlist_id': playlistId, - if (uri != null) 'uri': uri, - if (extras != null) 'extras': extras, - if (httpHeaders != null) 'http_headers': httpHeaders, - }); - } - - PlaylistMediaTableCompanion copyWith( - {Value? id, - Value? playlistId, - Value? uri, - Value?>? extras, - Value?>? httpHeaders}) { - return PlaylistMediaTableCompanion( - id: id ?? this.id, - playlistId: playlistId ?? this.playlistId, - uri: uri ?? this.uri, - extras: extras ?? this.extras, - httpHeaders: httpHeaders ?? this.httpHeaders, - ); - } - - @override - Map toColumns(bool nullToAbsent) { - final map = {}; - if (id.present) { - map['id'] = Variable(id.value); - } - if (playlistId.present) { - map['playlist_id'] = Variable(playlistId.value); - } - if (uri.present) { - map['uri'] = Variable(uri.value); - } - if (extras.present) { - map['extras'] = Variable( - $PlaylistMediaTableTable.$converterextrasn.toSql(extras.value)); - } - if (httpHeaders.present) { - map['http_headers'] = Variable($PlaylistMediaTableTable - .$converterhttpHeadersn - .toSql(httpHeaders.value)); - } - return map; - } - - @override - String toString() { - return (StringBuffer('PlaylistMediaTableCompanion(') - ..write('id: $id, ') - ..write('playlistId: $playlistId, ') - ..write('uri: $uri, ') - ..write('extras: $extras, ') - ..write('httpHeaders: $httpHeaders') + ..write('collections: $collections, ') + ..write('tracks: $tracks, ') + ..write('currentIndex: $currentIndex') ..write(')')) .toString(); } @@ -3677,7 +3011,6 @@ class $HistoryTableTable extends HistoryTable type: DriftSqlType.dateTime, requiredDuringInsert: false, defaultValue: currentDateAndTime); - static const VerificationMeta _typeMeta = const VerificationMeta('type'); @override late final GeneratedColumnWithTypeConverter type = GeneratedColumn('type', aliasedName, false, @@ -3688,7 +3021,6 @@ class $HistoryTableTable extends HistoryTable late final GeneratedColumn itemId = GeneratedColumn( 'item_id', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: true); - static const VerificationMeta _dataMeta = const VerificationMeta('data'); @override late final GeneratedColumnWithTypeConverter, String> data = GeneratedColumn('data', aliasedName, false, @@ -3714,14 +3046,12 @@ class $HistoryTableTable extends HistoryTable context.handle(_createdAtMeta, createdAt.isAcceptableOrUnknown(data['created_at']!, _createdAtMeta)); } - context.handle(_typeMeta, const VerificationResult.success()); if (data.containsKey('item_id')) { context.handle(_itemIdMeta, itemId.isAcceptableOrUnknown(data['item_id']!, _itemIdMeta)); } else if (isInserting) { context.missing(_itemIdMeta); } - context.handle(_dataMeta, const VerificationResult.success()); return context; } @@ -3980,7 +3310,6 @@ class $LyricsTableTable extends LyricsTable late final GeneratedColumn trackId = GeneratedColumn( 'track_id', aliasedName, false, type: DriftSqlType.string, requiredDuringInsert: true); - static const VerificationMeta _dataMeta = const VerificationMeta('data'); @override late final GeneratedColumnWithTypeConverter data = GeneratedColumn('data', aliasedName, false, @@ -4007,7 +3336,6 @@ class $LyricsTableTable extends LyricsTable } else if (isInserting) { context.missing(_trackIdMeta); } - context.handle(_dataMeta, const VerificationResult.success()); return context; } @@ -4179,6 +3507,622 @@ class LyricsTableCompanion extends UpdateCompanion { } } +class $PluginsTableTable extends PluginsTable + with TableInfo<$PluginsTableTable, PluginsTableData> { + @override + final GeneratedDatabase attachedDatabase; + final String? _alias; + $PluginsTableTable(this.attachedDatabase, [this._alias]); + static const VerificationMeta _idMeta = const VerificationMeta('id'); + @override + late final GeneratedColumn id = GeneratedColumn( + 'id', aliasedName, false, + hasAutoIncrement: true, + type: DriftSqlType.int, + requiredDuringInsert: false, + defaultConstraints: + GeneratedColumn.constraintIsAlways('PRIMARY KEY AUTOINCREMENT')); + static const VerificationMeta _nameMeta = const VerificationMeta('name'); + @override + late final GeneratedColumn name = GeneratedColumn( + 'name', aliasedName, false, + additionalChecks: + GeneratedColumn.checkTextLength(minTextLength: 1, maxTextLength: 50), + type: DriftSqlType.string, + requiredDuringInsert: true); + static const VerificationMeta _descriptionMeta = + const VerificationMeta('description'); + @override + late final GeneratedColumn description = GeneratedColumn( + 'description', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true); + static const VerificationMeta _versionMeta = + const VerificationMeta('version'); + @override + late final GeneratedColumn version = GeneratedColumn( + 'version', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true); + static const VerificationMeta _authorMeta = const VerificationMeta('author'); + @override + late final GeneratedColumn author = GeneratedColumn( + 'author', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true); + static const VerificationMeta _entryPointMeta = + const VerificationMeta('entryPoint'); + @override + late final GeneratedColumn entryPoint = GeneratedColumn( + 'entry_point', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true); + @override + late final GeneratedColumnWithTypeConverter, String> apis = + GeneratedColumn('apis', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true) + .withConverter>($PluginsTableTable.$converterapis); + @override + late final GeneratedColumnWithTypeConverter, String> abilities = + GeneratedColumn('abilities', aliasedName, false, + type: DriftSqlType.string, requiredDuringInsert: true) + .withConverter>($PluginsTableTable.$converterabilities); + static const VerificationMeta _selectedForMetadataMeta = + const VerificationMeta('selectedForMetadata'); + @override + late final GeneratedColumn selectedForMetadata = GeneratedColumn( + 'selected_for_metadata', aliasedName, false, + type: DriftSqlType.bool, + requiredDuringInsert: false, + defaultConstraints: GeneratedColumn.constraintIsAlways( + 'CHECK ("selected_for_metadata" IN (0, 1))'), + defaultValue: const Constant(false)); + static const VerificationMeta _selectedForAudioSourceMeta = + const VerificationMeta('selectedForAudioSource'); + @override + late final GeneratedColumn selectedForAudioSource = + GeneratedColumn('selected_for_audio_source', aliasedName, false, + type: DriftSqlType.bool, + requiredDuringInsert: false, + defaultConstraints: GeneratedColumn.constraintIsAlways( + 'CHECK ("selected_for_audio_source" IN (0, 1))'), + defaultValue: const Constant(false)); + static const VerificationMeta _repositoryMeta = + const VerificationMeta('repository'); + @override + late final GeneratedColumn repository = GeneratedColumn( + 'repository', aliasedName, true, + type: DriftSqlType.string, requiredDuringInsert: false); + static const VerificationMeta _pluginApiVersionMeta = + const VerificationMeta('pluginApiVersion'); + @override + late final GeneratedColumn pluginApiVersion = GeneratedColumn( + 'plugin_api_version', aliasedName, false, + type: DriftSqlType.string, + requiredDuringInsert: false, + defaultValue: const Constant('2.0.0')); + @override + List get $columns => [ + id, + name, + description, + version, + author, + entryPoint, + apis, + abilities, + selectedForMetadata, + selectedForAudioSource, + repository, + pluginApiVersion + ]; + @override + String get aliasedName => _alias ?? actualTableName; + @override + String get actualTableName => $name; + static const String $name = 'plugins_table'; + @override + VerificationContext validateIntegrity(Insertable instance, + {bool isInserting = false}) { + final context = VerificationContext(); + final data = instance.toColumns(true); + if (data.containsKey('id')) { + context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); + } + if (data.containsKey('name')) { + context.handle( + _nameMeta, name.isAcceptableOrUnknown(data['name']!, _nameMeta)); + } else if (isInserting) { + context.missing(_nameMeta); + } + if (data.containsKey('description')) { + context.handle( + _descriptionMeta, + description.isAcceptableOrUnknown( + data['description']!, _descriptionMeta)); + } else if (isInserting) { + context.missing(_descriptionMeta); + } + if (data.containsKey('version')) { + context.handle(_versionMeta, + version.isAcceptableOrUnknown(data['version']!, _versionMeta)); + } else if (isInserting) { + context.missing(_versionMeta); + } + if (data.containsKey('author')) { + context.handle(_authorMeta, + author.isAcceptableOrUnknown(data['author']!, _authorMeta)); + } else if (isInserting) { + context.missing(_authorMeta); + } + if (data.containsKey('entry_point')) { + context.handle( + _entryPointMeta, + entryPoint.isAcceptableOrUnknown( + data['entry_point']!, _entryPointMeta)); + } else if (isInserting) { + context.missing(_entryPointMeta); + } + if (data.containsKey('selected_for_metadata')) { + context.handle( + _selectedForMetadataMeta, + selectedForMetadata.isAcceptableOrUnknown( + data['selected_for_metadata']!, _selectedForMetadataMeta)); + } + if (data.containsKey('selected_for_audio_source')) { + context.handle( + _selectedForAudioSourceMeta, + selectedForAudioSource.isAcceptableOrUnknown( + data['selected_for_audio_source']!, _selectedForAudioSourceMeta)); + } + if (data.containsKey('repository')) { + context.handle( + _repositoryMeta, + repository.isAcceptableOrUnknown( + data['repository']!, _repositoryMeta)); + } + if (data.containsKey('plugin_api_version')) { + context.handle( + _pluginApiVersionMeta, + pluginApiVersion.isAcceptableOrUnknown( + data['plugin_api_version']!, _pluginApiVersionMeta)); + } + return context; + } + + @override + Set get $primaryKey => {id}; + @override + PluginsTableData map(Map data, {String? tablePrefix}) { + final effectivePrefix = tablePrefix != null ? '$tablePrefix.' : ''; + return PluginsTableData( + id: attachedDatabase.typeMapping + .read(DriftSqlType.int, data['${effectivePrefix}id'])!, + name: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}name'])!, + description: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}description'])!, + version: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}version'])!, + author: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}author'])!, + entryPoint: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}entry_point'])!, + apis: $PluginsTableTable.$converterapis.fromSql(attachedDatabase + .typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}apis'])!), + abilities: $PluginsTableTable.$converterabilities.fromSql(attachedDatabase + .typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}abilities'])!), + selectedForMetadata: attachedDatabase.typeMapping.read( + DriftSqlType.bool, data['${effectivePrefix}selected_for_metadata'])!, + selectedForAudioSource: attachedDatabase.typeMapping.read( + DriftSqlType.bool, + data['${effectivePrefix}selected_for_audio_source'])!, + repository: attachedDatabase.typeMapping + .read(DriftSqlType.string, data['${effectivePrefix}repository']), + pluginApiVersion: attachedDatabase.typeMapping.read( + DriftSqlType.string, data['${effectivePrefix}plugin_api_version'])!, + ); + } + + @override + $PluginsTableTable createAlias(String alias) { + return $PluginsTableTable(attachedDatabase, alias); + } + + static TypeConverter, String> $converterapis = + const StringListConverter(); + static TypeConverter, String> $converterabilities = + const StringListConverter(); +} + +class PluginsTableData extends DataClass + implements Insertable { + final int id; + final String name; + final String description; + final String version; + final String author; + final String entryPoint; + final List apis; + final List abilities; + final bool selectedForMetadata; + final bool selectedForAudioSource; + final String? repository; + final String pluginApiVersion; + const PluginsTableData( + {required this.id, + required this.name, + required this.description, + required this.version, + required this.author, + required this.entryPoint, + required this.apis, + required this.abilities, + required this.selectedForMetadata, + required this.selectedForAudioSource, + this.repository, + required this.pluginApiVersion}); + @override + Map toColumns(bool nullToAbsent) { + final map = {}; + map['id'] = Variable(id); + map['name'] = Variable(name); + map['description'] = Variable(description); + map['version'] = Variable(version); + map['author'] = Variable(author); + map['entry_point'] = Variable(entryPoint); + { + map['apis'] = + Variable($PluginsTableTable.$converterapis.toSql(apis)); + } + { + map['abilities'] = Variable( + $PluginsTableTable.$converterabilities.toSql(abilities)); + } + map['selected_for_metadata'] = Variable(selectedForMetadata); + map['selected_for_audio_source'] = Variable(selectedForAudioSource); + if (!nullToAbsent || repository != null) { + map['repository'] = Variable(repository); + } + map['plugin_api_version'] = Variable(pluginApiVersion); + return map; + } + + PluginsTableCompanion toCompanion(bool nullToAbsent) { + return PluginsTableCompanion( + id: Value(id), + name: Value(name), + description: Value(description), + version: Value(version), + author: Value(author), + entryPoint: Value(entryPoint), + apis: Value(apis), + abilities: Value(abilities), + selectedForMetadata: Value(selectedForMetadata), + selectedForAudioSource: Value(selectedForAudioSource), + repository: repository == null && nullToAbsent + ? const Value.absent() + : Value(repository), + pluginApiVersion: Value(pluginApiVersion), + ); + } + + factory PluginsTableData.fromJson(Map json, + {ValueSerializer? serializer}) { + serializer ??= driftRuntimeOptions.defaultSerializer; + return PluginsTableData( + id: serializer.fromJson(json['id']), + name: serializer.fromJson(json['name']), + description: serializer.fromJson(json['description']), + version: serializer.fromJson(json['version']), + author: serializer.fromJson(json['author']), + entryPoint: serializer.fromJson(json['entryPoint']), + apis: serializer.fromJson>(json['apis']), + abilities: serializer.fromJson>(json['abilities']), + selectedForMetadata: + serializer.fromJson(json['selectedForMetadata']), + selectedForAudioSource: + serializer.fromJson(json['selectedForAudioSource']), + repository: serializer.fromJson(json['repository']), + pluginApiVersion: serializer.fromJson(json['pluginApiVersion']), + ); + } + @override + Map toJson({ValueSerializer? serializer}) { + serializer ??= driftRuntimeOptions.defaultSerializer; + return { + 'id': serializer.toJson(id), + 'name': serializer.toJson(name), + 'description': serializer.toJson(description), + 'version': serializer.toJson(version), + 'author': serializer.toJson(author), + 'entryPoint': serializer.toJson(entryPoint), + 'apis': serializer.toJson>(apis), + 'abilities': serializer.toJson>(abilities), + 'selectedForMetadata': serializer.toJson(selectedForMetadata), + 'selectedForAudioSource': serializer.toJson(selectedForAudioSource), + 'repository': serializer.toJson(repository), + 'pluginApiVersion': serializer.toJson(pluginApiVersion), + }; + } + + PluginsTableData copyWith( + {int? id, + String? name, + String? description, + String? version, + String? author, + String? entryPoint, + List? apis, + List? abilities, + bool? selectedForMetadata, + bool? selectedForAudioSource, + Value repository = const Value.absent(), + String? pluginApiVersion}) => + PluginsTableData( + id: id ?? this.id, + name: name ?? this.name, + description: description ?? this.description, + version: version ?? this.version, + author: author ?? this.author, + entryPoint: entryPoint ?? this.entryPoint, + apis: apis ?? this.apis, + abilities: abilities ?? this.abilities, + selectedForMetadata: selectedForMetadata ?? this.selectedForMetadata, + selectedForAudioSource: + selectedForAudioSource ?? this.selectedForAudioSource, + repository: repository.present ? repository.value : this.repository, + pluginApiVersion: pluginApiVersion ?? this.pluginApiVersion, + ); + PluginsTableData copyWithCompanion(PluginsTableCompanion data) { + return PluginsTableData( + id: data.id.present ? data.id.value : this.id, + name: data.name.present ? data.name.value : this.name, + description: + data.description.present ? data.description.value : this.description, + version: data.version.present ? data.version.value : this.version, + author: data.author.present ? data.author.value : this.author, + entryPoint: + data.entryPoint.present ? data.entryPoint.value : this.entryPoint, + apis: data.apis.present ? data.apis.value : this.apis, + abilities: data.abilities.present ? data.abilities.value : this.abilities, + selectedForMetadata: data.selectedForMetadata.present + ? data.selectedForMetadata.value + : this.selectedForMetadata, + selectedForAudioSource: data.selectedForAudioSource.present + ? data.selectedForAudioSource.value + : this.selectedForAudioSource, + repository: + data.repository.present ? data.repository.value : this.repository, + pluginApiVersion: data.pluginApiVersion.present + ? data.pluginApiVersion.value + : this.pluginApiVersion, + ); + } + + @override + String toString() { + return (StringBuffer('PluginsTableData(') + ..write('id: $id, ') + ..write('name: $name, ') + ..write('description: $description, ') + ..write('version: $version, ') + ..write('author: $author, ') + ..write('entryPoint: $entryPoint, ') + ..write('apis: $apis, ') + ..write('abilities: $abilities, ') + ..write('selectedForMetadata: $selectedForMetadata, ') + ..write('selectedForAudioSource: $selectedForAudioSource, ') + ..write('repository: $repository, ') + ..write('pluginApiVersion: $pluginApiVersion') + ..write(')')) + .toString(); + } + + @override + int get hashCode => Object.hash( + id, + name, + description, + version, + author, + entryPoint, + apis, + abilities, + selectedForMetadata, + selectedForAudioSource, + repository, + pluginApiVersion); + @override + bool operator ==(Object other) => + identical(this, other) || + (other is PluginsTableData && + other.id == this.id && + other.name == this.name && + other.description == this.description && + other.version == this.version && + other.author == this.author && + other.entryPoint == this.entryPoint && + other.apis == this.apis && + other.abilities == this.abilities && + other.selectedForMetadata == this.selectedForMetadata && + other.selectedForAudioSource == this.selectedForAudioSource && + other.repository == this.repository && + other.pluginApiVersion == this.pluginApiVersion); +} + +class PluginsTableCompanion extends UpdateCompanion { + final Value id; + final Value name; + final Value description; + final Value version; + final Value author; + final Value entryPoint; + final Value> apis; + final Value> abilities; + final Value selectedForMetadata; + final Value selectedForAudioSource; + final Value repository; + final Value pluginApiVersion; + const PluginsTableCompanion({ + this.id = const Value.absent(), + this.name = const Value.absent(), + this.description = const Value.absent(), + this.version = const Value.absent(), + this.author = const Value.absent(), + this.entryPoint = const Value.absent(), + this.apis = const Value.absent(), + this.abilities = const Value.absent(), + this.selectedForMetadata = const Value.absent(), + this.selectedForAudioSource = const Value.absent(), + this.repository = const Value.absent(), + this.pluginApiVersion = const Value.absent(), + }); + PluginsTableCompanion.insert({ + this.id = const Value.absent(), + required String name, + required String description, + required String version, + required String author, + required String entryPoint, + required List apis, + required List abilities, + this.selectedForMetadata = const Value.absent(), + this.selectedForAudioSource = const Value.absent(), + this.repository = const Value.absent(), + this.pluginApiVersion = const Value.absent(), + }) : name = Value(name), + description = Value(description), + version = Value(version), + author = Value(author), + entryPoint = Value(entryPoint), + apis = Value(apis), + abilities = Value(abilities); + static Insertable custom({ + Expression? id, + Expression? name, + Expression? description, + Expression? version, + Expression? author, + Expression? entryPoint, + Expression? apis, + Expression? abilities, + Expression? selectedForMetadata, + Expression? selectedForAudioSource, + Expression? repository, + Expression? pluginApiVersion, + }) { + return RawValuesInsertable({ + if (id != null) 'id': id, + if (name != null) 'name': name, + if (description != null) 'description': description, + if (version != null) 'version': version, + if (author != null) 'author': author, + if (entryPoint != null) 'entry_point': entryPoint, + if (apis != null) 'apis': apis, + if (abilities != null) 'abilities': abilities, + if (selectedForMetadata != null) + 'selected_for_metadata': selectedForMetadata, + if (selectedForAudioSource != null) + 'selected_for_audio_source': selectedForAudioSource, + if (repository != null) 'repository': repository, + if (pluginApiVersion != null) 'plugin_api_version': pluginApiVersion, + }); + } + + PluginsTableCompanion copyWith( + {Value? id, + Value? name, + Value? description, + Value? version, + Value? author, + Value? entryPoint, + Value>? apis, + Value>? abilities, + Value? selectedForMetadata, + Value? selectedForAudioSource, + Value? repository, + Value? pluginApiVersion}) { + return PluginsTableCompanion( + id: id ?? this.id, + name: name ?? this.name, + description: description ?? this.description, + version: version ?? this.version, + author: author ?? this.author, + entryPoint: entryPoint ?? this.entryPoint, + apis: apis ?? this.apis, + abilities: abilities ?? this.abilities, + selectedForMetadata: selectedForMetadata ?? this.selectedForMetadata, + selectedForAudioSource: + selectedForAudioSource ?? this.selectedForAudioSource, + repository: repository ?? this.repository, + pluginApiVersion: pluginApiVersion ?? this.pluginApiVersion, + ); + } + + @override + Map toColumns(bool nullToAbsent) { + final map = {}; + if (id.present) { + map['id'] = Variable(id.value); + } + if (name.present) { + map['name'] = Variable(name.value); + } + if (description.present) { + map['description'] = Variable(description.value); + } + if (version.present) { + map['version'] = Variable(version.value); + } + if (author.present) { + map['author'] = Variable(author.value); + } + if (entryPoint.present) { + map['entry_point'] = Variable(entryPoint.value); + } + if (apis.present) { + map['apis'] = + Variable($PluginsTableTable.$converterapis.toSql(apis.value)); + } + if (abilities.present) { + map['abilities'] = Variable( + $PluginsTableTable.$converterabilities.toSql(abilities.value)); + } + if (selectedForMetadata.present) { + map['selected_for_metadata'] = Variable(selectedForMetadata.value); + } + if (selectedForAudioSource.present) { + map['selected_for_audio_source'] = + Variable(selectedForAudioSource.value); + } + if (repository.present) { + map['repository'] = Variable(repository.value); + } + if (pluginApiVersion.present) { + map['plugin_api_version'] = Variable(pluginApiVersion.value); + } + return map; + } + + @override + String toString() { + return (StringBuffer('PluginsTableCompanion(') + ..write('id: $id, ') + ..write('name: $name, ') + ..write('description: $description, ') + ..write('version: $version, ') + ..write('author: $author, ') + ..write('entryPoint: $entryPoint, ') + ..write('apis: $apis, ') + ..write('abilities: $abilities, ') + ..write('selectedForMetadata: $selectedForMetadata, ') + ..write('selectedForAudioSource: $selectedForAudioSource, ') + ..write('repository: $repository, ') + ..write('pluginApiVersion: $pluginApiVersion') + ..write(')')) + .toString(); + } +} + abstract class _$AppDatabase extends GeneratedDatabase { _$AppDatabase(QueryExecutor e) : super(e); $AppDatabaseManager get managers => $AppDatabaseManager(this); @@ -4194,15 +4138,11 @@ abstract class _$AppDatabase extends GeneratedDatabase { $SourceMatchTableTable(this); late final $AudioPlayerStateTableTable audioPlayerStateTable = $AudioPlayerStateTableTable(this); - late final $PlaylistTableTable playlistTable = $PlaylistTableTable(this); - late final $PlaylistMediaTableTable playlistMediaTable = - $PlaylistMediaTableTable(this); late final $HistoryTableTable historyTable = $HistoryTableTable(this); late final $LyricsTableTable lyricsTable = $LyricsTableTable(this); + late final $PluginsTableTable pluginsTable = $PluginsTableTable(this); late final Index uniqueBlacklist = Index('unique_blacklist', 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); - late final Index uniqTrackMatch = Index('uniq_track_match', - 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); @override Iterable> get allTables => allSchemaEntities.whereType>(); @@ -4215,12 +4155,10 @@ abstract class _$AppDatabase extends GeneratedDatabase { skipSegmentTable, sourceMatchTable, audioPlayerStateTable, - playlistTable, - playlistMediaTable, historyTable, lyricsTable, - uniqueBlacklist, - uniqTrackMatch + pluginsTable, + uniqueBlacklist ]; } @@ -4545,7 +4483,6 @@ typedef $$BlacklistTableTableProcessedTableManager = ProcessedTableManager< typedef $$PreferencesTableTableCreateCompanionBuilder = PreferencesTableCompanion Function({ Value id, - Value audioQuality, Value albumColorSync, Value amoledDarkTheme, Value checkUpdate, @@ -4561,21 +4498,18 @@ typedef $$PreferencesTableTableCreateCompanionBuilder Value searchMode, Value downloadLocation, Value> localLibraryLocation, - Value pipedInstance, - Value invidiousInstance, Value themeMode, - Value audioSource, - Value streamMusicCodec, - Value downloadMusicCodec, + Value audioSourceId, + Value youtubeClientEngine, Value discordPresence, Value endlessPlayback, Value enableConnect, + Value connectPort, Value cacheMusic, }); typedef $$PreferencesTableTableUpdateCompanionBuilder = PreferencesTableCompanion Function({ Value id, - Value audioQuality, Value albumColorSync, Value amoledDarkTheme, Value checkUpdate, @@ -4591,15 +4525,13 @@ typedef $$PreferencesTableTableUpdateCompanionBuilder Value searchMode, Value downloadLocation, Value> localLibraryLocation, - Value pipedInstance, - Value invidiousInstance, Value themeMode, - Value audioSource, - Value streamMusicCodec, - Value downloadMusicCodec, + Value audioSourceId, + Value youtubeClientEngine, Value discordPresence, Value endlessPlayback, Value enableConnect, + Value connectPort, Value cacheMusic, }); @@ -4615,11 +4547,6 @@ class $$PreferencesTableTableFilterComposer ColumnFilters get id => $composableBuilder( column: $table.id, builder: (column) => ColumnFilters(column)); - ColumnWithTypeConverterFilters - get audioQuality => $composableBuilder( - column: $table.audioQuality, - builder: (column) => ColumnWithTypeConverterFilters(column)); - ColumnFilters get albumColorSync => $composableBuilder( column: $table.albumColorSync, builder: (column) => ColumnFilters(column)); @@ -4685,31 +4612,18 @@ class $$PreferencesTableTableFilterComposer column: $table.localLibraryLocation, builder: (column) => ColumnWithTypeConverterFilters(column)); - ColumnFilters get pipedInstance => $composableBuilder( - column: $table.pipedInstance, builder: (column) => ColumnFilters(column)); - - ColumnFilters get invidiousInstance => $composableBuilder( - column: $table.invidiousInstance, - builder: (column) => ColumnFilters(column)); - ColumnWithTypeConverterFilters get themeMode => $composableBuilder( column: $table.themeMode, builder: (column) => ColumnWithTypeConverterFilters(column)); - ColumnWithTypeConverterFilters - get audioSource => $composableBuilder( - column: $table.audioSource, - builder: (column) => ColumnWithTypeConverterFilters(column)); + ColumnFilters get audioSourceId => $composableBuilder( + column: $table.audioSourceId, builder: (column) => ColumnFilters(column)); - ColumnWithTypeConverterFilters - get streamMusicCodec => $composableBuilder( - column: $table.streamMusicCodec, - builder: (column) => ColumnWithTypeConverterFilters(column)); - - ColumnWithTypeConverterFilters - get downloadMusicCodec => $composableBuilder( - column: $table.downloadMusicCodec, + ColumnWithTypeConverterFilters + get youtubeClientEngine => $composableBuilder( + column: $table.youtubeClientEngine, builder: (column) => ColumnWithTypeConverterFilters(column)); ColumnFilters get discordPresence => $composableBuilder( @@ -4723,6 +4637,9 @@ class $$PreferencesTableTableFilterComposer ColumnFilters get enableConnect => $composableBuilder( column: $table.enableConnect, builder: (column) => ColumnFilters(column)); + ColumnFilters get connectPort => $composableBuilder( + column: $table.connectPort, builder: (column) => ColumnFilters(column)); + ColumnFilters get cacheMusic => $composableBuilder( column: $table.cacheMusic, builder: (column) => ColumnFilters(column)); } @@ -4739,10 +4656,6 @@ class $$PreferencesTableTableOrderingComposer ColumnOrderings get id => $composableBuilder( column: $table.id, builder: (column) => ColumnOrderings(column)); - ColumnOrderings get audioQuality => $composableBuilder( - column: $table.audioQuality, - builder: (column) => ColumnOrderings(column)); - ColumnOrderings get albumColorSync => $composableBuilder( column: $table.albumColorSync, builder: (column) => ColumnOrderings(column)); @@ -4798,26 +4711,15 @@ class $$PreferencesTableTableOrderingComposer column: $table.localLibraryLocation, builder: (column) => ColumnOrderings(column)); - ColumnOrderings get pipedInstance => $composableBuilder( - column: $table.pipedInstance, - builder: (column) => ColumnOrderings(column)); - - ColumnOrderings get invidiousInstance => $composableBuilder( - column: $table.invidiousInstance, - builder: (column) => ColumnOrderings(column)); - ColumnOrderings get themeMode => $composableBuilder( column: $table.themeMode, builder: (column) => ColumnOrderings(column)); - ColumnOrderings get audioSource => $composableBuilder( - column: $table.audioSource, builder: (column) => ColumnOrderings(column)); - - ColumnOrderings get streamMusicCodec => $composableBuilder( - column: $table.streamMusicCodec, + ColumnOrderings get audioSourceId => $composableBuilder( + column: $table.audioSourceId, builder: (column) => ColumnOrderings(column)); - ColumnOrderings get downloadMusicCodec => $composableBuilder( - column: $table.downloadMusicCodec, + ColumnOrderings get youtubeClientEngine => $composableBuilder( + column: $table.youtubeClientEngine, builder: (column) => ColumnOrderings(column)); ColumnOrderings get discordPresence => $composableBuilder( @@ -4832,6 +4734,9 @@ class $$PreferencesTableTableOrderingComposer column: $table.enableConnect, builder: (column) => ColumnOrderings(column)); + ColumnOrderings get connectPort => $composableBuilder( + column: $table.connectPort, builder: (column) => ColumnOrderings(column)); + ColumnOrderings get cacheMusic => $composableBuilder( column: $table.cacheMusic, builder: (column) => ColumnOrderings(column)); } @@ -4848,10 +4753,6 @@ class $$PreferencesTableTableAnnotationComposer GeneratedColumn get id => $composableBuilder(column: $table.id, builder: (column) => column); - GeneratedColumnWithTypeConverter get audioQuality => - $composableBuilder( - column: $table.audioQuality, builder: (column) => column); - GeneratedColumn get albumColorSync => $composableBuilder( column: $table.albumColorSync, builder: (column) => column); @@ -4902,26 +4803,15 @@ class $$PreferencesTableTableAnnotationComposer get localLibraryLocation => $composableBuilder( column: $table.localLibraryLocation, builder: (column) => column); - GeneratedColumn get pipedInstance => $composableBuilder( - column: $table.pipedInstance, builder: (column) => column); - - GeneratedColumn get invidiousInstance => $composableBuilder( - column: $table.invidiousInstance, builder: (column) => column); - GeneratedColumnWithTypeConverter get themeMode => $composableBuilder(column: $table.themeMode, builder: (column) => column); - GeneratedColumnWithTypeConverter get audioSource => - $composableBuilder( - column: $table.audioSource, builder: (column) => column); + GeneratedColumn get audioSourceId => $composableBuilder( + column: $table.audioSourceId, builder: (column) => column); - GeneratedColumnWithTypeConverter get streamMusicCodec => - $composableBuilder( - column: $table.streamMusicCodec, builder: (column) => column); - - GeneratedColumnWithTypeConverter - get downloadMusicCodec => $composableBuilder( - column: $table.downloadMusicCodec, builder: (column) => column); + GeneratedColumnWithTypeConverter + get youtubeClientEngine => $composableBuilder( + column: $table.youtubeClientEngine, builder: (column) => column); GeneratedColumn get discordPresence => $composableBuilder( column: $table.discordPresence, builder: (column) => column); @@ -4932,6 +4822,9 @@ class $$PreferencesTableTableAnnotationComposer GeneratedColumn get enableConnect => $composableBuilder( column: $table.enableConnect, builder: (column) => column); + GeneratedColumn get connectPort => $composableBuilder( + column: $table.connectPort, builder: (column) => column); + GeneratedColumn get cacheMusic => $composableBuilder( column: $table.cacheMusic, builder: (column) => column); } @@ -4965,7 +4858,6 @@ class $$PreferencesTableTableTableManager extends RootTableManager< $$PreferencesTableTableAnnotationComposer($db: db, $table: table), updateCompanionCallback: ({ Value id = const Value.absent(), - Value audioQuality = const Value.absent(), Value albumColorSync = const Value.absent(), Value amoledDarkTheme = const Value.absent(), Value checkUpdate = const Value.absent(), @@ -4981,20 +4873,18 @@ class $$PreferencesTableTableTableManager extends RootTableManager< Value searchMode = const Value.absent(), Value downloadLocation = const Value.absent(), Value> localLibraryLocation = const Value.absent(), - Value pipedInstance = const Value.absent(), - Value invidiousInstance = const Value.absent(), Value themeMode = const Value.absent(), - Value audioSource = const Value.absent(), - Value streamMusicCodec = const Value.absent(), - Value downloadMusicCodec = const Value.absent(), + Value audioSourceId = const Value.absent(), + Value youtubeClientEngine = + const Value.absent(), Value discordPresence = const Value.absent(), Value endlessPlayback = const Value.absent(), Value enableConnect = const Value.absent(), + Value connectPort = const Value.absent(), Value cacheMusic = const Value.absent(), }) => PreferencesTableCompanion( id: id, - audioQuality: audioQuality, albumColorSync: albumColorSync, amoledDarkTheme: amoledDarkTheme, checkUpdate: checkUpdate, @@ -5010,20 +4900,17 @@ class $$PreferencesTableTableTableManager extends RootTableManager< searchMode: searchMode, downloadLocation: downloadLocation, localLibraryLocation: localLibraryLocation, - pipedInstance: pipedInstance, - invidiousInstance: invidiousInstance, themeMode: themeMode, - audioSource: audioSource, - streamMusicCodec: streamMusicCodec, - downloadMusicCodec: downloadMusicCodec, + audioSourceId: audioSourceId, + youtubeClientEngine: youtubeClientEngine, discordPresence: discordPresence, endlessPlayback: endlessPlayback, enableConnect: enableConnect, + connectPort: connectPort, cacheMusic: cacheMusic, ), createCompanionCallback: ({ Value id = const Value.absent(), - Value audioQuality = const Value.absent(), Value albumColorSync = const Value.absent(), Value amoledDarkTheme = const Value.absent(), Value checkUpdate = const Value.absent(), @@ -5039,20 +4926,18 @@ class $$PreferencesTableTableTableManager extends RootTableManager< Value searchMode = const Value.absent(), Value downloadLocation = const Value.absent(), Value> localLibraryLocation = const Value.absent(), - Value pipedInstance = const Value.absent(), - Value invidiousInstance = const Value.absent(), Value themeMode = const Value.absent(), - Value audioSource = const Value.absent(), - Value streamMusicCodec = const Value.absent(), - Value downloadMusicCodec = const Value.absent(), + Value audioSourceId = const Value.absent(), + Value youtubeClientEngine = + const Value.absent(), Value discordPresence = const Value.absent(), Value endlessPlayback = const Value.absent(), Value enableConnect = const Value.absent(), + Value connectPort = const Value.absent(), Value cacheMusic = const Value.absent(), }) => PreferencesTableCompanion.insert( id: id, - audioQuality: audioQuality, albumColorSync: albumColorSync, amoledDarkTheme: amoledDarkTheme, checkUpdate: checkUpdate, @@ -5068,15 +4953,13 @@ class $$PreferencesTableTableTableManager extends RootTableManager< searchMode: searchMode, downloadLocation: downloadLocation, localLibraryLocation: localLibraryLocation, - pipedInstance: pipedInstance, - invidiousInstance: invidiousInstance, themeMode: themeMode, - audioSource: audioSource, - streamMusicCodec: streamMusicCodec, - downloadMusicCodec: downloadMusicCodec, + audioSourceId: audioSourceId, + youtubeClientEngine: youtubeClientEngine, discordPresence: discordPresence, endlessPlayback: endlessPlayback, enableConnect: enableConnect, + connectPort: connectPort, cacheMusic: cacheMusic, ), withReferenceMapper: (p0) => p0 @@ -5433,16 +5316,16 @@ typedef $$SourceMatchTableTableCreateCompanionBuilder = SourceMatchTableCompanion Function({ Value id, required String trackId, - required String sourceId, - Value sourceType, + Value sourceInfo, + required String sourceType, Value createdAt, }); typedef $$SourceMatchTableTableUpdateCompanionBuilder = SourceMatchTableCompanion Function({ Value id, Value trackId, - Value sourceId, - Value sourceType, + Value sourceInfo, + Value sourceType, Value createdAt, }); @@ -5461,13 +5344,11 @@ class $$SourceMatchTableTableFilterComposer ColumnFilters get trackId => $composableBuilder( column: $table.trackId, builder: (column) => ColumnFilters(column)); - ColumnFilters get sourceId => $composableBuilder( - column: $table.sourceId, builder: (column) => ColumnFilters(column)); + ColumnFilters get sourceInfo => $composableBuilder( + column: $table.sourceInfo, builder: (column) => ColumnFilters(column)); - ColumnWithTypeConverterFilters - get sourceType => $composableBuilder( - column: $table.sourceType, - builder: (column) => ColumnWithTypeConverterFilters(column)); + ColumnFilters get sourceType => $composableBuilder( + column: $table.sourceType, builder: (column) => ColumnFilters(column)); ColumnFilters get createdAt => $composableBuilder( column: $table.createdAt, builder: (column) => ColumnFilters(column)); @@ -5488,8 +5369,8 @@ class $$SourceMatchTableTableOrderingComposer ColumnOrderings get trackId => $composableBuilder( column: $table.trackId, builder: (column) => ColumnOrderings(column)); - ColumnOrderings get sourceId => $composableBuilder( - column: $table.sourceId, builder: (column) => ColumnOrderings(column)); + ColumnOrderings get sourceInfo => $composableBuilder( + column: $table.sourceInfo, builder: (column) => ColumnOrderings(column)); ColumnOrderings get sourceType => $composableBuilder( column: $table.sourceType, builder: (column) => ColumnOrderings(column)); @@ -5513,12 +5394,11 @@ class $$SourceMatchTableTableAnnotationComposer GeneratedColumn get trackId => $composableBuilder(column: $table.trackId, builder: (column) => column); - GeneratedColumn get sourceId => - $composableBuilder(column: $table.sourceId, builder: (column) => column); + GeneratedColumn get sourceInfo => $composableBuilder( + column: $table.sourceInfo, builder: (column) => column); - GeneratedColumnWithTypeConverter get sourceType => - $composableBuilder( - column: $table.sourceType, builder: (column) => column); + GeneratedColumn get sourceType => $composableBuilder( + column: $table.sourceType, builder: (column) => column); GeneratedColumn get createdAt => $composableBuilder(column: $table.createdAt, builder: (column) => column); @@ -5554,28 +5434,28 @@ class $$SourceMatchTableTableTableManager extends RootTableManager< updateCompanionCallback: ({ Value id = const Value.absent(), Value trackId = const Value.absent(), - Value sourceId = const Value.absent(), - Value sourceType = const Value.absent(), + Value sourceInfo = const Value.absent(), + Value sourceType = const Value.absent(), Value createdAt = const Value.absent(), }) => SourceMatchTableCompanion( id: id, trackId: trackId, - sourceId: sourceId, + sourceInfo: sourceInfo, sourceType: sourceType, createdAt: createdAt, ), createCompanionCallback: ({ Value id = const Value.absent(), required String trackId, - required String sourceId, - Value sourceType = const Value.absent(), + Value sourceInfo = const Value.absent(), + required String sourceType, Value createdAt = const Value.absent(), }) => SourceMatchTableCompanion.insert( id: id, trackId: trackId, - sourceId: sourceId, + sourceInfo: sourceInfo, sourceType: sourceType, createdAt: createdAt, ), @@ -5609,6 +5489,8 @@ typedef $$AudioPlayerStateTableTableCreateCompanionBuilder required PlaylistMode loopMode, required bool shuffled, required List collections, + Value> tracks, + Value currentIndex, }); typedef $$AudioPlayerStateTableTableUpdateCompanionBuilder = AudioPlayerStateTableCompanion Function({ @@ -5617,29 +5499,10 @@ typedef $$AudioPlayerStateTableTableUpdateCompanionBuilder Value loopMode, Value shuffled, Value> collections, + Value> tracks, + Value currentIndex, }); -final class $$AudioPlayerStateTableTableReferences extends BaseReferences< - _$AppDatabase, $AudioPlayerStateTableTable, AudioPlayerStateTableData> { - $$AudioPlayerStateTableTableReferences( - super.$_db, super.$_table, super.$_typedResult); - - static MultiTypedResultKey<$PlaylistTableTable, List> - _playlistTableRefsTable(_$AppDatabase db) => - MultiTypedResultKey.fromTable(db.playlistTable, - aliasName: $_aliasNameGenerator(db.audioPlayerStateTable.id, - db.playlistTable.audioPlayerStateId)); - - $$PlaylistTableTableProcessedTableManager get playlistTableRefs { - final manager = $$PlaylistTableTableTableManager($_db, $_db.playlistTable) - .filter((f) => f.audioPlayerStateId.id($_item.id)); - - final cache = $_typedResult.readTableOrNull(_playlistTableRefsTable($_db)); - return ProcessedTableManager( - manager.$state.copyWith(prefetchedData: cache)); - } -} - class $$AudioPlayerStateTableTableFilterComposer extends Composer<_$AppDatabase, $AudioPlayerStateTableTable> { $$AudioPlayerStateTableTableFilterComposer({ @@ -5668,26 +5531,14 @@ class $$AudioPlayerStateTableTableFilterComposer column: $table.collections, builder: (column) => ColumnWithTypeConverterFilters(column)); - Expression playlistTableRefs( - Expression Function($$PlaylistTableTableFilterComposer f) f) { - final $$PlaylistTableTableFilterComposer composer = $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.id, - referencedTable: $db.playlistTable, - getReferencedColumn: (t) => t.audioPlayerStateId, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistTableTableFilterComposer( - $db: $db, - $table: $db.playlistTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return f(composer); - } + ColumnWithTypeConverterFilters, + List, String> + get tracks => $composableBuilder( + column: $table.tracks, + builder: (column) => ColumnWithTypeConverterFilters(column)); + + ColumnFilters get currentIndex => $composableBuilder( + column: $table.currentIndex, builder: (column) => ColumnFilters(column)); } class $$AudioPlayerStateTableTableOrderingComposer @@ -5713,6 +5564,13 @@ class $$AudioPlayerStateTableTableOrderingComposer ColumnOrderings get collections => $composableBuilder( column: $table.collections, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get tracks => $composableBuilder( + column: $table.tracks, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get currentIndex => $composableBuilder( + column: $table.currentIndex, + builder: (column) => ColumnOrderings(column)); } class $$AudioPlayerStateTableTableAnnotationComposer @@ -5740,26 +5598,12 @@ class $$AudioPlayerStateTableTableAnnotationComposer $composableBuilder( column: $table.collections, builder: (column) => column); - Expression playlistTableRefs( - Expression Function($$PlaylistTableTableAnnotationComposer a) f) { - final $$PlaylistTableTableAnnotationComposer composer = $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.id, - referencedTable: $db.playlistTable, - getReferencedColumn: (t) => t.audioPlayerStateId, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistTableTableAnnotationComposer( - $db: $db, - $table: $db.playlistTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return f(composer); - } + GeneratedColumnWithTypeConverter, String> + get tracks => $composableBuilder( + column: $table.tracks, builder: (column) => column); + + GeneratedColumn get currentIndex => $composableBuilder( + column: $table.currentIndex, builder: (column) => column); } class $$AudioPlayerStateTableTableTableManager extends RootTableManager< @@ -5771,9 +5615,13 @@ class $$AudioPlayerStateTableTableTableManager extends RootTableManager< $$AudioPlayerStateTableTableAnnotationComposer, $$AudioPlayerStateTableTableCreateCompanionBuilder, $$AudioPlayerStateTableTableUpdateCompanionBuilder, - (AudioPlayerStateTableData, $$AudioPlayerStateTableTableReferences), + ( + AudioPlayerStateTableData, + BaseReferences<_$AppDatabase, $AudioPlayerStateTableTable, + AudioPlayerStateTableData> + ), AudioPlayerStateTableData, - PrefetchHooks Function({bool playlistTableRefs})> { + PrefetchHooks Function()> { $$AudioPlayerStateTableTableTableManager( _$AppDatabase db, $AudioPlayerStateTableTable table) : super(TableManagerState( @@ -5794,6 +5642,8 @@ class $$AudioPlayerStateTableTableTableManager extends RootTableManager< Value loopMode = const Value.absent(), Value shuffled = const Value.absent(), Value> collections = const Value.absent(), + Value> tracks = const Value.absent(), + Value currentIndex = const Value.absent(), }) => AudioPlayerStateTableCompanion( id: id, @@ -5801,6 +5651,8 @@ class $$AudioPlayerStateTableTableTableManager extends RootTableManager< loopMode: loopMode, shuffled: shuffled, collections: collections, + tracks: tracks, + currentIndex: currentIndex, ), createCompanionCallback: ({ Value id = const Value.absent(), @@ -5808,6 +5660,8 @@ class $$AudioPlayerStateTableTableTableManager extends RootTableManager< required PlaylistMode loopMode, required bool shuffled, required List collections, + Value> tracks = const Value.absent(), + Value currentIndex = const Value.absent(), }) => AudioPlayerStateTableCompanion.insert( id: id, @@ -5815,39 +5669,13 @@ class $$AudioPlayerStateTableTableTableManager extends RootTableManager< loopMode: loopMode, shuffled: shuffled, collections: collections, + tracks: tracks, + currentIndex: currentIndex, ), withReferenceMapper: (p0) => p0 - .map((e) => ( - e.readTable(table), - $$AudioPlayerStateTableTableReferences(db, table, e) - )) + .map((e) => (e.readTable(table), BaseReferences(db, table, e))) .toList(), - prefetchHooksCallback: ({playlistTableRefs = false}) { - return PrefetchHooks( - db: db, - explicitlyWatchedTables: [ - if (playlistTableRefs) db.playlistTable - ], - addJoins: null, - getPrefetchedDataCallback: (items) async { - return [ - if (playlistTableRefs) - await $_getPrefetchedData( - currentTable: table, - referencedTable: $$AudioPlayerStateTableTableReferences - ._playlistTableRefsTable(db), - managerFromTypedResult: (p0) => - $$AudioPlayerStateTableTableReferences( - db, table, p0) - .playlistTableRefs, - referencedItemsForCurrentItem: - (item, referencedItems) => referencedItems - .where((e) => e.audioPlayerStateId == item.id), - typedResults: items) - ]; - }, - ); - }, + prefetchHooksCallback: null, )); } @@ -5861,610 +5689,13 @@ typedef $$AudioPlayerStateTableTableProcessedTableManager $$AudioPlayerStateTableTableAnnotationComposer, $$AudioPlayerStateTableTableCreateCompanionBuilder, $$AudioPlayerStateTableTableUpdateCompanionBuilder, - (AudioPlayerStateTableData, $$AudioPlayerStateTableTableReferences), + ( + AudioPlayerStateTableData, + BaseReferences<_$AppDatabase, $AudioPlayerStateTableTable, + AudioPlayerStateTableData> + ), AudioPlayerStateTableData, - PrefetchHooks Function({bool playlistTableRefs})>; -typedef $$PlaylistTableTableCreateCompanionBuilder = PlaylistTableCompanion - Function({ - Value id, - required int audioPlayerStateId, - required int index, -}); -typedef $$PlaylistTableTableUpdateCompanionBuilder = PlaylistTableCompanion - Function({ - Value id, - Value audioPlayerStateId, - Value index, -}); - -final class $$PlaylistTableTableReferences extends BaseReferences<_$AppDatabase, - $PlaylistTableTable, PlaylistTableData> { - $$PlaylistTableTableReferences( - super.$_db, super.$_table, super.$_typedResult); - - static $AudioPlayerStateTableTable _audioPlayerStateIdTable( - _$AppDatabase db) => - db.audioPlayerStateTable.createAlias($_aliasNameGenerator( - db.playlistTable.audioPlayerStateId, db.audioPlayerStateTable.id)); - - $$AudioPlayerStateTableTableProcessedTableManager get audioPlayerStateId { - final manager = $$AudioPlayerStateTableTableTableManager( - $_db, $_db.audioPlayerStateTable) - .filter((f) => f.id($_item.audioPlayerStateId!)); - final item = $_typedResult.readTableOrNull(_audioPlayerStateIdTable($_db)); - if (item == null) return manager; - return ProcessedTableManager( - manager.$state.copyWith(prefetchedData: [item])); - } - - static MultiTypedResultKey<$PlaylistMediaTableTable, - List> _playlistMediaTableRefsTable( - _$AppDatabase db) => - MultiTypedResultKey.fromTable(db.playlistMediaTable, - aliasName: $_aliasNameGenerator( - db.playlistTable.id, db.playlistMediaTable.playlistId)); - - $$PlaylistMediaTableTableProcessedTableManager get playlistMediaTableRefs { - final manager = - $$PlaylistMediaTableTableTableManager($_db, $_db.playlistMediaTable) - .filter((f) => f.playlistId.id($_item.id)); - - final cache = - $_typedResult.readTableOrNull(_playlistMediaTableRefsTable($_db)); - return ProcessedTableManager( - manager.$state.copyWith(prefetchedData: cache)); - } -} - -class $$PlaylistTableTableFilterComposer - extends Composer<_$AppDatabase, $PlaylistTableTable> { - $$PlaylistTableTableFilterComposer({ - required super.$db, - required super.$table, - super.joinBuilder, - super.$addJoinBuilderToRootComposer, - super.$removeJoinBuilderFromRootComposer, - }); - ColumnFilters get id => $composableBuilder( - column: $table.id, builder: (column) => ColumnFilters(column)); - - ColumnFilters get index => $composableBuilder( - column: $table.index, builder: (column) => ColumnFilters(column)); - - $$AudioPlayerStateTableTableFilterComposer get audioPlayerStateId { - final $$AudioPlayerStateTableTableFilterComposer composer = - $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.audioPlayerStateId, - referencedTable: $db.audioPlayerStateTable, - getReferencedColumn: (t) => t.id, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$AudioPlayerStateTableTableFilterComposer( - $db: $db, - $table: $db.audioPlayerStateTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return composer; - } - - Expression playlistMediaTableRefs( - Expression Function($$PlaylistMediaTableTableFilterComposer f) f) { - final $$PlaylistMediaTableTableFilterComposer composer = $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.id, - referencedTable: $db.playlistMediaTable, - getReferencedColumn: (t) => t.playlistId, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistMediaTableTableFilterComposer( - $db: $db, - $table: $db.playlistMediaTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return f(composer); - } -} - -class $$PlaylistTableTableOrderingComposer - extends Composer<_$AppDatabase, $PlaylistTableTable> { - $$PlaylistTableTableOrderingComposer({ - required super.$db, - required super.$table, - super.joinBuilder, - super.$addJoinBuilderToRootComposer, - super.$removeJoinBuilderFromRootComposer, - }); - ColumnOrderings get id => $composableBuilder( - column: $table.id, builder: (column) => ColumnOrderings(column)); - - ColumnOrderings get index => $composableBuilder( - column: $table.index, builder: (column) => ColumnOrderings(column)); - - $$AudioPlayerStateTableTableOrderingComposer get audioPlayerStateId { - final $$AudioPlayerStateTableTableOrderingComposer composer = - $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.audioPlayerStateId, - referencedTable: $db.audioPlayerStateTable, - getReferencedColumn: (t) => t.id, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$AudioPlayerStateTableTableOrderingComposer( - $db: $db, - $table: $db.audioPlayerStateTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return composer; - } -} - -class $$PlaylistTableTableAnnotationComposer - extends Composer<_$AppDatabase, $PlaylistTableTable> { - $$PlaylistTableTableAnnotationComposer({ - required super.$db, - required super.$table, - super.joinBuilder, - super.$addJoinBuilderToRootComposer, - super.$removeJoinBuilderFromRootComposer, - }); - GeneratedColumn get id => - $composableBuilder(column: $table.id, builder: (column) => column); - - GeneratedColumn get index => - $composableBuilder(column: $table.index, builder: (column) => column); - - $$AudioPlayerStateTableTableAnnotationComposer get audioPlayerStateId { - final $$AudioPlayerStateTableTableAnnotationComposer composer = - $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.audioPlayerStateId, - referencedTable: $db.audioPlayerStateTable, - getReferencedColumn: (t) => t.id, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$AudioPlayerStateTableTableAnnotationComposer( - $db: $db, - $table: $db.audioPlayerStateTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return composer; - } - - Expression playlistMediaTableRefs( - Expression Function($$PlaylistMediaTableTableAnnotationComposer a) f) { - final $$PlaylistMediaTableTableAnnotationComposer composer = - $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.id, - referencedTable: $db.playlistMediaTable, - getReferencedColumn: (t) => t.playlistId, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistMediaTableTableAnnotationComposer( - $db: $db, - $table: $db.playlistMediaTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return f(composer); - } -} - -class $$PlaylistTableTableTableManager extends RootTableManager< - _$AppDatabase, - $PlaylistTableTable, - PlaylistTableData, - $$PlaylistTableTableFilterComposer, - $$PlaylistTableTableOrderingComposer, - $$PlaylistTableTableAnnotationComposer, - $$PlaylistTableTableCreateCompanionBuilder, - $$PlaylistTableTableUpdateCompanionBuilder, - (PlaylistTableData, $$PlaylistTableTableReferences), - PlaylistTableData, - PrefetchHooks Function( - {bool audioPlayerStateId, bool playlistMediaTableRefs})> { - $$PlaylistTableTableTableManager(_$AppDatabase db, $PlaylistTableTable table) - : super(TableManagerState( - db: db, - table: table, - createFilteringComposer: () => - $$PlaylistTableTableFilterComposer($db: db, $table: table), - createOrderingComposer: () => - $$PlaylistTableTableOrderingComposer($db: db, $table: table), - createComputedFieldComposer: () => - $$PlaylistTableTableAnnotationComposer($db: db, $table: table), - updateCompanionCallback: ({ - Value id = const Value.absent(), - Value audioPlayerStateId = const Value.absent(), - Value index = const Value.absent(), - }) => - PlaylistTableCompanion( - id: id, - audioPlayerStateId: audioPlayerStateId, - index: index, - ), - createCompanionCallback: ({ - Value id = const Value.absent(), - required int audioPlayerStateId, - required int index, - }) => - PlaylistTableCompanion.insert( - id: id, - audioPlayerStateId: audioPlayerStateId, - index: index, - ), - withReferenceMapper: (p0) => p0 - .map((e) => ( - e.readTable(table), - $$PlaylistTableTableReferences(db, table, e) - )) - .toList(), - prefetchHooksCallback: ( - {audioPlayerStateId = false, playlistMediaTableRefs = false}) { - return PrefetchHooks( - db: db, - explicitlyWatchedTables: [ - if (playlistMediaTableRefs) db.playlistMediaTable - ], - addJoins: < - T extends TableManagerState< - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic>>(state) { - if (audioPlayerStateId) { - state = state.withJoin( - currentTable: table, - currentColumn: table.audioPlayerStateId, - referencedTable: $$PlaylistTableTableReferences - ._audioPlayerStateIdTable(db), - referencedColumn: $$PlaylistTableTableReferences - ._audioPlayerStateIdTable(db) - .id, - ) as T; - } - - return state; - }, - getPrefetchedDataCallback: (items) async { - return [ - if (playlistMediaTableRefs) - await $_getPrefetchedData( - currentTable: table, - referencedTable: $$PlaylistTableTableReferences - ._playlistMediaTableRefsTable(db), - managerFromTypedResult: (p0) => - $$PlaylistTableTableReferences(db, table, p0) - .playlistMediaTableRefs, - referencedItemsForCurrentItem: - (item, referencedItems) => referencedItems - .where((e) => e.playlistId == item.id), - typedResults: items) - ]; - }, - ); - }, - )); -} - -typedef $$PlaylistTableTableProcessedTableManager = ProcessedTableManager< - _$AppDatabase, - $PlaylistTableTable, - PlaylistTableData, - $$PlaylistTableTableFilterComposer, - $$PlaylistTableTableOrderingComposer, - $$PlaylistTableTableAnnotationComposer, - $$PlaylistTableTableCreateCompanionBuilder, - $$PlaylistTableTableUpdateCompanionBuilder, - (PlaylistTableData, $$PlaylistTableTableReferences), - PlaylistTableData, - PrefetchHooks Function( - {bool audioPlayerStateId, bool playlistMediaTableRefs})>; -typedef $$PlaylistMediaTableTableCreateCompanionBuilder - = PlaylistMediaTableCompanion Function({ - Value id, - required int playlistId, - required String uri, - Value?> extras, - Value?> httpHeaders, -}); -typedef $$PlaylistMediaTableTableUpdateCompanionBuilder - = PlaylistMediaTableCompanion Function({ - Value id, - Value playlistId, - Value uri, - Value?> extras, - Value?> httpHeaders, -}); - -final class $$PlaylistMediaTableTableReferences extends BaseReferences< - _$AppDatabase, $PlaylistMediaTableTable, PlaylistMediaTableData> { - $$PlaylistMediaTableTableReferences( - super.$_db, super.$_table, super.$_typedResult); - - static $PlaylistTableTable _playlistIdTable(_$AppDatabase db) => - db.playlistTable.createAlias($_aliasNameGenerator( - db.playlistMediaTable.playlistId, db.playlistTable.id)); - - $$PlaylistTableTableProcessedTableManager get playlistId { - final manager = $$PlaylistTableTableTableManager($_db, $_db.playlistTable) - .filter((f) => f.id($_item.playlistId!)); - final item = $_typedResult.readTableOrNull(_playlistIdTable($_db)); - if (item == null) return manager; - return ProcessedTableManager( - manager.$state.copyWith(prefetchedData: [item])); - } -} - -class $$PlaylistMediaTableTableFilterComposer - extends Composer<_$AppDatabase, $PlaylistMediaTableTable> { - $$PlaylistMediaTableTableFilterComposer({ - required super.$db, - required super.$table, - super.joinBuilder, - super.$addJoinBuilderToRootComposer, - super.$removeJoinBuilderFromRootComposer, - }); - ColumnFilters get id => $composableBuilder( - column: $table.id, builder: (column) => ColumnFilters(column)); - - ColumnFilters get uri => $composableBuilder( - column: $table.uri, builder: (column) => ColumnFilters(column)); - - ColumnWithTypeConverterFilters?, Map, - String> - get extras => $composableBuilder( - column: $table.extras, - builder: (column) => ColumnWithTypeConverterFilters(column)); - - ColumnWithTypeConverterFilters?, Map, - String> - get httpHeaders => $composableBuilder( - column: $table.httpHeaders, - builder: (column) => ColumnWithTypeConverterFilters(column)); - - $$PlaylistTableTableFilterComposer get playlistId { - final $$PlaylistTableTableFilterComposer composer = $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.playlistId, - referencedTable: $db.playlistTable, - getReferencedColumn: (t) => t.id, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistTableTableFilterComposer( - $db: $db, - $table: $db.playlistTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return composer; - } -} - -class $$PlaylistMediaTableTableOrderingComposer - extends Composer<_$AppDatabase, $PlaylistMediaTableTable> { - $$PlaylistMediaTableTableOrderingComposer({ - required super.$db, - required super.$table, - super.joinBuilder, - super.$addJoinBuilderToRootComposer, - super.$removeJoinBuilderFromRootComposer, - }); - ColumnOrderings get id => $composableBuilder( - column: $table.id, builder: (column) => ColumnOrderings(column)); - - ColumnOrderings get uri => $composableBuilder( - column: $table.uri, builder: (column) => ColumnOrderings(column)); - - ColumnOrderings get extras => $composableBuilder( - column: $table.extras, builder: (column) => ColumnOrderings(column)); - - ColumnOrderings get httpHeaders => $composableBuilder( - column: $table.httpHeaders, builder: (column) => ColumnOrderings(column)); - - $$PlaylistTableTableOrderingComposer get playlistId { - final $$PlaylistTableTableOrderingComposer composer = $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.playlistId, - referencedTable: $db.playlistTable, - getReferencedColumn: (t) => t.id, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistTableTableOrderingComposer( - $db: $db, - $table: $db.playlistTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return composer; - } -} - -class $$PlaylistMediaTableTableAnnotationComposer - extends Composer<_$AppDatabase, $PlaylistMediaTableTable> { - $$PlaylistMediaTableTableAnnotationComposer({ - required super.$db, - required super.$table, - super.joinBuilder, - super.$addJoinBuilderToRootComposer, - super.$removeJoinBuilderFromRootComposer, - }); - GeneratedColumn get id => - $composableBuilder(column: $table.id, builder: (column) => column); - - GeneratedColumn get uri => - $composableBuilder(column: $table.uri, builder: (column) => column); - - GeneratedColumnWithTypeConverter?, String> get extras => - $composableBuilder(column: $table.extras, builder: (column) => column); - - GeneratedColumnWithTypeConverter?, String> - get httpHeaders => $composableBuilder( - column: $table.httpHeaders, builder: (column) => column); - - $$PlaylistTableTableAnnotationComposer get playlistId { - final $$PlaylistTableTableAnnotationComposer composer = $composerBuilder( - composer: this, - getCurrentColumn: (t) => t.playlistId, - referencedTable: $db.playlistTable, - getReferencedColumn: (t) => t.id, - builder: (joinBuilder, - {$addJoinBuilderToRootComposer, - $removeJoinBuilderFromRootComposer}) => - $$PlaylistTableTableAnnotationComposer( - $db: $db, - $table: $db.playlistTable, - $addJoinBuilderToRootComposer: $addJoinBuilderToRootComposer, - joinBuilder: joinBuilder, - $removeJoinBuilderFromRootComposer: - $removeJoinBuilderFromRootComposer, - )); - return composer; - } -} - -class $$PlaylistMediaTableTableTableManager extends RootTableManager< - _$AppDatabase, - $PlaylistMediaTableTable, - PlaylistMediaTableData, - $$PlaylistMediaTableTableFilterComposer, - $$PlaylistMediaTableTableOrderingComposer, - $$PlaylistMediaTableTableAnnotationComposer, - $$PlaylistMediaTableTableCreateCompanionBuilder, - $$PlaylistMediaTableTableUpdateCompanionBuilder, - (PlaylistMediaTableData, $$PlaylistMediaTableTableReferences), - PlaylistMediaTableData, - PrefetchHooks Function({bool playlistId})> { - $$PlaylistMediaTableTableTableManager( - _$AppDatabase db, $PlaylistMediaTableTable table) - : super(TableManagerState( - db: db, - table: table, - createFilteringComposer: () => - $$PlaylistMediaTableTableFilterComposer($db: db, $table: table), - createOrderingComposer: () => - $$PlaylistMediaTableTableOrderingComposer($db: db, $table: table), - createComputedFieldComposer: () => - $$PlaylistMediaTableTableAnnotationComposer( - $db: db, $table: table), - updateCompanionCallback: ({ - Value id = const Value.absent(), - Value playlistId = const Value.absent(), - Value uri = const Value.absent(), - Value?> extras = const Value.absent(), - Value?> httpHeaders = const Value.absent(), - }) => - PlaylistMediaTableCompanion( - id: id, - playlistId: playlistId, - uri: uri, - extras: extras, - httpHeaders: httpHeaders, - ), - createCompanionCallback: ({ - Value id = const Value.absent(), - required int playlistId, - required String uri, - Value?> extras = const Value.absent(), - Value?> httpHeaders = const Value.absent(), - }) => - PlaylistMediaTableCompanion.insert( - id: id, - playlistId: playlistId, - uri: uri, - extras: extras, - httpHeaders: httpHeaders, - ), - withReferenceMapper: (p0) => p0 - .map((e) => ( - e.readTable(table), - $$PlaylistMediaTableTableReferences(db, table, e) - )) - .toList(), - prefetchHooksCallback: ({playlistId = false}) { - return PrefetchHooks( - db: db, - explicitlyWatchedTables: [], - addJoins: < - T extends TableManagerState< - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic, - dynamic>>(state) { - if (playlistId) { - state = state.withJoin( - currentTable: table, - currentColumn: table.playlistId, - referencedTable: $$PlaylistMediaTableTableReferences - ._playlistIdTable(db), - referencedColumn: $$PlaylistMediaTableTableReferences - ._playlistIdTable(db) - .id, - ) as T; - } - - return state; - }, - getPrefetchedDataCallback: (items) async { - return []; - }, - ); - }, - )); -} - -typedef $$PlaylistMediaTableTableProcessedTableManager = ProcessedTableManager< - _$AppDatabase, - $PlaylistMediaTableTable, - PlaylistMediaTableData, - $$PlaylistMediaTableTableFilterComposer, - $$PlaylistMediaTableTableOrderingComposer, - $$PlaylistMediaTableTableAnnotationComposer, - $$PlaylistMediaTableTableCreateCompanionBuilder, - $$PlaylistMediaTableTableUpdateCompanionBuilder, - (PlaylistMediaTableData, $$PlaylistMediaTableTableReferences), - PlaylistMediaTableData, - PrefetchHooks Function({bool playlistId})>; + PrefetchHooks Function()>; typedef $$HistoryTableTableCreateCompanionBuilder = HistoryTableCompanion Function({ Value id, @@ -6776,6 +6007,288 @@ typedef $$LyricsTableTableProcessedTableManager = ProcessedTableManager< ), LyricsTableData, PrefetchHooks Function()>; +typedef $$PluginsTableTableCreateCompanionBuilder = PluginsTableCompanion + Function({ + Value id, + required String name, + required String description, + required String version, + required String author, + required String entryPoint, + required List apis, + required List abilities, + Value selectedForMetadata, + Value selectedForAudioSource, + Value repository, + Value pluginApiVersion, +}); +typedef $$PluginsTableTableUpdateCompanionBuilder = PluginsTableCompanion + Function({ + Value id, + Value name, + Value description, + Value version, + Value author, + Value entryPoint, + Value> apis, + Value> abilities, + Value selectedForMetadata, + Value selectedForAudioSource, + Value repository, + Value pluginApiVersion, +}); + +class $$PluginsTableTableFilterComposer + extends Composer<_$AppDatabase, $PluginsTableTable> { + $$PluginsTableTableFilterComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + ColumnFilters get id => $composableBuilder( + column: $table.id, builder: (column) => ColumnFilters(column)); + + ColumnFilters get name => $composableBuilder( + column: $table.name, builder: (column) => ColumnFilters(column)); + + ColumnFilters get description => $composableBuilder( + column: $table.description, builder: (column) => ColumnFilters(column)); + + ColumnFilters get version => $composableBuilder( + column: $table.version, builder: (column) => ColumnFilters(column)); + + ColumnFilters get author => $composableBuilder( + column: $table.author, builder: (column) => ColumnFilters(column)); + + ColumnFilters get entryPoint => $composableBuilder( + column: $table.entryPoint, builder: (column) => ColumnFilters(column)); + + ColumnWithTypeConverterFilters, List, String> get apis => + $composableBuilder( + column: $table.apis, + builder: (column) => ColumnWithTypeConverterFilters(column)); + + ColumnWithTypeConverterFilters, List, String> + get abilities => $composableBuilder( + column: $table.abilities, + builder: (column) => ColumnWithTypeConverterFilters(column)); + + ColumnFilters get selectedForMetadata => $composableBuilder( + column: $table.selectedForMetadata, + builder: (column) => ColumnFilters(column)); + + ColumnFilters get selectedForAudioSource => $composableBuilder( + column: $table.selectedForAudioSource, + builder: (column) => ColumnFilters(column)); + + ColumnFilters get repository => $composableBuilder( + column: $table.repository, builder: (column) => ColumnFilters(column)); + + ColumnFilters get pluginApiVersion => $composableBuilder( + column: $table.pluginApiVersion, + builder: (column) => ColumnFilters(column)); +} + +class $$PluginsTableTableOrderingComposer + extends Composer<_$AppDatabase, $PluginsTableTable> { + $$PluginsTableTableOrderingComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + ColumnOrderings get id => $composableBuilder( + column: $table.id, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get name => $composableBuilder( + column: $table.name, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get description => $composableBuilder( + column: $table.description, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get version => $composableBuilder( + column: $table.version, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get author => $composableBuilder( + column: $table.author, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get entryPoint => $composableBuilder( + column: $table.entryPoint, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get apis => $composableBuilder( + column: $table.apis, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get abilities => $composableBuilder( + column: $table.abilities, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get selectedForMetadata => $composableBuilder( + column: $table.selectedForMetadata, + builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get selectedForAudioSource => $composableBuilder( + column: $table.selectedForAudioSource, + builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get repository => $composableBuilder( + column: $table.repository, builder: (column) => ColumnOrderings(column)); + + ColumnOrderings get pluginApiVersion => $composableBuilder( + column: $table.pluginApiVersion, + builder: (column) => ColumnOrderings(column)); +} + +class $$PluginsTableTableAnnotationComposer + extends Composer<_$AppDatabase, $PluginsTableTable> { + $$PluginsTableTableAnnotationComposer({ + required super.$db, + required super.$table, + super.joinBuilder, + super.$addJoinBuilderToRootComposer, + super.$removeJoinBuilderFromRootComposer, + }); + GeneratedColumn get id => + $composableBuilder(column: $table.id, builder: (column) => column); + + GeneratedColumn get name => + $composableBuilder(column: $table.name, builder: (column) => column); + + GeneratedColumn get description => $composableBuilder( + column: $table.description, builder: (column) => column); + + GeneratedColumn get version => + $composableBuilder(column: $table.version, builder: (column) => column); + + GeneratedColumn get author => + $composableBuilder(column: $table.author, builder: (column) => column); + + GeneratedColumn get entryPoint => $composableBuilder( + column: $table.entryPoint, builder: (column) => column); + + GeneratedColumnWithTypeConverter, String> get apis => + $composableBuilder(column: $table.apis, builder: (column) => column); + + GeneratedColumnWithTypeConverter, String> get abilities => + $composableBuilder(column: $table.abilities, builder: (column) => column); + + GeneratedColumn get selectedForMetadata => $composableBuilder( + column: $table.selectedForMetadata, builder: (column) => column); + + GeneratedColumn get selectedForAudioSource => $composableBuilder( + column: $table.selectedForAudioSource, builder: (column) => column); + + GeneratedColumn get repository => $composableBuilder( + column: $table.repository, builder: (column) => column); + + GeneratedColumn get pluginApiVersion => $composableBuilder( + column: $table.pluginApiVersion, builder: (column) => column); +} + +class $$PluginsTableTableTableManager extends RootTableManager< + _$AppDatabase, + $PluginsTableTable, + PluginsTableData, + $$PluginsTableTableFilterComposer, + $$PluginsTableTableOrderingComposer, + $$PluginsTableTableAnnotationComposer, + $$PluginsTableTableCreateCompanionBuilder, + $$PluginsTableTableUpdateCompanionBuilder, + ( + PluginsTableData, + BaseReferences<_$AppDatabase, $PluginsTableTable, PluginsTableData> + ), + PluginsTableData, + PrefetchHooks Function()> { + $$PluginsTableTableTableManager(_$AppDatabase db, $PluginsTableTable table) + : super(TableManagerState( + db: db, + table: table, + createFilteringComposer: () => + $$PluginsTableTableFilterComposer($db: db, $table: table), + createOrderingComposer: () => + $$PluginsTableTableOrderingComposer($db: db, $table: table), + createComputedFieldComposer: () => + $$PluginsTableTableAnnotationComposer($db: db, $table: table), + updateCompanionCallback: ({ + Value id = const Value.absent(), + Value name = const Value.absent(), + Value description = const Value.absent(), + Value version = const Value.absent(), + Value author = const Value.absent(), + Value entryPoint = const Value.absent(), + Value> apis = const Value.absent(), + Value> abilities = const Value.absent(), + Value selectedForMetadata = const Value.absent(), + Value selectedForAudioSource = const Value.absent(), + Value repository = const Value.absent(), + Value pluginApiVersion = const Value.absent(), + }) => + PluginsTableCompanion( + id: id, + name: name, + description: description, + version: version, + author: author, + entryPoint: entryPoint, + apis: apis, + abilities: abilities, + selectedForMetadata: selectedForMetadata, + selectedForAudioSource: selectedForAudioSource, + repository: repository, + pluginApiVersion: pluginApiVersion, + ), + createCompanionCallback: ({ + Value id = const Value.absent(), + required String name, + required String description, + required String version, + required String author, + required String entryPoint, + required List apis, + required List abilities, + Value selectedForMetadata = const Value.absent(), + Value selectedForAudioSource = const Value.absent(), + Value repository = const Value.absent(), + Value pluginApiVersion = const Value.absent(), + }) => + PluginsTableCompanion.insert( + id: id, + name: name, + description: description, + version: version, + author: author, + entryPoint: entryPoint, + apis: apis, + abilities: abilities, + selectedForMetadata: selectedForMetadata, + selectedForAudioSource: selectedForAudioSource, + repository: repository, + pluginApiVersion: pluginApiVersion, + ), + withReferenceMapper: (p0) => p0 + .map((e) => (e.readTable(table), BaseReferences(db, table, e))) + .toList(), + prefetchHooksCallback: null, + )); +} + +typedef $$PluginsTableTableProcessedTableManager = ProcessedTableManager< + _$AppDatabase, + $PluginsTableTable, + PluginsTableData, + $$PluginsTableTableFilterComposer, + $$PluginsTableTableOrderingComposer, + $$PluginsTableTableAnnotationComposer, + $$PluginsTableTableCreateCompanionBuilder, + $$PluginsTableTableUpdateCompanionBuilder, + ( + PluginsTableData, + BaseReferences<_$AppDatabase, $PluginsTableTable, PluginsTableData> + ), + PluginsTableData, + PrefetchHooks Function()>; class $AppDatabaseManager { final _$AppDatabase _db; @@ -6794,12 +6307,10 @@ class $AppDatabaseManager { $$SourceMatchTableTableTableManager(_db, _db.sourceMatchTable); $$AudioPlayerStateTableTableTableManager get audioPlayerStateTable => $$AudioPlayerStateTableTableTableManager(_db, _db.audioPlayerStateTable); - $$PlaylistTableTableTableManager get playlistTable => - $$PlaylistTableTableTableManager(_db, _db.playlistTable); - $$PlaylistMediaTableTableTableManager get playlistMediaTable => - $$PlaylistMediaTableTableTableManager(_db, _db.playlistMediaTable); $$HistoryTableTableTableManager get historyTable => $$HistoryTableTableTableManager(_db, _db.historyTable); $$LyricsTableTableTableManager get lyricsTable => $$LyricsTableTableTableManager(_db, _db.lyricsTable); + $$PluginsTableTableTableManager get pluginsTable => + $$PluginsTableTableTableManager(_db, _db.pluginsTable); } diff --git a/lib/models/database/database.steps.dart b/lib/models/database/database.steps.dart index 40546bdb..42cbdf6d 100644 --- a/lib/models/database/database.steps.dart +++ b/lib/models/database/database.steps.dart @@ -3,9 +3,8 @@ import 'package:drift/internal/versioned_schema.dart' as i0; import 'package:drift/drift.dart' as i1; import 'package:drift/drift.dart'; // ignore_for_file: type=lint,unused_import import 'package:flutter/material.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:spotube/utils/migrations/adapters.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/market.dart'; // GENERATED BY drift_dev, DO NOT MODIFY. final class Schema2 extends i0.VersionedSchema { @@ -330,8 +329,7 @@ class Shape2 extends i0.VersionedTable { i1.GeneratedColumn _column_7(String aliasedName) => i1.GeneratedColumn('audio_quality', aliasedName, false, - type: i1.DriftSqlType.string, - defaultValue: Constant(SourceQualities.high.name)); + type: i1.DriftSqlType.string, defaultValue: Constant("high")); i1.GeneratedColumn _column_8(String aliasedName) => i1.GeneratedColumn('album_color_sync', aliasedName, false, type: i1.DriftSqlType.bool, @@ -418,16 +416,13 @@ i1.GeneratedColumn _column_25(String aliasedName) => defaultValue: Constant(ThemeMode.system.name)); i1.GeneratedColumn _column_26(String aliasedName) => i1.GeneratedColumn('audio_source', aliasedName, false, - type: i1.DriftSqlType.string, - defaultValue: Constant(AudioSource.youtube.name)); + type: i1.DriftSqlType.string, defaultValue: Constant("youtube")); i1.GeneratedColumn _column_27(String aliasedName) => i1.GeneratedColumn('stream_music_codec', aliasedName, false, - type: i1.DriftSqlType.string, - defaultValue: Constant(SourceCodecs.weba.name)); + type: i1.DriftSqlType.string, defaultValue: Constant("weba")); i1.GeneratedColumn _column_28(String aliasedName) => i1.GeneratedColumn('download_music_codec', aliasedName, false, - type: i1.DriftSqlType.string, - defaultValue: Constant(SourceCodecs.m4a.name)); + type: i1.DriftSqlType.string, defaultValue: Constant("m4a")); i1.GeneratedColumn _column_29(String aliasedName) => i1.GeneratedColumn('discord_presence', aliasedName, false, type: i1.DriftSqlType.bool, @@ -512,8 +507,7 @@ i1.GeneratedColumn _column_38(String aliasedName) => type: i1.DriftSqlType.string); i1.GeneratedColumn _column_39(String aliasedName) => i1.GeneratedColumn('source_type', aliasedName, false, - type: i1.DriftSqlType.string, - defaultValue: Constant(SourceType.youtube.name)); + type: i1.DriftSqlType.string, defaultValue: Constant("youtube")); class Shape6 extends i0.VersionedTable { Shape6({required super.source, required super.alias}) : super.aliased(); @@ -907,9 +901,1854 @@ i1.GeneratedColumn _column_53(String aliasedName) => defaultConstraints: i1.GeneratedColumn.constraintIsAlways( 'CHECK ("cache_music" IN (0, 1))'), defaultValue: const Constant(true)); + +final class Schema4 extends i0.VersionedSchema { + Schema4({required super.database}) : super(version: 4); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + playlistTable, + playlistMediaTable, + historyTable, + lyricsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape12 preferencesTable = Shape12( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_7, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_16, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_23, + _column_24, + _column_25, + _column_26, + _column_54, + _column_27, + _column_28, + _column_29, + _column_30, + _column_31, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape5 sourceMatchTable = Shape5( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_38, + _column_39, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape6 audioPlayerStateTable = Shape6( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape7 playlistTable = Shape7( + source: i0.VersionedTable( + entityName: 'playlist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_44, + _column_45, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape8 playlistMediaTable = Shape8( + source: i0.VersionedTable( + entityName: 'playlist_media_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_46, + _column_47, + _column_48, + _column_49, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); +} + +class Shape12 extends i0.VersionedTable { + Shape12({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get audioQuality => + columnsByName['audio_quality']! as i1.GeneratedColumn; + i1.GeneratedColumn get albumColorSync => + columnsByName['album_color_sync']! as i1.GeneratedColumn; + i1.GeneratedColumn get amoledDarkTheme => + columnsByName['amoled_dark_theme']! as i1.GeneratedColumn; + i1.GeneratedColumn get checkUpdate => + columnsByName['check_update']! as i1.GeneratedColumn; + i1.GeneratedColumn get normalizeAudio => + columnsByName['normalize_audio']! as i1.GeneratedColumn; + i1.GeneratedColumn get showSystemTrayIcon => + columnsByName['show_system_tray_icon']! as i1.GeneratedColumn; + i1.GeneratedColumn get systemTitleBar => + columnsByName['system_title_bar']! as i1.GeneratedColumn; + i1.GeneratedColumn get skipNonMusic => + columnsByName['skip_non_music']! as i1.GeneratedColumn; + i1.GeneratedColumn get closeBehavior => + columnsByName['close_behavior']! as i1.GeneratedColumn; + i1.GeneratedColumn get accentColorScheme => + columnsByName['accent_color_scheme']! as i1.GeneratedColumn; + i1.GeneratedColumn get layoutMode => + columnsByName['layout_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get locale => + columnsByName['locale']! as i1.GeneratedColumn; + i1.GeneratedColumn get market => + columnsByName['market']! as i1.GeneratedColumn; + i1.GeneratedColumn get searchMode => + columnsByName['search_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get downloadLocation => + columnsByName['download_location']! as i1.GeneratedColumn; + i1.GeneratedColumn get localLibraryLocation => + columnsByName['local_library_location']! as i1.GeneratedColumn; + i1.GeneratedColumn get pipedInstance => + columnsByName['piped_instance']! as i1.GeneratedColumn; + i1.GeneratedColumn get invidiousInstance => + columnsByName['invidious_instance']! as i1.GeneratedColumn; + i1.GeneratedColumn get themeMode => + columnsByName['theme_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get audioSource => + columnsByName['audio_source']! as i1.GeneratedColumn; + i1.GeneratedColumn get youtubeClientEngine => + columnsByName['youtube_client_engine']! as i1.GeneratedColumn; + i1.GeneratedColumn get streamMusicCodec => + columnsByName['stream_music_codec']! as i1.GeneratedColumn; + i1.GeneratedColumn get downloadMusicCodec => + columnsByName['download_music_codec']! as i1.GeneratedColumn; + i1.GeneratedColumn get discordPresence => + columnsByName['discord_presence']! as i1.GeneratedColumn; + i1.GeneratedColumn get endlessPlayback => + columnsByName['endless_playback']! as i1.GeneratedColumn; + i1.GeneratedColumn get enableConnect => + columnsByName['enable_connect']! as i1.GeneratedColumn; + i1.GeneratedColumn get cacheMusic => + columnsByName['cache_music']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_54(String aliasedName) => + i1.GeneratedColumn('youtube_client_engine', aliasedName, false, + type: i1.DriftSqlType.string, + defaultValue: Constant(YoutubeClientEngine.youtubeExplode.name)); + +final class Schema5 extends i0.VersionedSchema { + Schema5({required super.database}) : super(version: 5); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + playlistTable, + playlistMediaTable, + historyTable, + lyricsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape12 preferencesTable = Shape12( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_7, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_55, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_23, + _column_24, + _column_25, + _column_26, + _column_54, + _column_27, + _column_28, + _column_29, + _column_30, + _column_31, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape5 sourceMatchTable = Shape5( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_38, + _column_39, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape6 audioPlayerStateTable = Shape6( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape7 playlistTable = Shape7( + source: i0.VersionedTable( + entityName: 'playlist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_44, + _column_45, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape8 playlistMediaTable = Shape8( + source: i0.VersionedTable( + entityName: 'playlist_media_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_46, + _column_47, + _column_48, + _column_49, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); +} + +i1.GeneratedColumn _column_55(String aliasedName) => + i1.GeneratedColumn('accent_color_scheme', aliasedName, false, + type: i1.DriftSqlType.string, + defaultValue: const Constant("Orange:0xFFf97315")); + +final class Schema6 extends i0.VersionedSchema { + Schema6({required super.database}) : super(version: 6); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + playlistTable, + playlistMediaTable, + historyTable, + lyricsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape13 preferencesTable = Shape13( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_7, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_55, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_23, + _column_24, + _column_25, + _column_26, + _column_54, + _column_27, + _column_28, + _column_29, + _column_30, + _column_31, + _column_56, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape5 sourceMatchTable = Shape5( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_38, + _column_39, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape6 audioPlayerStateTable = Shape6( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape7 playlistTable = Shape7( + source: i0.VersionedTable( + entityName: 'playlist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_44, + _column_45, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape8 playlistMediaTable = Shape8( + source: i0.VersionedTable( + entityName: 'playlist_media_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_46, + _column_47, + _column_48, + _column_49, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); +} + +class Shape13 extends i0.VersionedTable { + Shape13({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get audioQuality => + columnsByName['audio_quality']! as i1.GeneratedColumn; + i1.GeneratedColumn get albumColorSync => + columnsByName['album_color_sync']! as i1.GeneratedColumn; + i1.GeneratedColumn get amoledDarkTheme => + columnsByName['amoled_dark_theme']! as i1.GeneratedColumn; + i1.GeneratedColumn get checkUpdate => + columnsByName['check_update']! as i1.GeneratedColumn; + i1.GeneratedColumn get normalizeAudio => + columnsByName['normalize_audio']! as i1.GeneratedColumn; + i1.GeneratedColumn get showSystemTrayIcon => + columnsByName['show_system_tray_icon']! as i1.GeneratedColumn; + i1.GeneratedColumn get systemTitleBar => + columnsByName['system_title_bar']! as i1.GeneratedColumn; + i1.GeneratedColumn get skipNonMusic => + columnsByName['skip_non_music']! as i1.GeneratedColumn; + i1.GeneratedColumn get closeBehavior => + columnsByName['close_behavior']! as i1.GeneratedColumn; + i1.GeneratedColumn get accentColorScheme => + columnsByName['accent_color_scheme']! as i1.GeneratedColumn; + i1.GeneratedColumn get layoutMode => + columnsByName['layout_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get locale => + columnsByName['locale']! as i1.GeneratedColumn; + i1.GeneratedColumn get market => + columnsByName['market']! as i1.GeneratedColumn; + i1.GeneratedColumn get searchMode => + columnsByName['search_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get downloadLocation => + columnsByName['download_location']! as i1.GeneratedColumn; + i1.GeneratedColumn get localLibraryLocation => + columnsByName['local_library_location']! as i1.GeneratedColumn; + i1.GeneratedColumn get pipedInstance => + columnsByName['piped_instance']! as i1.GeneratedColumn; + i1.GeneratedColumn get invidiousInstance => + columnsByName['invidious_instance']! as i1.GeneratedColumn; + i1.GeneratedColumn get themeMode => + columnsByName['theme_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get audioSource => + columnsByName['audio_source']! as i1.GeneratedColumn; + i1.GeneratedColumn get youtubeClientEngine => + columnsByName['youtube_client_engine']! as i1.GeneratedColumn; + i1.GeneratedColumn get streamMusicCodec => + columnsByName['stream_music_codec']! as i1.GeneratedColumn; + i1.GeneratedColumn get downloadMusicCodec => + columnsByName['download_music_codec']! as i1.GeneratedColumn; + i1.GeneratedColumn get discordPresence => + columnsByName['discord_presence']! as i1.GeneratedColumn; + i1.GeneratedColumn get endlessPlayback => + columnsByName['endless_playback']! as i1.GeneratedColumn; + i1.GeneratedColumn get enableConnect => + columnsByName['enable_connect']! as i1.GeneratedColumn; + i1.GeneratedColumn get connectPort => + columnsByName['connect_port']! as i1.GeneratedColumn; + i1.GeneratedColumn get cacheMusic => + columnsByName['cache_music']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_56(String aliasedName) => + i1.GeneratedColumn('connect_port', aliasedName, false, + type: i1.DriftSqlType.int, defaultValue: const Constant(-1)); + +final class Schema7 extends i0.VersionedSchema { + Schema7({required super.database}) : super(version: 7); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + historyTable, + lyricsTable, + metadataPluginsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape13 preferencesTable = Shape13( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_7, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_55, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_23, + _column_24, + _column_25, + _column_26, + _column_54, + _column_27, + _column_28, + _column_29, + _column_30, + _column_31, + _column_56, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape5 sourceMatchTable = Shape5( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_38, + _column_39, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape14 audioPlayerStateTable = Shape14( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + _column_57, + _column_58, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape15 metadataPluginsTable = Shape15( + source: i0.VersionedTable( + entityName: 'metadata_plugins_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_59, + _column_60, + _column_61, + _column_62, + _column_63, + _column_64, + _column_65, + _column_66, + _column_67, + _column_68, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); +} + +class Shape14 extends i0.VersionedTable { + Shape14({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get playing => + columnsByName['playing']! as i1.GeneratedColumn; + i1.GeneratedColumn get loopMode => + columnsByName['loop_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get shuffled => + columnsByName['shuffled']! as i1.GeneratedColumn; + i1.GeneratedColumn get collections => + columnsByName['collections']! as i1.GeneratedColumn; + i1.GeneratedColumn get tracks => + columnsByName['tracks']! as i1.GeneratedColumn; + i1.GeneratedColumn get currentIndex => + columnsByName['current_index']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_57(String aliasedName) => + i1.GeneratedColumn('tracks', aliasedName, false, + type: i1.DriftSqlType.string, defaultValue: const Constant("[]")); +i1.GeneratedColumn _column_58(String aliasedName) => + i1.GeneratedColumn('current_index', aliasedName, false, + type: i1.DriftSqlType.int, defaultValue: const Constant(0)); + +class Shape15 extends i0.VersionedTable { + Shape15({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get name => + columnsByName['name']! as i1.GeneratedColumn; + i1.GeneratedColumn get description => + columnsByName['description']! as i1.GeneratedColumn; + i1.GeneratedColumn get version => + columnsByName['version']! as i1.GeneratedColumn; + i1.GeneratedColumn get author => + columnsByName['author']! as i1.GeneratedColumn; + i1.GeneratedColumn get entryPoint => + columnsByName['entry_point']! as i1.GeneratedColumn; + i1.GeneratedColumn get apis => + columnsByName['apis']! as i1.GeneratedColumn; + i1.GeneratedColumn get abilities => + columnsByName['abilities']! as i1.GeneratedColumn; + i1.GeneratedColumn get selected => + columnsByName['selected']! as i1.GeneratedColumn; + i1.GeneratedColumn get repository => + columnsByName['repository']! as i1.GeneratedColumn; + i1.GeneratedColumn get pluginApiVersion => + columnsByName['plugin_api_version']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_59(String aliasedName) => + i1.GeneratedColumn('name', aliasedName, false, + additionalChecks: i1.GeneratedColumn.checkTextLength( + minTextLength: 1, maxTextLength: 50), + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_60(String aliasedName) => + i1.GeneratedColumn('description', aliasedName, false, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_61(String aliasedName) => + i1.GeneratedColumn('version', aliasedName, false, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_62(String aliasedName) => + i1.GeneratedColumn('author', aliasedName, false, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_63(String aliasedName) => + i1.GeneratedColumn('entry_point', aliasedName, false, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_64(String aliasedName) => + i1.GeneratedColumn('apis', aliasedName, false, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_65(String aliasedName) => + i1.GeneratedColumn('abilities', aliasedName, false, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_66(String aliasedName) => + i1.GeneratedColumn('selected', aliasedName, false, + type: i1.DriftSqlType.bool, + defaultConstraints: i1.GeneratedColumn.constraintIsAlways( + 'CHECK ("selected" IN (0, 1))'), + defaultValue: const Constant(false)); +i1.GeneratedColumn _column_67(String aliasedName) => + i1.GeneratedColumn('repository', aliasedName, true, + type: i1.DriftSqlType.string); +i1.GeneratedColumn _column_68(String aliasedName) => + i1.GeneratedColumn('plugin_api_version', aliasedName, false, + type: i1.DriftSqlType.string); + +final class Schema8 extends i0.VersionedSchema { + Schema8({required super.database}) : super(version: 8); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + historyTable, + lyricsTable, + metadataPluginsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape13 preferencesTable = Shape13( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_7, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_69, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_23, + _column_24, + _column_25, + _column_26, + _column_54, + _column_27, + _column_28, + _column_29, + _column_30, + _column_31, + _column_56, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape5 sourceMatchTable = Shape5( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_38, + _column_39, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape14 audioPlayerStateTable = Shape14( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + _column_57, + _column_58, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape15 metadataPluginsTable = Shape15( + source: i0.VersionedTable( + entityName: 'metadata_plugins_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_59, + _column_60, + _column_61, + _column_62, + _column_63, + _column_64, + _column_65, + _column_66, + _column_67, + _column_70, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); +} + +i1.GeneratedColumn _column_69(String aliasedName) => + i1.GeneratedColumn('accent_color_scheme', aliasedName, false, + type: i1.DriftSqlType.string, + defaultValue: const Constant("Slate:0xff64748b")); +i1.GeneratedColumn _column_70(String aliasedName) => + i1.GeneratedColumn('plugin_api_version', aliasedName, false, + type: i1.DriftSqlType.string, defaultValue: const Constant('1.0.0')); + +final class Schema9 extends i0.VersionedSchema { + Schema9({required super.database}) : super(version: 9); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + historyTable, + lyricsTable, + pluginsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape13 preferencesTable = Shape13( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_7, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_69, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_23, + _column_24, + _column_25, + _column_26, + _column_54, + _column_27, + _column_28, + _column_29, + _column_30, + _column_31, + _column_56, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape5 sourceMatchTable = Shape5( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_38, + _column_39, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape14 audioPlayerStateTable = Shape14( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + _column_57, + _column_58, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape16 pluginsTable = Shape16( + source: i0.VersionedTable( + entityName: 'plugins_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_59, + _column_60, + _column_61, + _column_62, + _column_63, + _column_64, + _column_65, + _column_71, + _column_72, + _column_67, + _column_73, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_id, source_type)'); +} + +class Shape16 extends i0.VersionedTable { + Shape16({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get name => + columnsByName['name']! as i1.GeneratedColumn; + i1.GeneratedColumn get description => + columnsByName['description']! as i1.GeneratedColumn; + i1.GeneratedColumn get version => + columnsByName['version']! as i1.GeneratedColumn; + i1.GeneratedColumn get author => + columnsByName['author']! as i1.GeneratedColumn; + i1.GeneratedColumn get entryPoint => + columnsByName['entry_point']! as i1.GeneratedColumn; + i1.GeneratedColumn get apis => + columnsByName['apis']! as i1.GeneratedColumn; + i1.GeneratedColumn get abilities => + columnsByName['abilities']! as i1.GeneratedColumn; + i1.GeneratedColumn get selectedForMetadata => + columnsByName['selected_for_metadata']! as i1.GeneratedColumn; + i1.GeneratedColumn get selectedForAudioSource => + columnsByName['selected_for_audio_source']! as i1.GeneratedColumn; + i1.GeneratedColumn get repository => + columnsByName['repository']! as i1.GeneratedColumn; + i1.GeneratedColumn get pluginApiVersion => + columnsByName['plugin_api_version']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_71(String aliasedName) => + i1.GeneratedColumn('selected_for_metadata', aliasedName, false, + type: i1.DriftSqlType.bool, + defaultConstraints: i1.GeneratedColumn.constraintIsAlways( + 'CHECK ("selected_for_metadata" IN (0, 1))'), + defaultValue: const Constant(false)); +i1.GeneratedColumn _column_72(String aliasedName) => + i1.GeneratedColumn('selected_for_audio_source', aliasedName, false, + type: i1.DriftSqlType.bool, + defaultConstraints: i1.GeneratedColumn.constraintIsAlways( + 'CHECK ("selected_for_audio_source" IN (0, 1))'), + defaultValue: const Constant(false)); +i1.GeneratedColumn _column_73(String aliasedName) => + i1.GeneratedColumn('plugin_api_version', aliasedName, false, + type: i1.DriftSqlType.string, defaultValue: const Constant('2.0.0')); + +final class Schema10 extends i0.VersionedSchema { + Schema10({required super.database}) : super(version: 10); + @override + late final List entities = [ + authenticationTable, + blacklistTable, + preferencesTable, + scrobblerTable, + skipSegmentTable, + sourceMatchTable, + audioPlayerStateTable, + historyTable, + lyricsTable, + pluginsTable, + uniqueBlacklist, + uniqTrackMatch, + ]; + late final Shape0 authenticationTable = Shape0( + source: i0.VersionedTable( + entityName: 'authentication_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_1, + _column_2, + _column_3, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape1 blacklistTable = Shape1( + source: i0.VersionedTable( + entityName: 'blacklist_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_4, + _column_5, + _column_6, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape17 preferencesTable = Shape17( + source: i0.VersionedTable( + entityName: 'preferences_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_8, + _column_9, + _column_10, + _column_11, + _column_12, + _column_13, + _column_14, + _column_15, + _column_69, + _column_17, + _column_18, + _column_19, + _column_20, + _column_21, + _column_22, + _column_25, + _column_74, + _column_54, + _column_29, + _column_30, + _column_31, + _column_56, + _column_53, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape3 scrobblerTable = Shape3( + source: i0.VersionedTable( + entityName: 'scrobbler_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_33, + _column_34, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape4 skipSegmentTable = Shape4( + source: i0.VersionedTable( + entityName: 'skip_segment_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_35, + _column_36, + _column_37, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape18 sourceMatchTable = Shape18( + source: i0.VersionedTable( + entityName: 'source_match_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_75, + _column_76, + _column_32, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape14 audioPlayerStateTable = Shape14( + source: i0.VersionedTable( + entityName: 'audio_player_state_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_40, + _column_41, + _column_42, + _column_43, + _column_57, + _column_58, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape9 historyTable = Shape9( + source: i0.VersionedTable( + entityName: 'history_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_32, + _column_50, + _column_51, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape10 lyricsTable = Shape10( + source: i0.VersionedTable( + entityName: 'lyrics_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_37, + _column_52, + ], + attachedDatabase: database, + ), + alias: null); + late final Shape16 pluginsTable = Shape16( + source: i0.VersionedTable( + entityName: 'plugins_table', + withoutRowId: false, + isStrict: false, + tableConstraints: [], + columns: [ + _column_0, + _column_59, + _column_60, + _column_61, + _column_62, + _column_63, + _column_64, + _column_65, + _column_71, + _column_72, + _column_67, + _column_73, + ], + attachedDatabase: database, + ), + alias: null); + final i1.Index uniqueBlacklist = i1.Index('unique_blacklist', + 'CREATE UNIQUE INDEX unique_blacklist ON blacklist_table (element_type, element_id)'); + final i1.Index uniqTrackMatch = i1.Index('uniq_track_match', + 'CREATE UNIQUE INDEX uniq_track_match ON source_match_table (track_id, source_info, source_type)'); +} + +class Shape17 extends i0.VersionedTable { + Shape17({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get albumColorSync => + columnsByName['album_color_sync']! as i1.GeneratedColumn; + i1.GeneratedColumn get amoledDarkTheme => + columnsByName['amoled_dark_theme']! as i1.GeneratedColumn; + i1.GeneratedColumn get checkUpdate => + columnsByName['check_update']! as i1.GeneratedColumn; + i1.GeneratedColumn get normalizeAudio => + columnsByName['normalize_audio']! as i1.GeneratedColumn; + i1.GeneratedColumn get showSystemTrayIcon => + columnsByName['show_system_tray_icon']! as i1.GeneratedColumn; + i1.GeneratedColumn get systemTitleBar => + columnsByName['system_title_bar']! as i1.GeneratedColumn; + i1.GeneratedColumn get skipNonMusic => + columnsByName['skip_non_music']! as i1.GeneratedColumn; + i1.GeneratedColumn get closeBehavior => + columnsByName['close_behavior']! as i1.GeneratedColumn; + i1.GeneratedColumn get accentColorScheme => + columnsByName['accent_color_scheme']! as i1.GeneratedColumn; + i1.GeneratedColumn get layoutMode => + columnsByName['layout_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get locale => + columnsByName['locale']! as i1.GeneratedColumn; + i1.GeneratedColumn get market => + columnsByName['market']! as i1.GeneratedColumn; + i1.GeneratedColumn get searchMode => + columnsByName['search_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get downloadLocation => + columnsByName['download_location']! as i1.GeneratedColumn; + i1.GeneratedColumn get localLibraryLocation => + columnsByName['local_library_location']! as i1.GeneratedColumn; + i1.GeneratedColumn get themeMode => + columnsByName['theme_mode']! as i1.GeneratedColumn; + i1.GeneratedColumn get audioSourceId => + columnsByName['audio_source_id']! as i1.GeneratedColumn; + i1.GeneratedColumn get youtubeClientEngine => + columnsByName['youtube_client_engine']! as i1.GeneratedColumn; + i1.GeneratedColumn get discordPresence => + columnsByName['discord_presence']! as i1.GeneratedColumn; + i1.GeneratedColumn get endlessPlayback => + columnsByName['endless_playback']! as i1.GeneratedColumn; + i1.GeneratedColumn get enableConnect => + columnsByName['enable_connect']! as i1.GeneratedColumn; + i1.GeneratedColumn get connectPort => + columnsByName['connect_port']! as i1.GeneratedColumn; + i1.GeneratedColumn get cacheMusic => + columnsByName['cache_music']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_74(String aliasedName) => + i1.GeneratedColumn('audio_source_id', aliasedName, true, + type: i1.DriftSqlType.string); + +class Shape18 extends i0.VersionedTable { + Shape18({required super.source, required super.alias}) : super.aliased(); + i1.GeneratedColumn get id => + columnsByName['id']! as i1.GeneratedColumn; + i1.GeneratedColumn get trackId => + columnsByName['track_id']! as i1.GeneratedColumn; + i1.GeneratedColumn get sourceInfo => + columnsByName['source_info']! as i1.GeneratedColumn; + i1.GeneratedColumn get sourceType => + columnsByName['source_type']! as i1.GeneratedColumn; + i1.GeneratedColumn get createdAt => + columnsByName['created_at']! as i1.GeneratedColumn; +} + +i1.GeneratedColumn _column_75(String aliasedName) => + i1.GeneratedColumn('source_info', aliasedName, false, + type: i1.DriftSqlType.string, defaultValue: const Constant("{}")); +i1.GeneratedColumn _column_76(String aliasedName) => + i1.GeneratedColumn('source_type', aliasedName, false, + type: i1.DriftSqlType.string); i0.MigrationStepWithVersion migrationSteps({ required Future Function(i1.Migrator m, Schema2 schema) from1To2, required Future Function(i1.Migrator m, Schema3 schema) from2To3, + required Future Function(i1.Migrator m, Schema4 schema) from3To4, + required Future Function(i1.Migrator m, Schema5 schema) from4To5, + required Future Function(i1.Migrator m, Schema6 schema) from5To6, + required Future Function(i1.Migrator m, Schema7 schema) from6To7, + required Future Function(i1.Migrator m, Schema8 schema) from7To8, + required Future Function(i1.Migrator m, Schema9 schema) from8To9, + required Future Function(i1.Migrator m, Schema10 schema) from9To10, }) { return (currentVersion, database) async { switch (currentVersion) { @@ -923,6 +2762,41 @@ i0.MigrationStepWithVersion migrationSteps({ final migrator = i1.Migrator(database, schema); await from2To3(migrator, schema); return 3; + case 3: + final schema = Schema4(database: database); + final migrator = i1.Migrator(database, schema); + await from3To4(migrator, schema); + return 4; + case 4: + final schema = Schema5(database: database); + final migrator = i1.Migrator(database, schema); + await from4To5(migrator, schema); + return 5; + case 5: + final schema = Schema6(database: database); + final migrator = i1.Migrator(database, schema); + await from5To6(migrator, schema); + return 6; + case 6: + final schema = Schema7(database: database); + final migrator = i1.Migrator(database, schema); + await from6To7(migrator, schema); + return 7; + case 7: + final schema = Schema8(database: database); + final migrator = i1.Migrator(database, schema); + await from7To8(migrator, schema); + return 8; + case 8: + final schema = Schema9(database: database); + final migrator = i1.Migrator(database, schema); + await from8To9(migrator, schema); + return 9; + case 9: + final schema = Schema10(database: database); + final migrator = i1.Migrator(database, schema); + await from9To10(migrator, schema); + return 10; default: throw ArgumentError.value('Unknown migration from $currentVersion'); } @@ -932,9 +2806,23 @@ i0.MigrationStepWithVersion migrationSteps({ i1.OnUpgrade stepByStep({ required Future Function(i1.Migrator m, Schema2 schema) from1To2, required Future Function(i1.Migrator m, Schema3 schema) from2To3, + required Future Function(i1.Migrator m, Schema4 schema) from3To4, + required Future Function(i1.Migrator m, Schema5 schema) from4To5, + required Future Function(i1.Migrator m, Schema6 schema) from5To6, + required Future Function(i1.Migrator m, Schema7 schema) from6To7, + required Future Function(i1.Migrator m, Schema8 schema) from7To8, + required Future Function(i1.Migrator m, Schema9 schema) from8To9, + required Future Function(i1.Migrator m, Schema10 schema) from9To10, }) => i0.VersionedSchema.stepByStepHelper( step: migrationSteps( from1To2: from1To2, from2To3: from2To3, + from3To4: from3To4, + from4To5: from4To5, + from5To6: from5To6, + from6To7: from6To7, + from7To8: from7To8, + from8To9: from8To9, + from9To10: from9To10, )); diff --git a/lib/models/database/tables/audio_player_state.dart b/lib/models/database/tables/audio_player_state.dart index 3e49cf6f..bd570da7 100644 --- a/lib/models/database/tables/audio_player_state.dart +++ b/lib/models/database/tables/audio_player_state.dart @@ -6,22 +6,29 @@ class AudioPlayerStateTable extends Table { TextColumn get loopMode => textEnum()(); BoolColumn get shuffled => boolean()(); TextColumn get collections => text().map(const StringListConverter())(); + TextColumn get tracks => text() + .map(const SpotubeTrackObjectListConverter()) + .withDefault(const Constant("[]"))(); + IntColumn get currentIndex => integer().withDefault(const Constant(0))(); } -class PlaylistTable extends Table { - IntColumn get id => integer().autoIncrement()(); - IntColumn get audioPlayerStateId => - integer().references(AudioPlayerStateTable, #id)(); - IntColumn get index => integer()(); -} +class SpotubeTrackObjectListConverter + extends TypeConverter, String> { + const SpotubeTrackObjectListConverter(); -class PlaylistMediaTable extends Table { - IntColumn get id => integer().autoIncrement()(); - IntColumn get playlistId => integer().references(PlaylistTable, #id)(); + @override + List fromSql(String fromDb) { + final raw = (jsonDecode(fromDb) as List).cast(); - TextColumn get uri => text()(); - TextColumn get extras => - text().nullable().map(const MapTypeConverter())(); - TextColumn get httpHeaders => - text().nullable().map(const MapTypeConverter())(); + return raw + .map((e) => SpotubeTrackObject.fromJson(e.cast())) + .toList(); + } + + @override + String toSql(List value) { + return jsonEncode( + value.map((e) => e.toJson()).toList(), + ); + } } diff --git a/lib/models/database/tables/history.dart b/lib/models/database/tables/history.dart index 23c16f17..f074e248 100644 --- a/lib/models/database/tables/history.dart +++ b/lib/models/database/tables/history.dart @@ -16,10 +16,16 @@ class HistoryTable extends Table { } extension HistoryItemParseExtension on HistoryTableData { - PlaylistSimple? get playlist => - type == HistoryEntryType.playlist ? PlaylistSimple.fromJson(data) : null; - AlbumSimple? get album => - type == HistoryEntryType.album ? AlbumSimple.fromJson(data) : null; - Track? get track => - type == HistoryEntryType.track ? Track.fromJson(data) : null; + SpotubeSimplePlaylistObject? get playlist => + type == HistoryEntryType.playlist && !data.containsKey("external_urls") + ? SpotubeSimplePlaylistObject.fromJson(data) + : null; + SpotubeSimpleAlbumObject? get album => + type == HistoryEntryType.album && !data.containsKey("external_urls") + ? SpotubeSimpleAlbumObject.fromJson(data) + : null; + SpotubeTrackObject? get track => + type == HistoryEntryType.track && !data.containsKey("external_urls") + ? SpotubeTrackObject.fromJson(data) + : null; } diff --git a/lib/models/database/tables/metadata_plugins.dart b/lib/models/database/tables/metadata_plugins.dart new file mode 100644 index 00000000..3447497d --- /dev/null +++ b/lib/models/database/tables/metadata_plugins.dart @@ -0,0 +1,19 @@ +part of '../database.dart'; + +class PluginsTable extends Table { + IntColumn get id => integer().autoIncrement()(); + TextColumn get name => text().withLength(min: 1, max: 50)(); + TextColumn get description => text()(); + TextColumn get version => text()(); + TextColumn get author => text()(); + TextColumn get entryPoint => text()(); + TextColumn get apis => text().map(const StringListConverter())(); + TextColumn get abilities => text().map(const StringListConverter())(); + BoolColumn get selectedForMetadata => + boolean().withDefault(const Constant(false))(); + BoolColumn get selectedForAudioSource => + boolean().withDefault(const Constant(false))(); + TextColumn get repository => text().nullable()(); + TextColumn get pluginApiVersion => + text().withDefault(const Constant('2.0.0'))(); +} diff --git a/lib/models/database/tables/preferences.dart b/lib/models/database/tables/preferences.dart index c3904c84..3029e2a8 100644 --- a/lib/models/database/tables/preferences.dart +++ b/lib/models/database/tables/preferences.dart @@ -11,21 +11,23 @@ enum CloseBehavior { close, } -enum AudioSource { - youtube, - piped, - jiosaavn, - invidious; - - String get label => name[0].toUpperCase() + name.substring(1); -} - -enum MusicCodec { - m4a._("M4a (Best for downloaded music)"), - weba._("WebA (Best for streamed music)\nDoesn't support audio metadata"); +enum YoutubeClientEngine { + ytDlp("yt-dlp"), + youtubeExplode("YouTubeExplode"), + newPipe("NewPipe"); final String label; - const MusicCodec._(this.label); + + const YoutubeClientEngine(this.label); + + bool isAvailableForPlatform() { + return switch (this) { + YoutubeClientEngine.youtubeExplode => + YouTubeExplodeEngine.isAvailableForPlatform, + YoutubeClientEngine.ytDlp => YtDlpEngine.isAvailableForPlatform, + YoutubeClientEngine.newPipe => NewPipeEngine.isAvailableForPlatform, + }; + } } enum SearchMode { @@ -43,8 +45,6 @@ enum SearchMode { class PreferencesTable extends Table { IntColumn get id => integer().autoIncrement()(); - TextColumn get audioQuality => textEnum() - .withDefault(Constant(SourceQualities.high.name))(); BoolColumn get albumColorSync => boolean().withDefault(const Constant(true))(); BoolColumn get amoledDarkTheme => @@ -60,7 +60,7 @@ class PreferencesTable extends Table { TextColumn get closeBehavior => textEnum() .withDefault(Constant(CloseBehavior.close.name))(); TextColumn get accentColorScheme => text() - .withDefault(const Constant("Blue:0xFF2196F3")) + .withDefault(const Constant("Slate:0xff64748b")) .map(const SpotubeColorConverter())(); TextColumn get layoutMode => textEnum().withDefault(Constant(LayoutMode.adaptive.name))(); @@ -76,31 +76,24 @@ class PreferencesTable extends Table { TextColumn get downloadLocation => text().withDefault(const Constant(""))(); TextColumn get localLibraryLocation => text().withDefault(const Constant("")).map(const StringListConverter())(); - TextColumn get pipedInstance => - text().withDefault(const Constant("https://pipedapi.kavin.rocks"))(); - TextColumn get invidiousInstance => - text().withDefault(const Constant("https://inv.nadeko.net"))(); TextColumn get themeMode => textEnum().withDefault(Constant(ThemeMode.system.name))(); - TextColumn get audioSource => - textEnum().withDefault(Constant(AudioSource.youtube.name))(); - TextColumn get streamMusicCodec => - textEnum().withDefault(Constant(SourceCodecs.weba.name))(); - TextColumn get downloadMusicCodec => - textEnum().withDefault(Constant(SourceCodecs.m4a.name))(); + TextColumn get audioSourceId => text().nullable()(); + TextColumn get youtubeClientEngine => textEnum() + .withDefault(Constant(YoutubeClientEngine.youtubeExplode.name))(); BoolColumn get discordPresence => boolean().withDefault(const Constant(true))(); BoolColumn get endlessPlayback => boolean().withDefault(const Constant(true))(); BoolColumn get enableConnect => boolean().withDefault(const Constant(false))(); + IntColumn get connectPort => integer().withDefault(const Constant(-1))(); BoolColumn get cacheMusic => boolean().withDefault(const Constant(true))(); // Default values as PreferencesTableData static PreferencesTableData defaults() { return PreferencesTableData( id: 0, - audioQuality: SourceQualities.high, albumColorSync: true, amoledDarkTheme: false, checkUpdate: true, @@ -109,23 +102,23 @@ class PreferencesTable extends Table { systemTitleBar: false, skipNonMusic: false, closeBehavior: CloseBehavior.close, - accentColorScheme: SpotubeColor(Colors.blue.value, name: "Blue"), + accentColorScheme: SpotubeColor(Colors.slate.value, name: "Slate"), layoutMode: LayoutMode.adaptive, locale: const Locale("system", "system"), market: Market.US, searchMode: SearchMode.youtube, downloadLocation: "", localLibraryLocation: [], - pipedInstance: "https://pipedapi.kavin.rocks", - invidiousInstance: "https://inv.nadeko.net", themeMode: ThemeMode.system, - audioSource: AudioSource.youtube, - streamMusicCodec: SourceCodecs.m4a, - downloadMusicCodec: SourceCodecs.m4a, + audioSourceId: null, + youtubeClientEngine: kIsIOS + ? YoutubeClientEngine.youtubeExplode + : YoutubeClientEngine.newPipe, discordPresence: true, endlessPlayback: true, enableConnect: false, cacheMusic: true, + connectPort: -1, ); } } diff --git a/lib/models/database/tables/source_match.dart b/lib/models/database/tables/source_match.dart index 78d0eb05..66a4959c 100644 --- a/lib/models/database/tables/source_match.dart +++ b/lib/models/database/tables/source_match.dart @@ -1,25 +1,9 @@ part of '../database.dart'; -enum SourceType { - youtube._("YouTube"), - youtubeMusic._("YouTube Music"), - jiosaavn._("JioSaavn"); - - final String label; - - const SourceType._(this.label); -} - -@TableIndex( - name: "uniq_track_match", - columns: {#trackId, #sourceId, #sourceType}, - unique: true, -) class SourceMatchTable extends Table { IntColumn get id => integer().autoIncrement()(); TextColumn get trackId => text()(); - TextColumn get sourceId => text()(); - TextColumn get sourceType => - textEnum().withDefault(Constant(SourceType.youtube.name))(); + TextColumn get sourceInfo => text().withDefault(const Constant("{}"))(); + TextColumn get sourceType => text()(); DateTimeColumn get createdAt => dateTime().withDefault(currentDateAndTime)(); } diff --git a/lib/models/database/typeconverters/color.dart b/lib/models/database/typeconverters/color.dart index 70c27374..513921a2 100644 --- a/lib/models/database/typeconverters/color.dart +++ b/lib/models/database/typeconverters/color.dart @@ -10,7 +10,7 @@ class ColorConverter extends TypeConverter { @override int toSql(Color value) { - return value.value; + return value.toARGB32(); } } diff --git a/lib/models/database/typeconverters/map_list.dart b/lib/models/database/typeconverters/map_list.dart new file mode 100644 index 00000000..b92e781d --- /dev/null +++ b/lib/models/database/typeconverters/map_list.dart @@ -0,0 +1,20 @@ +part of '../database.dart'; + +class MapListConverter + extends TypeConverter>, String> { + const MapListConverter(); + + @override + List> fromSql(String fromDb) { + return fromDb + .split(",") + .where((e) => e.isNotEmpty) + .map((e) => json.decode(e) as Map) + .toList(); + } + + @override + String toSql(List> value) { + return value.map((e) => json.encode(e)).join(","); + } +} diff --git a/lib/models/local_track.dart b/lib/models/local_track.dart deleted file mode 100644 index def3b64f..00000000 --- a/lib/models/local_track.dart +++ /dev/null @@ -1,44 +0,0 @@ -import 'package:spotify/spotify.dart'; - -class LocalTrack extends Track { - final String path; - - LocalTrack.fromTrack({ - required Track track, - required this.path, - }) : super() { - album = track.album; - artists = track.artists; - availableMarkets = track.availableMarkets; - discNumber = track.discNumber; - durationMs = track.durationMs; - explicit = track.explicit; - externalIds = track.externalIds; - externalUrls = track.externalUrls; - href = track.href; - id = track.id; - isPlayable = track.isPlayable; - linkedFrom = track.linkedFrom; - name = track.name; - popularity = track.popularity; - previewUrl = track.previewUrl; - trackNumber = track.trackNumber; - type = track.type; - uri = track.uri; - } - - factory LocalTrack.fromJson(Map json) { - return LocalTrack.fromTrack( - track: Track.fromJson(json), - path: json['path'], - ); - } - - @override - Map toJson() { - return { - ...super.toJson(), - 'path': path, - }; - } -} diff --git a/lib/models/metadata/album.dart b/lib/models/metadata/album.dart new file mode 100644 index 00000000..bc9022de --- /dev/null +++ b/lib/models/metadata/album.dart @@ -0,0 +1,42 @@ +part of 'metadata.dart'; + +enum SpotubeAlbumType { + album, + single, + compilation, +} + +@freezed +class SpotubeFullAlbumObject with _$SpotubeFullAlbumObject { + factory SpotubeFullAlbumObject({ + required String id, + required String name, + required List artists, + @Default([]) List images, + required String releaseDate, + required String externalUri, + required int totalTracks, + required SpotubeAlbumType albumType, + String? recordLabel, + List? genres, + }) = _SpotubeFullAlbumObject; + + factory SpotubeFullAlbumObject.fromJson(Map json) => + _$SpotubeFullAlbumObjectFromJson(json); +} + +@freezed +class SpotubeSimpleAlbumObject with _$SpotubeSimpleAlbumObject { + factory SpotubeSimpleAlbumObject({ + required String id, + required String name, + required String externalUri, + required List artists, + @Default([]) List images, + required SpotubeAlbumType albumType, + String? releaseDate, + }) = _SpotubeSimpleAlbumObject; + + factory SpotubeSimpleAlbumObject.fromJson(Map json) => + _$SpotubeSimpleAlbumObjectFromJson(json); +} diff --git a/lib/models/metadata/artist.dart b/lib/models/metadata/artist.dart new file mode 100644 index 00000000..24d8f55c --- /dev/null +++ b/lib/models/metadata/artist.dart @@ -0,0 +1,41 @@ +part of 'metadata.dart'; + +@freezed +class SpotubeFullArtistObject with _$SpotubeFullArtistObject { + factory SpotubeFullArtistObject({ + required String id, + required String name, + required String externalUri, + @Default([]) List images, + List? genres, + int? followers, + }) = _SpotubeFullArtistObject; + + factory SpotubeFullArtistObject.fromJson(Map json) => + _$SpotubeFullArtistObjectFromJson(json); +} + +@freezed +class SpotubeSimpleArtistObject with _$SpotubeSimpleArtistObject { + factory SpotubeSimpleArtistObject({ + required String id, + required String name, + required String externalUri, + List? images, + }) = _SpotubeSimpleArtistObject; + + factory SpotubeSimpleArtistObject.fromJson(Map json) => + _$SpotubeSimpleArtistObjectFromJson(json); +} + +extension SpotubeFullArtistObjectAsString on List { + String asString() { + return map((e) => e.name).join(", "); + } +} + +extension SpotubeSimpleArtistObjectAsString on List { + String asString() { + return map((e) => e.name).join(", "); + } +} diff --git a/lib/models/metadata/audio_source.dart b/lib/models/metadata/audio_source.dart new file mode 100644 index 00000000..4fb790ea --- /dev/null +++ b/lib/models/metadata/audio_source.dart @@ -0,0 +1,110 @@ +part of 'metadata.dart'; + +final oneOptionalDecimalFormatter = NumberFormat('0.#', 'en_US'); + +enum SpotubeMediaCompressionType { + lossy, + lossless, +} + +@Freezed(unionKey: 'type') +class SpotubeAudioSourceContainerPreset + with _$SpotubeAudioSourceContainerPreset { + const SpotubeAudioSourceContainerPreset._(); + + @FreezedUnionValue("lossy") + factory SpotubeAudioSourceContainerPreset.lossy({ + required SpotubeMediaCompressionType type, + required String name, + required List qualities, + }) = SpotubeAudioSourceContainerPresetLossy; + + @FreezedUnionValue("lossless") + factory SpotubeAudioSourceContainerPreset.lossless({ + required SpotubeMediaCompressionType type, + required String name, + required List qualities, + }) = SpotubeAudioSourceContainerPresetLossless; + + factory SpotubeAudioSourceContainerPreset.fromJson( + Map json) => + _$SpotubeAudioSourceContainerPresetFromJson(json); + + String getFileExtension() { + return switch (name) { + "mp4" => "m4a", + "webm" => "weba", + _ => name, + }; + } +} + +@freezed +class SpotubeAudioLossyContainerQuality + with _$SpotubeAudioLossyContainerQuality { + const SpotubeAudioLossyContainerQuality._(); + + factory SpotubeAudioLossyContainerQuality({ + required int bitrate, // bits per second + }) = _SpotubeAudioLossyContainerQuality; + + factory SpotubeAudioLossyContainerQuality.fromJson( + Map json) => + _$SpotubeAudioLossyContainerQualityFromJson(json); + + @override + toString() { + return "${oneOptionalDecimalFormatter.format(bitrate / 1000)}kbps"; + } +} + +@freezed +class SpotubeAudioLosslessContainerQuality + with _$SpotubeAudioLosslessContainerQuality { + const SpotubeAudioLosslessContainerQuality._(); + + factory SpotubeAudioLosslessContainerQuality({ + required int bitDepth, // bit + required int sampleRate, // hz + }) = _SpotubeAudioLosslessContainerQuality; + + factory SpotubeAudioLosslessContainerQuality.fromJson( + Map json) => + _$SpotubeAudioLosslessContainerQualityFromJson(json); + + @override + toString() { + return "${bitDepth}bit • ${oneOptionalDecimalFormatter.format(sampleRate / 1000)}kHz"; + } +} + +@freezed +class SpotubeAudioSourceMatchObject with _$SpotubeAudioSourceMatchObject { + factory SpotubeAudioSourceMatchObject({ + required String id, + required String title, + required List artists, + required Duration duration, + String? thumbnail, + required String externalUri, + }) = _SpotubeAudioSourceMatchObject; + + factory SpotubeAudioSourceMatchObject.fromJson(Map json) => + _$SpotubeAudioSourceMatchObjectFromJson(json); +} + +@freezed +class SpotubeAudioSourceStreamObject with _$SpotubeAudioSourceStreamObject { + factory SpotubeAudioSourceStreamObject({ + required String url, + required String container, + required SpotubeMediaCompressionType type, + String? codec, + double? bitrate, + int? bitDepth, + double? sampleRate, + }) = _SpotubeAudioSourceStreamObject; + + factory SpotubeAudioSourceStreamObject.fromJson(Map json) => + _$SpotubeAudioSourceStreamObjectFromJson(json); +} diff --git a/lib/models/metadata/browse.dart b/lib/models/metadata/browse.dart new file mode 100644 index 00000000..e2a69181 --- /dev/null +++ b/lib/models/metadata/browse.dart @@ -0,0 +1,21 @@ +part of 'metadata.dart'; + +@Freezed(genericArgumentFactories: true) +class SpotubeBrowseSectionObject with _$SpotubeBrowseSectionObject { + factory SpotubeBrowseSectionObject({ + required String id, + required String title, + required String externalUri, + required bool browseMore, + required List items, + }) = _SpotubeBrowseSectionObject; + + factory SpotubeBrowseSectionObject.fromJson( + Map json, + T Function(Map json) fromJsonT, + ) => + _$SpotubeBrowseSectionObjectFromJson( + json, + (json) => fromJsonT(json as Map), + ); +} diff --git a/lib/models/metadata/fields.dart b/lib/models/metadata/fields.dart new file mode 100644 index 00000000..11d6656d --- /dev/null +++ b/lib/models/metadata/fields.dart @@ -0,0 +1,26 @@ +part of 'metadata.dart'; + +enum FormFieldVariant { text, password, number } + +@Freezed(unionKey: 'objectType') +class MetadataFormFieldObject with _$MetadataFormFieldObject { + @FreezedUnionValue("input") + factory MetadataFormFieldObject.input({ + required String objectType, + required String id, + @Default(FormFieldVariant.text) FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex, + }) = MetadataFormFieldInputObject; + + @FreezedUnionValue("text") + factory MetadataFormFieldObject.text({ + required String objectType, + required String text, + }) = MetadataFormFieldTextObject; + + factory MetadataFormFieldObject.fromJson(Map json) => + _$MetadataFormFieldObjectFromJson(json); +} diff --git a/lib/models/metadata/image.dart b/lib/models/metadata/image.dart new file mode 100644 index 00000000..2ee0f748 --- /dev/null +++ b/lib/models/metadata/image.dart @@ -0,0 +1,94 @@ +part of 'metadata.dart'; + +@freezed +class SpotubeImageObject with _$SpotubeImageObject { + factory SpotubeImageObject({ + required String url, + int? width, + int? height, + }) = _SpotubeImageObject; + + factory SpotubeImageObject.fromJson(Map json) => + _$SpotubeImageObjectFromJson(json); +} + +enum ImagePlaceholder { + albumArt, + artist, + collection, + online, +} + +final placeholderUrlMap = { + ImagePlaceholder.albumArt: Assets.images.albumPlaceholder.path, + ImagePlaceholder.artist: Assets.images.userPlaceholder.path, + ImagePlaceholder.collection: Assets.images.placeholder.path, + ImagePlaceholder.online: + "https://avatars.dicebear.com/api/bottts/${PrimitiveUtils.uuid.v4()}.png", +}; + +extension SpotubeImageExtensions on List? { + /// Returns the URL of the image at the specified index. + String asUrlString({ + int index = 1, + required ImagePlaceholder placeholder, + }) { + final sortedImage = this?.sorted((a, b) => a.width!.compareTo(b.width!)); + + return sortedImage != null && sortedImage.isNotEmpty + ? sortedImage[ + index > sortedImage.length - 1 ? sortedImage.length - 1 : index] + .url + : placeholderUrlMap[placeholder]!; + } + + Uri asUri({ + int index = 1, + required ImagePlaceholder placeholder, + }) { + final url = asUrlString(placeholder: placeholder, index: index); + if (url.startsWith("http")) { + return Uri.parse(url); + } + return Uri.file(url); + } + + String smallest(ImagePlaceholder placeholder) { + final sortedImage = this?.sorted((a, b) { + final widthComparison = (a.width ?? 0).compareTo(b.width ?? 0); + if (widthComparison != 0) return widthComparison; + return (a.height ?? 0).compareTo(b.height ?? 0); + }); + + return sortedImage != null && sortedImage.isNotEmpty + ? sortedImage.first.url + : placeholderUrlMap[placeholder]!; + } + + String from200PxTo300PxOrSmallestImage([ + ImagePlaceholder placeholder = ImagePlaceholder.albumArt, + ]) { + final placeholderUrl = placeholderUrlMap[placeholder]!; + + // Sort images by width and height to find the smallest one + final sortedImage = this?.sorted((a, b) { + final widthComparison = (a.width ?? 0).compareTo(b.width ?? 0); + if (widthComparison != 0) return widthComparison; + return (a.height ?? 0).compareTo(b.height ?? 0); + }); + + return sortedImage != null && sortedImage.isNotEmpty + ? sortedImage.firstWhere( + (image) { + final width = image.width ?? 0; + final height = image.height ?? 0; + return width >= 200 && + height >= 200 && + width <= 300 && + height <= 300; + }, + orElse: () => sortedImage.first, + ).url + : placeholderUrl; + } +} diff --git a/lib/models/metadata/market.dart b/lib/models/metadata/market.dart new file mode 100644 index 00000000..caaef957 --- /dev/null +++ b/lib/models/metadata/market.dart @@ -0,0 +1,252 @@ +enum Market { + AD, + AE, + AF, + AG, + AI, + AL, + AM, + AO, + AQ, + AR, + AS, + AT, + AU, + AW, + AX, + AZ, + BA, + BB, + BD, + BE, + BF, + BG, + BH, + BI, + BJ, + BL, + BM, + BN, + BO, + BQ, + BR, + BS, + BT, + BV, + BW, + BY, + BZ, + CA, + CC, + CD, + CF, + CG, + CH, + CI, + CK, + CL, + CM, + CN, + CO, + CR, + CU, + CV, + CW, + CX, + CY, + CZ, + DE, + DJ, + DK, + DM, + DO, + DZ, + EC, + EE, + EG, + EH, + ER, + ES, + ET, + FI, + FJ, + FK, + FM, + FO, + FR, + GA, + GB, + GD, + GE, + GF, + GG, + GH, + GI, + GL, + GM, + GN, + GP, + GQ, + GR, + GS, + GT, + GU, + GW, + GY, + HK, + HM, + HN, + HR, + HT, + HU, + ID, + IE, + IL, + IM, + IN, + IO, + IQ, + IR, + IS, + IT, + JE, + JM, + JO, + JP, + KE, + KG, + KH, + KI, + KM, + KN, + KP, + KR, + KW, + KY, + KZ, + LA, + LB, + LC, + LI, + LK, + LR, + LS, + LT, + LU, + LV, + LY, + MA, + MC, + MD, + ME, + MF, + MG, + MH, + MK, + ML, + MM, + MN, + MO, + MP, + MQ, + MR, + MS, + MT, + MU, + MV, + MW, + MX, + MY, + MZ, + NA, + NC, + NE, + NF, + NG, + NI, + NL, + NO, + NP, + NR, + NU, + NZ, + OM, + PA, + PE, + PF, + PG, + PH, + PK, + PL, + PM, + PN, + PR, + PS, + PT, + PW, + PY, + QA, + RE, + RO, + RS, + RU, + RW, + SA, + SB, + SC, + SD, + SE, + SG, + SH, + SI, + SJ, + SK, + SL, + SM, + SN, + SO, + SR, + SS, + ST, + SV, + SX, + SY, + SZ, + TC, + TD, + TF, + TG, + TH, + TJ, + TK, + TL, + TM, + TN, + TO, + TR, + TT, + TV, + TW, + TZ, + UA, + UG, + UM, + US, + UY, + UZ, + VA, + VC, + VE, + VG, + VI, + VN, + VU, + WF, + WS, + XK, + YE, + YT, + ZA, + ZM, + ZW, +} diff --git a/lib/models/metadata/metadata.dart b/lib/models/metadata/metadata.dart new file mode 100644 index 00000000..e68bcd14 --- /dev/null +++ b/lib/models/metadata/metadata.dart @@ -0,0 +1,32 @@ +library metadata_objects; + +import 'dart:io'; +import 'dart:typed_data'; + +import 'package:collection/collection.dart'; +import 'package:freezed_annotation/freezed_annotation.dart'; +import 'package:intl/intl.dart'; +import 'package:metadata_god/metadata_god.dart'; +import 'package:mime/mime.dart'; +import 'package:path/path.dart'; +import 'package:spotube/collections/assets.gen.dart'; +import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/utils/primitive_utils.dart'; + +part 'metadata.g.dart'; +part 'metadata.freezed.dart'; + +part 'audio_source.dart'; +part 'album.dart'; +part 'artist.dart'; +part 'browse.dart'; +part 'fields.dart'; +part 'image.dart'; +part 'pagination.dart'; +part 'playlist.dart'; +part 'search.dart'; +part 'track.dart'; +part 'user.dart'; + +part 'plugin.dart'; +part 'repository.dart'; diff --git a/lib/models/metadata/metadata.freezed.dart b/lib/models/metadata/metadata.freezed.dart new file mode 100644 index 00000000..fee1cbc2 --- /dev/null +++ b/lib/models/metadata/metadata.freezed.dart @@ -0,0 +1,6774 @@ +// coverage:ignore-file +// GENERATED CODE - DO NOT MODIFY BY HAND +// ignore_for_file: type=lint +// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark + +part of 'metadata.dart'; + +// ************************************************************************** +// FreezedGenerator +// ************************************************************************** + +T _$identity(T value) => value; + +final _privateConstructorUsedError = UnsupportedError( + 'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); + +SpotubeAudioSourceContainerPreset _$SpotubeAudioSourceContainerPresetFromJson( + Map json) { + switch (json['type']) { + case 'lossy': + return SpotubeAudioSourceContainerPresetLossy.fromJson(json); + case 'lossless': + return SpotubeAudioSourceContainerPresetLossless.fromJson(json); + + default: + throw CheckedFromJsonException( + json, + 'type', + 'SpotubeAudioSourceContainerPreset', + 'Invalid union type "${json['type']}"!'); + } +} + +/// @nodoc +mixin _$SpotubeAudioSourceContainerPreset { + SpotubeMediaCompressionType get type => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + List get qualities => throw _privateConstructorUsedError; + @optionalTypeArgs + TResult when({ + required TResult Function(SpotubeMediaCompressionType type, String name, + List qualities) + lossy, + required TResult Function(SpotubeMediaCompressionType type, String name, + List qualities) + lossless, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossy, + TResult? Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossless, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult maybeWhen({ + TResult Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossy, + TResult Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossless, + required TResult orElse(), + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult map({ + required TResult Function(SpotubeAudioSourceContainerPresetLossy value) + lossy, + required TResult Function(SpotubeAudioSourceContainerPresetLossless value) + lossless, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(SpotubeAudioSourceContainerPresetLossy value)? lossy, + TResult? Function(SpotubeAudioSourceContainerPresetLossless value)? + lossless, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult maybeMap({ + TResult Function(SpotubeAudioSourceContainerPresetLossy value)? lossy, + TResult Function(SpotubeAudioSourceContainerPresetLossless value)? lossless, + required TResult orElse(), + }) => + throw _privateConstructorUsedError; + + /// Serializes this SpotubeAudioSourceContainerPreset to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeAudioSourceContainerPresetCopyWith + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeAudioSourceContainerPresetCopyWith<$Res> { + factory $SpotubeAudioSourceContainerPresetCopyWith( + SpotubeAudioSourceContainerPreset value, + $Res Function(SpotubeAudioSourceContainerPreset) then) = + _$SpotubeAudioSourceContainerPresetCopyWithImpl<$Res, + SpotubeAudioSourceContainerPreset>; + @useResult + $Res call({SpotubeMediaCompressionType type, String name}); +} + +/// @nodoc +class _$SpotubeAudioSourceContainerPresetCopyWithImpl<$Res, + $Val extends SpotubeAudioSourceContainerPreset> + implements $SpotubeAudioSourceContainerPresetCopyWith<$Res> { + _$SpotubeAudioSourceContainerPresetCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? type = null, + Object? name = null, + }) { + return _then(_value.copyWith( + type: null == type + ? _value.type + : type // ignore: cast_nullable_to_non_nullable + as SpotubeMediaCompressionType, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeAudioSourceContainerPresetLossyImplCopyWith<$Res> + implements $SpotubeAudioSourceContainerPresetCopyWith<$Res> { + factory _$$SpotubeAudioSourceContainerPresetLossyImplCopyWith( + _$SpotubeAudioSourceContainerPresetLossyImpl value, + $Res Function(_$SpotubeAudioSourceContainerPresetLossyImpl) then) = + __$$SpotubeAudioSourceContainerPresetLossyImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {SpotubeMediaCompressionType type, + String name, + List qualities}); +} + +/// @nodoc +class __$$SpotubeAudioSourceContainerPresetLossyImplCopyWithImpl<$Res> + extends _$SpotubeAudioSourceContainerPresetCopyWithImpl<$Res, + _$SpotubeAudioSourceContainerPresetLossyImpl> + implements _$$SpotubeAudioSourceContainerPresetLossyImplCopyWith<$Res> { + __$$SpotubeAudioSourceContainerPresetLossyImplCopyWithImpl( + _$SpotubeAudioSourceContainerPresetLossyImpl _value, + $Res Function(_$SpotubeAudioSourceContainerPresetLossyImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? type = null, + Object? name = null, + Object? qualities = null, + }) { + return _then(_$SpotubeAudioSourceContainerPresetLossyImpl( + type: null == type + ? _value.type + : type // ignore: cast_nullable_to_non_nullable + as SpotubeMediaCompressionType, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + qualities: null == qualities + ? _value._qualities + : qualities // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeAudioSourceContainerPresetLossyImpl + extends SpotubeAudioSourceContainerPresetLossy { + _$SpotubeAudioSourceContainerPresetLossyImpl( + {required this.type, + required this.name, + required final List qualities}) + : _qualities = qualities, + super._(); + + factory _$SpotubeAudioSourceContainerPresetLossyImpl.fromJson( + Map json) => + _$$SpotubeAudioSourceContainerPresetLossyImplFromJson(json); + + @override + final SpotubeMediaCompressionType type; + @override + final String name; + final List _qualities; + @override + List get qualities { + if (_qualities is EqualUnmodifiableListView) return _qualities; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_qualities); + } + + @override + String toString() { + return 'SpotubeAudioSourceContainerPreset.lossy(type: $type, name: $name, qualities: $qualities)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeAudioSourceContainerPresetLossyImpl && + (identical(other.type, type) || other.type == type) && + (identical(other.name, name) || other.name == name) && + const DeepCollectionEquality() + .equals(other._qualities, _qualities)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, type, name, const DeepCollectionEquality().hash(_qualities)); + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeAudioSourceContainerPresetLossyImplCopyWith< + _$SpotubeAudioSourceContainerPresetLossyImpl> + get copyWith => + __$$SpotubeAudioSourceContainerPresetLossyImplCopyWithImpl< + _$SpotubeAudioSourceContainerPresetLossyImpl>(this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function(SpotubeMediaCompressionType type, String name, + List qualities) + lossy, + required TResult Function(SpotubeMediaCompressionType type, String name, + List qualities) + lossless, + }) { + return lossy(type, name, qualities); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossy, + TResult? Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossless, + }) { + return lossy?.call(type, name, qualities); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossy, + TResult Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossless, + required TResult orElse(), + }) { + if (lossy != null) { + return lossy(type, name, qualities); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(SpotubeAudioSourceContainerPresetLossy value) + lossy, + required TResult Function(SpotubeAudioSourceContainerPresetLossless value) + lossless, + }) { + return lossy(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(SpotubeAudioSourceContainerPresetLossy value)? lossy, + TResult? Function(SpotubeAudioSourceContainerPresetLossless value)? + lossless, + }) { + return lossy?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(SpotubeAudioSourceContainerPresetLossy value)? lossy, + TResult Function(SpotubeAudioSourceContainerPresetLossless value)? lossless, + required TResult orElse(), + }) { + if (lossy != null) { + return lossy(this); + } + return orElse(); + } + + @override + Map toJson() { + return _$$SpotubeAudioSourceContainerPresetLossyImplToJson( + this, + ); + } +} + +abstract class SpotubeAudioSourceContainerPresetLossy + extends SpotubeAudioSourceContainerPreset { + factory SpotubeAudioSourceContainerPresetLossy( + {required final SpotubeMediaCompressionType type, + required final String name, + required final List qualities}) = + _$SpotubeAudioSourceContainerPresetLossyImpl; + SpotubeAudioSourceContainerPresetLossy._() : super._(); + + factory SpotubeAudioSourceContainerPresetLossy.fromJson( + Map json) = + _$SpotubeAudioSourceContainerPresetLossyImpl.fromJson; + + @override + SpotubeMediaCompressionType get type; + @override + String get name; + @override + List get qualities; + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeAudioSourceContainerPresetLossyImplCopyWith< + _$SpotubeAudioSourceContainerPresetLossyImpl> + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class _$$SpotubeAudioSourceContainerPresetLosslessImplCopyWith<$Res> + implements $SpotubeAudioSourceContainerPresetCopyWith<$Res> { + factory _$$SpotubeAudioSourceContainerPresetLosslessImplCopyWith( + _$SpotubeAudioSourceContainerPresetLosslessImpl value, + $Res Function(_$SpotubeAudioSourceContainerPresetLosslessImpl) then) = + __$$SpotubeAudioSourceContainerPresetLosslessImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {SpotubeMediaCompressionType type, + String name, + List qualities}); +} + +/// @nodoc +class __$$SpotubeAudioSourceContainerPresetLosslessImplCopyWithImpl<$Res> + extends _$SpotubeAudioSourceContainerPresetCopyWithImpl<$Res, + _$SpotubeAudioSourceContainerPresetLosslessImpl> + implements _$$SpotubeAudioSourceContainerPresetLosslessImplCopyWith<$Res> { + __$$SpotubeAudioSourceContainerPresetLosslessImplCopyWithImpl( + _$SpotubeAudioSourceContainerPresetLosslessImpl _value, + $Res Function(_$SpotubeAudioSourceContainerPresetLosslessImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? type = null, + Object? name = null, + Object? qualities = null, + }) { + return _then(_$SpotubeAudioSourceContainerPresetLosslessImpl( + type: null == type + ? _value.type + : type // ignore: cast_nullable_to_non_nullable + as SpotubeMediaCompressionType, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + qualities: null == qualities + ? _value._qualities + : qualities // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeAudioSourceContainerPresetLosslessImpl + extends SpotubeAudioSourceContainerPresetLossless { + _$SpotubeAudioSourceContainerPresetLosslessImpl( + {required this.type, + required this.name, + required final List qualities}) + : _qualities = qualities, + super._(); + + factory _$SpotubeAudioSourceContainerPresetLosslessImpl.fromJson( + Map json) => + _$$SpotubeAudioSourceContainerPresetLosslessImplFromJson(json); + + @override + final SpotubeMediaCompressionType type; + @override + final String name; + final List _qualities; + @override + List get qualities { + if (_qualities is EqualUnmodifiableListView) return _qualities; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_qualities); + } + + @override + String toString() { + return 'SpotubeAudioSourceContainerPreset.lossless(type: $type, name: $name, qualities: $qualities)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeAudioSourceContainerPresetLosslessImpl && + (identical(other.type, type) || other.type == type) && + (identical(other.name, name) || other.name == name) && + const DeepCollectionEquality() + .equals(other._qualities, _qualities)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, type, name, const DeepCollectionEquality().hash(_qualities)); + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeAudioSourceContainerPresetLosslessImplCopyWith< + _$SpotubeAudioSourceContainerPresetLosslessImpl> + get copyWith => + __$$SpotubeAudioSourceContainerPresetLosslessImplCopyWithImpl< + _$SpotubeAudioSourceContainerPresetLosslessImpl>( + this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function(SpotubeMediaCompressionType type, String name, + List qualities) + lossy, + required TResult Function(SpotubeMediaCompressionType type, String name, + List qualities) + lossless, + }) { + return lossless(type, name, qualities); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossy, + TResult? Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossless, + }) { + return lossless?.call(type, name, qualities); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossy, + TResult Function(SpotubeMediaCompressionType type, String name, + List qualities)? + lossless, + required TResult orElse(), + }) { + if (lossless != null) { + return lossless(type, name, qualities); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(SpotubeAudioSourceContainerPresetLossy value) + lossy, + required TResult Function(SpotubeAudioSourceContainerPresetLossless value) + lossless, + }) { + return lossless(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(SpotubeAudioSourceContainerPresetLossy value)? lossy, + TResult? Function(SpotubeAudioSourceContainerPresetLossless value)? + lossless, + }) { + return lossless?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(SpotubeAudioSourceContainerPresetLossy value)? lossy, + TResult Function(SpotubeAudioSourceContainerPresetLossless value)? lossless, + required TResult orElse(), + }) { + if (lossless != null) { + return lossless(this); + } + return orElse(); + } + + @override + Map toJson() { + return _$$SpotubeAudioSourceContainerPresetLosslessImplToJson( + this, + ); + } +} + +abstract class SpotubeAudioSourceContainerPresetLossless + extends SpotubeAudioSourceContainerPreset { + factory SpotubeAudioSourceContainerPresetLossless( + {required final SpotubeMediaCompressionType type, + required final String name, + required final List + qualities}) = _$SpotubeAudioSourceContainerPresetLosslessImpl; + SpotubeAudioSourceContainerPresetLossless._() : super._(); + + factory SpotubeAudioSourceContainerPresetLossless.fromJson( + Map json) = + _$SpotubeAudioSourceContainerPresetLosslessImpl.fromJson; + + @override + SpotubeMediaCompressionType get type; + @override + String get name; + @override + List get qualities; + + /// Create a copy of SpotubeAudioSourceContainerPreset + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeAudioSourceContainerPresetLosslessImplCopyWith< + _$SpotubeAudioSourceContainerPresetLosslessImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeAudioLossyContainerQuality _$SpotubeAudioLossyContainerQualityFromJson( + Map json) { + return _SpotubeAudioLossyContainerQuality.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeAudioLossyContainerQuality { + int get bitrate => throw _privateConstructorUsedError; + + /// Serializes this SpotubeAudioLossyContainerQuality to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeAudioLossyContainerQuality + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeAudioLossyContainerQualityCopyWith + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeAudioLossyContainerQualityCopyWith<$Res> { + factory $SpotubeAudioLossyContainerQualityCopyWith( + SpotubeAudioLossyContainerQuality value, + $Res Function(SpotubeAudioLossyContainerQuality) then) = + _$SpotubeAudioLossyContainerQualityCopyWithImpl<$Res, + SpotubeAudioLossyContainerQuality>; + @useResult + $Res call({int bitrate}); +} + +/// @nodoc +class _$SpotubeAudioLossyContainerQualityCopyWithImpl<$Res, + $Val extends SpotubeAudioLossyContainerQuality> + implements $SpotubeAudioLossyContainerQualityCopyWith<$Res> { + _$SpotubeAudioLossyContainerQualityCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeAudioLossyContainerQuality + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? bitrate = null, + }) { + return _then(_value.copyWith( + bitrate: null == bitrate + ? _value.bitrate + : bitrate // ignore: cast_nullable_to_non_nullable + as int, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeAudioLossyContainerQualityImplCopyWith<$Res> + implements $SpotubeAudioLossyContainerQualityCopyWith<$Res> { + factory _$$SpotubeAudioLossyContainerQualityImplCopyWith( + _$SpotubeAudioLossyContainerQualityImpl value, + $Res Function(_$SpotubeAudioLossyContainerQualityImpl) then) = + __$$SpotubeAudioLossyContainerQualityImplCopyWithImpl<$Res>; + @override + @useResult + $Res call({int bitrate}); +} + +/// @nodoc +class __$$SpotubeAudioLossyContainerQualityImplCopyWithImpl<$Res> + extends _$SpotubeAudioLossyContainerQualityCopyWithImpl<$Res, + _$SpotubeAudioLossyContainerQualityImpl> + implements _$$SpotubeAudioLossyContainerQualityImplCopyWith<$Res> { + __$$SpotubeAudioLossyContainerQualityImplCopyWithImpl( + _$SpotubeAudioLossyContainerQualityImpl _value, + $Res Function(_$SpotubeAudioLossyContainerQualityImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeAudioLossyContainerQuality + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? bitrate = null, + }) { + return _then(_$SpotubeAudioLossyContainerQualityImpl( + bitrate: null == bitrate + ? _value.bitrate + : bitrate // ignore: cast_nullable_to_non_nullable + as int, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeAudioLossyContainerQualityImpl + extends _SpotubeAudioLossyContainerQuality { + _$SpotubeAudioLossyContainerQualityImpl({required this.bitrate}) : super._(); + + factory _$SpotubeAudioLossyContainerQualityImpl.fromJson( + Map json) => + _$$SpotubeAudioLossyContainerQualityImplFromJson(json); + + @override + final int bitrate; + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeAudioLossyContainerQualityImpl && + (identical(other.bitrate, bitrate) || other.bitrate == bitrate)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, bitrate); + + /// Create a copy of SpotubeAudioLossyContainerQuality + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeAudioLossyContainerQualityImplCopyWith< + _$SpotubeAudioLossyContainerQualityImpl> + get copyWith => __$$SpotubeAudioLossyContainerQualityImplCopyWithImpl< + _$SpotubeAudioLossyContainerQualityImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeAudioLossyContainerQualityImplToJson( + this, + ); + } +} + +abstract class _SpotubeAudioLossyContainerQuality + extends SpotubeAudioLossyContainerQuality { + factory _SpotubeAudioLossyContainerQuality({required final int bitrate}) = + _$SpotubeAudioLossyContainerQualityImpl; + _SpotubeAudioLossyContainerQuality._() : super._(); + + factory _SpotubeAudioLossyContainerQuality.fromJson( + Map json) = + _$SpotubeAudioLossyContainerQualityImpl.fromJson; + + @override + int get bitrate; + + /// Create a copy of SpotubeAudioLossyContainerQuality + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeAudioLossyContainerQualityImplCopyWith< + _$SpotubeAudioLossyContainerQualityImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeAudioLosslessContainerQuality + _$SpotubeAudioLosslessContainerQualityFromJson(Map json) { + return _SpotubeAudioLosslessContainerQuality.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeAudioLosslessContainerQuality { + int get bitDepth => throw _privateConstructorUsedError; // bit + int get sampleRate => throw _privateConstructorUsedError; + + /// Serializes this SpotubeAudioLosslessContainerQuality to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeAudioLosslessContainerQuality + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeAudioLosslessContainerQualityCopyWith< + SpotubeAudioLosslessContainerQuality> + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeAudioLosslessContainerQualityCopyWith<$Res> { + factory $SpotubeAudioLosslessContainerQualityCopyWith( + SpotubeAudioLosslessContainerQuality value, + $Res Function(SpotubeAudioLosslessContainerQuality) then) = + _$SpotubeAudioLosslessContainerQualityCopyWithImpl<$Res, + SpotubeAudioLosslessContainerQuality>; + @useResult + $Res call({int bitDepth, int sampleRate}); +} + +/// @nodoc +class _$SpotubeAudioLosslessContainerQualityCopyWithImpl<$Res, + $Val extends SpotubeAudioLosslessContainerQuality> + implements $SpotubeAudioLosslessContainerQualityCopyWith<$Res> { + _$SpotubeAudioLosslessContainerQualityCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeAudioLosslessContainerQuality + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? bitDepth = null, + Object? sampleRate = null, + }) { + return _then(_value.copyWith( + bitDepth: null == bitDepth + ? _value.bitDepth + : bitDepth // ignore: cast_nullable_to_non_nullable + as int, + sampleRate: null == sampleRate + ? _value.sampleRate + : sampleRate // ignore: cast_nullable_to_non_nullable + as int, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeAudioLosslessContainerQualityImplCopyWith<$Res> + implements $SpotubeAudioLosslessContainerQualityCopyWith<$Res> { + factory _$$SpotubeAudioLosslessContainerQualityImplCopyWith( + _$SpotubeAudioLosslessContainerQualityImpl value, + $Res Function(_$SpotubeAudioLosslessContainerQualityImpl) then) = + __$$SpotubeAudioLosslessContainerQualityImplCopyWithImpl<$Res>; + @override + @useResult + $Res call({int bitDepth, int sampleRate}); +} + +/// @nodoc +class __$$SpotubeAudioLosslessContainerQualityImplCopyWithImpl<$Res> + extends _$SpotubeAudioLosslessContainerQualityCopyWithImpl<$Res, + _$SpotubeAudioLosslessContainerQualityImpl> + implements _$$SpotubeAudioLosslessContainerQualityImplCopyWith<$Res> { + __$$SpotubeAudioLosslessContainerQualityImplCopyWithImpl( + _$SpotubeAudioLosslessContainerQualityImpl _value, + $Res Function(_$SpotubeAudioLosslessContainerQualityImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeAudioLosslessContainerQuality + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? bitDepth = null, + Object? sampleRate = null, + }) { + return _then(_$SpotubeAudioLosslessContainerQualityImpl( + bitDepth: null == bitDepth + ? _value.bitDepth + : bitDepth // ignore: cast_nullable_to_non_nullable + as int, + sampleRate: null == sampleRate + ? _value.sampleRate + : sampleRate // ignore: cast_nullable_to_non_nullable + as int, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeAudioLosslessContainerQualityImpl + extends _SpotubeAudioLosslessContainerQuality { + _$SpotubeAudioLosslessContainerQualityImpl( + {required this.bitDepth, required this.sampleRate}) + : super._(); + + factory _$SpotubeAudioLosslessContainerQualityImpl.fromJson( + Map json) => + _$$SpotubeAudioLosslessContainerQualityImplFromJson(json); + + @override + final int bitDepth; +// bit + @override + final int sampleRate; + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeAudioLosslessContainerQualityImpl && + (identical(other.bitDepth, bitDepth) || + other.bitDepth == bitDepth) && + (identical(other.sampleRate, sampleRate) || + other.sampleRate == sampleRate)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, bitDepth, sampleRate); + + /// Create a copy of SpotubeAudioLosslessContainerQuality + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeAudioLosslessContainerQualityImplCopyWith< + _$SpotubeAudioLosslessContainerQualityImpl> + get copyWith => __$$SpotubeAudioLosslessContainerQualityImplCopyWithImpl< + _$SpotubeAudioLosslessContainerQualityImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeAudioLosslessContainerQualityImplToJson( + this, + ); + } +} + +abstract class _SpotubeAudioLosslessContainerQuality + extends SpotubeAudioLosslessContainerQuality { + factory _SpotubeAudioLosslessContainerQuality( + {required final int bitDepth, required final int sampleRate}) = + _$SpotubeAudioLosslessContainerQualityImpl; + _SpotubeAudioLosslessContainerQuality._() : super._(); + + factory _SpotubeAudioLosslessContainerQuality.fromJson( + Map json) = + _$SpotubeAudioLosslessContainerQualityImpl.fromJson; + + @override + int get bitDepth; // bit + @override + int get sampleRate; + + /// Create a copy of SpotubeAudioLosslessContainerQuality + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeAudioLosslessContainerQualityImplCopyWith< + _$SpotubeAudioLosslessContainerQualityImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeAudioSourceMatchObject _$SpotubeAudioSourceMatchObjectFromJson( + Map json) { + return _SpotubeAudioSourceMatchObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeAudioSourceMatchObject { + String get id => throw _privateConstructorUsedError; + String get title => throw _privateConstructorUsedError; + List get artists => throw _privateConstructorUsedError; + Duration get duration => throw _privateConstructorUsedError; + String? get thumbnail => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + + /// Serializes this SpotubeAudioSourceMatchObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeAudioSourceMatchObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeAudioSourceMatchObjectCopyWith + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeAudioSourceMatchObjectCopyWith<$Res> { + factory $SpotubeAudioSourceMatchObjectCopyWith( + SpotubeAudioSourceMatchObject value, + $Res Function(SpotubeAudioSourceMatchObject) then) = + _$SpotubeAudioSourceMatchObjectCopyWithImpl<$Res, + SpotubeAudioSourceMatchObject>; + @useResult + $Res call( + {String id, + String title, + List artists, + Duration duration, + String? thumbnail, + String externalUri}); +} + +/// @nodoc +class _$SpotubeAudioSourceMatchObjectCopyWithImpl<$Res, + $Val extends SpotubeAudioSourceMatchObject> + implements $SpotubeAudioSourceMatchObjectCopyWith<$Res> { + _$SpotubeAudioSourceMatchObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeAudioSourceMatchObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? title = null, + Object? artists = null, + Object? duration = null, + Object? thumbnail = freezed, + Object? externalUri = null, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + title: null == title + ? _value.title + : title // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value.artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + duration: null == duration + ? _value.duration + : duration // ignore: cast_nullable_to_non_nullable + as Duration, + thumbnail: freezed == thumbnail + ? _value.thumbnail + : thumbnail // ignore: cast_nullable_to_non_nullable + as String?, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeAudioSourceMatchObjectImplCopyWith<$Res> + implements $SpotubeAudioSourceMatchObjectCopyWith<$Res> { + factory _$$SpotubeAudioSourceMatchObjectImplCopyWith( + _$SpotubeAudioSourceMatchObjectImpl value, + $Res Function(_$SpotubeAudioSourceMatchObjectImpl) then) = + __$$SpotubeAudioSourceMatchObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String title, + List artists, + Duration duration, + String? thumbnail, + String externalUri}); +} + +/// @nodoc +class __$$SpotubeAudioSourceMatchObjectImplCopyWithImpl<$Res> + extends _$SpotubeAudioSourceMatchObjectCopyWithImpl<$Res, + _$SpotubeAudioSourceMatchObjectImpl> + implements _$$SpotubeAudioSourceMatchObjectImplCopyWith<$Res> { + __$$SpotubeAudioSourceMatchObjectImplCopyWithImpl( + _$SpotubeAudioSourceMatchObjectImpl _value, + $Res Function(_$SpotubeAudioSourceMatchObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeAudioSourceMatchObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? title = null, + Object? artists = null, + Object? duration = null, + Object? thumbnail = freezed, + Object? externalUri = null, + }) { + return _then(_$SpotubeAudioSourceMatchObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + title: null == title + ? _value.title + : title // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value._artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + duration: null == duration + ? _value.duration + : duration // ignore: cast_nullable_to_non_nullable + as Duration, + thumbnail: freezed == thumbnail + ? _value.thumbnail + : thumbnail // ignore: cast_nullable_to_non_nullable + as String?, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeAudioSourceMatchObjectImpl + implements _SpotubeAudioSourceMatchObject { + _$SpotubeAudioSourceMatchObjectImpl( + {required this.id, + required this.title, + required final List artists, + required this.duration, + this.thumbnail, + required this.externalUri}) + : _artists = artists; + + factory _$SpotubeAudioSourceMatchObjectImpl.fromJson( + Map json) => + _$$SpotubeAudioSourceMatchObjectImplFromJson(json); + + @override + final String id; + @override + final String title; + final List _artists; + @override + List get artists { + if (_artists is EqualUnmodifiableListView) return _artists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_artists); + } + + @override + final Duration duration; + @override + final String? thumbnail; + @override + final String externalUri; + + @override + String toString() { + return 'SpotubeAudioSourceMatchObject(id: $id, title: $title, artists: $artists, duration: $duration, thumbnail: $thumbnail, externalUri: $externalUri)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeAudioSourceMatchObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.title, title) || other.title == title) && + const DeepCollectionEquality().equals(other._artists, _artists) && + (identical(other.duration, duration) || + other.duration == duration) && + (identical(other.thumbnail, thumbnail) || + other.thumbnail == thumbnail) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + id, + title, + const DeepCollectionEquality().hash(_artists), + duration, + thumbnail, + externalUri); + + /// Create a copy of SpotubeAudioSourceMatchObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeAudioSourceMatchObjectImplCopyWith< + _$SpotubeAudioSourceMatchObjectImpl> + get copyWith => __$$SpotubeAudioSourceMatchObjectImplCopyWithImpl< + _$SpotubeAudioSourceMatchObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeAudioSourceMatchObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeAudioSourceMatchObject + implements SpotubeAudioSourceMatchObject { + factory _SpotubeAudioSourceMatchObject( + {required final String id, + required final String title, + required final List artists, + required final Duration duration, + final String? thumbnail, + required final String externalUri}) = _$SpotubeAudioSourceMatchObjectImpl; + + factory _SpotubeAudioSourceMatchObject.fromJson(Map json) = + _$SpotubeAudioSourceMatchObjectImpl.fromJson; + + @override + String get id; + @override + String get title; + @override + List get artists; + @override + Duration get duration; + @override + String? get thumbnail; + @override + String get externalUri; + + /// Create a copy of SpotubeAudioSourceMatchObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeAudioSourceMatchObjectImplCopyWith< + _$SpotubeAudioSourceMatchObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeAudioSourceStreamObject _$SpotubeAudioSourceStreamObjectFromJson( + Map json) { + return _SpotubeAudioSourceStreamObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeAudioSourceStreamObject { + String get url => throw _privateConstructorUsedError; + String get container => throw _privateConstructorUsedError; + SpotubeMediaCompressionType get type => throw _privateConstructorUsedError; + String? get codec => throw _privateConstructorUsedError; + double? get bitrate => throw _privateConstructorUsedError; + int? get bitDepth => throw _privateConstructorUsedError; + double? get sampleRate => throw _privateConstructorUsedError; + + /// Serializes this SpotubeAudioSourceStreamObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeAudioSourceStreamObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeAudioSourceStreamObjectCopyWith + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeAudioSourceStreamObjectCopyWith<$Res> { + factory $SpotubeAudioSourceStreamObjectCopyWith( + SpotubeAudioSourceStreamObject value, + $Res Function(SpotubeAudioSourceStreamObject) then) = + _$SpotubeAudioSourceStreamObjectCopyWithImpl<$Res, + SpotubeAudioSourceStreamObject>; + @useResult + $Res call( + {String url, + String container, + SpotubeMediaCompressionType type, + String? codec, + double? bitrate, + int? bitDepth, + double? sampleRate}); +} + +/// @nodoc +class _$SpotubeAudioSourceStreamObjectCopyWithImpl<$Res, + $Val extends SpotubeAudioSourceStreamObject> + implements $SpotubeAudioSourceStreamObjectCopyWith<$Res> { + _$SpotubeAudioSourceStreamObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeAudioSourceStreamObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? url = null, + Object? container = null, + Object? type = null, + Object? codec = freezed, + Object? bitrate = freezed, + Object? bitDepth = freezed, + Object? sampleRate = freezed, + }) { + return _then(_value.copyWith( + url: null == url + ? _value.url + : url // ignore: cast_nullable_to_non_nullable + as String, + container: null == container + ? _value.container + : container // ignore: cast_nullable_to_non_nullable + as String, + type: null == type + ? _value.type + : type // ignore: cast_nullable_to_non_nullable + as SpotubeMediaCompressionType, + codec: freezed == codec + ? _value.codec + : codec // ignore: cast_nullable_to_non_nullable + as String?, + bitrate: freezed == bitrate + ? _value.bitrate + : bitrate // ignore: cast_nullable_to_non_nullable + as double?, + bitDepth: freezed == bitDepth + ? _value.bitDepth + : bitDepth // ignore: cast_nullable_to_non_nullable + as int?, + sampleRate: freezed == sampleRate + ? _value.sampleRate + : sampleRate // ignore: cast_nullable_to_non_nullable + as double?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeAudioSourceStreamObjectImplCopyWith<$Res> + implements $SpotubeAudioSourceStreamObjectCopyWith<$Res> { + factory _$$SpotubeAudioSourceStreamObjectImplCopyWith( + _$SpotubeAudioSourceStreamObjectImpl value, + $Res Function(_$SpotubeAudioSourceStreamObjectImpl) then) = + __$$SpotubeAudioSourceStreamObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String url, + String container, + SpotubeMediaCompressionType type, + String? codec, + double? bitrate, + int? bitDepth, + double? sampleRate}); +} + +/// @nodoc +class __$$SpotubeAudioSourceStreamObjectImplCopyWithImpl<$Res> + extends _$SpotubeAudioSourceStreamObjectCopyWithImpl<$Res, + _$SpotubeAudioSourceStreamObjectImpl> + implements _$$SpotubeAudioSourceStreamObjectImplCopyWith<$Res> { + __$$SpotubeAudioSourceStreamObjectImplCopyWithImpl( + _$SpotubeAudioSourceStreamObjectImpl _value, + $Res Function(_$SpotubeAudioSourceStreamObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeAudioSourceStreamObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? url = null, + Object? container = null, + Object? type = null, + Object? codec = freezed, + Object? bitrate = freezed, + Object? bitDepth = freezed, + Object? sampleRate = freezed, + }) { + return _then(_$SpotubeAudioSourceStreamObjectImpl( + url: null == url + ? _value.url + : url // ignore: cast_nullable_to_non_nullable + as String, + container: null == container + ? _value.container + : container // ignore: cast_nullable_to_non_nullable + as String, + type: null == type + ? _value.type + : type // ignore: cast_nullable_to_non_nullable + as SpotubeMediaCompressionType, + codec: freezed == codec + ? _value.codec + : codec // ignore: cast_nullable_to_non_nullable + as String?, + bitrate: freezed == bitrate + ? _value.bitrate + : bitrate // ignore: cast_nullable_to_non_nullable + as double?, + bitDepth: freezed == bitDepth + ? _value.bitDepth + : bitDepth // ignore: cast_nullable_to_non_nullable + as int?, + sampleRate: freezed == sampleRate + ? _value.sampleRate + : sampleRate // ignore: cast_nullable_to_non_nullable + as double?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeAudioSourceStreamObjectImpl + implements _SpotubeAudioSourceStreamObject { + _$SpotubeAudioSourceStreamObjectImpl( + {required this.url, + required this.container, + required this.type, + this.codec, + this.bitrate, + this.bitDepth, + this.sampleRate}); + + factory _$SpotubeAudioSourceStreamObjectImpl.fromJson( + Map json) => + _$$SpotubeAudioSourceStreamObjectImplFromJson(json); + + @override + final String url; + @override + final String container; + @override + final SpotubeMediaCompressionType type; + @override + final String? codec; + @override + final double? bitrate; + @override + final int? bitDepth; + @override + final double? sampleRate; + + @override + String toString() { + return 'SpotubeAudioSourceStreamObject(url: $url, container: $container, type: $type, codec: $codec, bitrate: $bitrate, bitDepth: $bitDepth, sampleRate: $sampleRate)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeAudioSourceStreamObjectImpl && + (identical(other.url, url) || other.url == url) && + (identical(other.container, container) || + other.container == container) && + (identical(other.type, type) || other.type == type) && + (identical(other.codec, codec) || other.codec == codec) && + (identical(other.bitrate, bitrate) || other.bitrate == bitrate) && + (identical(other.bitDepth, bitDepth) || + other.bitDepth == bitDepth) && + (identical(other.sampleRate, sampleRate) || + other.sampleRate == sampleRate)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, url, container, type, codec, bitrate, bitDepth, sampleRate); + + /// Create a copy of SpotubeAudioSourceStreamObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeAudioSourceStreamObjectImplCopyWith< + _$SpotubeAudioSourceStreamObjectImpl> + get copyWith => __$$SpotubeAudioSourceStreamObjectImplCopyWithImpl< + _$SpotubeAudioSourceStreamObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeAudioSourceStreamObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeAudioSourceStreamObject + implements SpotubeAudioSourceStreamObject { + factory _SpotubeAudioSourceStreamObject( + {required final String url, + required final String container, + required final SpotubeMediaCompressionType type, + final String? codec, + final double? bitrate, + final int? bitDepth, + final double? sampleRate}) = _$SpotubeAudioSourceStreamObjectImpl; + + factory _SpotubeAudioSourceStreamObject.fromJson(Map json) = + _$SpotubeAudioSourceStreamObjectImpl.fromJson; + + @override + String get url; + @override + String get container; + @override + SpotubeMediaCompressionType get type; + @override + String? get codec; + @override + double? get bitrate; + @override + int? get bitDepth; + @override + double? get sampleRate; + + /// Create a copy of SpotubeAudioSourceStreamObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeAudioSourceStreamObjectImplCopyWith< + _$SpotubeAudioSourceStreamObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeFullAlbumObject _$SpotubeFullAlbumObjectFromJson( + Map json) { + return _SpotubeFullAlbumObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeFullAlbumObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + List get artists => + throw _privateConstructorUsedError; + List get images => throw _privateConstructorUsedError; + String get releaseDate => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + int get totalTracks => throw _privateConstructorUsedError; + SpotubeAlbumType get albumType => throw _privateConstructorUsedError; + String? get recordLabel => throw _privateConstructorUsedError; + List? get genres => throw _privateConstructorUsedError; + + /// Serializes this SpotubeFullAlbumObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeFullAlbumObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeFullAlbumObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeFullAlbumObjectCopyWith<$Res> { + factory $SpotubeFullAlbumObjectCopyWith(SpotubeFullAlbumObject value, + $Res Function(SpotubeFullAlbumObject) then) = + _$SpotubeFullAlbumObjectCopyWithImpl<$Res, SpotubeFullAlbumObject>; + @useResult + $Res call( + {String id, + String name, + List artists, + List images, + String releaseDate, + String externalUri, + int totalTracks, + SpotubeAlbumType albumType, + String? recordLabel, + List? genres}); +} + +/// @nodoc +class _$SpotubeFullAlbumObjectCopyWithImpl<$Res, + $Val extends SpotubeFullAlbumObject> + implements $SpotubeFullAlbumObjectCopyWith<$Res> { + _$SpotubeFullAlbumObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeFullAlbumObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? artists = null, + Object? images = null, + Object? releaseDate = null, + Object? externalUri = null, + Object? totalTracks = null, + Object? albumType = null, + Object? recordLabel = freezed, + Object? genres = freezed, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value.artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + images: null == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List, + releaseDate: null == releaseDate + ? _value.releaseDate + : releaseDate // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + totalTracks: null == totalTracks + ? _value.totalTracks + : totalTracks // ignore: cast_nullable_to_non_nullable + as int, + albumType: null == albumType + ? _value.albumType + : albumType // ignore: cast_nullable_to_non_nullable + as SpotubeAlbumType, + recordLabel: freezed == recordLabel + ? _value.recordLabel + : recordLabel // ignore: cast_nullable_to_non_nullable + as String?, + genres: freezed == genres + ? _value.genres + : genres // ignore: cast_nullable_to_non_nullable + as List?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeFullAlbumObjectImplCopyWith<$Res> + implements $SpotubeFullAlbumObjectCopyWith<$Res> { + factory _$$SpotubeFullAlbumObjectImplCopyWith( + _$SpotubeFullAlbumObjectImpl value, + $Res Function(_$SpotubeFullAlbumObjectImpl) then) = + __$$SpotubeFullAlbumObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + List artists, + List images, + String releaseDate, + String externalUri, + int totalTracks, + SpotubeAlbumType albumType, + String? recordLabel, + List? genres}); +} + +/// @nodoc +class __$$SpotubeFullAlbumObjectImplCopyWithImpl<$Res> + extends _$SpotubeFullAlbumObjectCopyWithImpl<$Res, + _$SpotubeFullAlbumObjectImpl> + implements _$$SpotubeFullAlbumObjectImplCopyWith<$Res> { + __$$SpotubeFullAlbumObjectImplCopyWithImpl( + _$SpotubeFullAlbumObjectImpl _value, + $Res Function(_$SpotubeFullAlbumObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeFullAlbumObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? artists = null, + Object? images = null, + Object? releaseDate = null, + Object? externalUri = null, + Object? totalTracks = null, + Object? albumType = null, + Object? recordLabel = freezed, + Object? genres = freezed, + }) { + return _then(_$SpotubeFullAlbumObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value._artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + images: null == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List, + releaseDate: null == releaseDate + ? _value.releaseDate + : releaseDate // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + totalTracks: null == totalTracks + ? _value.totalTracks + : totalTracks // ignore: cast_nullable_to_non_nullable + as int, + albumType: null == albumType + ? _value.albumType + : albumType // ignore: cast_nullable_to_non_nullable + as SpotubeAlbumType, + recordLabel: freezed == recordLabel + ? _value.recordLabel + : recordLabel // ignore: cast_nullable_to_non_nullable + as String?, + genres: freezed == genres + ? _value._genres + : genres // ignore: cast_nullable_to_non_nullable + as List?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeFullAlbumObjectImpl implements _SpotubeFullAlbumObject { + _$SpotubeFullAlbumObjectImpl( + {required this.id, + required this.name, + required final List artists, + final List images = const [], + required this.releaseDate, + required this.externalUri, + required this.totalTracks, + required this.albumType, + this.recordLabel, + final List? genres}) + : _artists = artists, + _images = images, + _genres = genres; + + factory _$SpotubeFullAlbumObjectImpl.fromJson(Map json) => + _$$SpotubeFullAlbumObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + final List _artists; + @override + List get artists { + if (_artists is EqualUnmodifiableListView) return _artists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_artists); + } + + final List _images; + @override + @JsonKey() + List get images { + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_images); + } + + @override + final String releaseDate; + @override + final String externalUri; + @override + final int totalTracks; + @override + final SpotubeAlbumType albumType; + @override + final String? recordLabel; + final List? _genres; + @override + List? get genres { + final value = _genres; + if (value == null) return null; + if (_genres is EqualUnmodifiableListView) return _genres; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(value); + } + + @override + String toString() { + return 'SpotubeFullAlbumObject(id: $id, name: $name, artists: $artists, images: $images, releaseDate: $releaseDate, externalUri: $externalUri, totalTracks: $totalTracks, albumType: $albumType, recordLabel: $recordLabel, genres: $genres)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeFullAlbumObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + const DeepCollectionEquality().equals(other._artists, _artists) && + const DeepCollectionEquality().equals(other._images, _images) && + (identical(other.releaseDate, releaseDate) || + other.releaseDate == releaseDate) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + (identical(other.totalTracks, totalTracks) || + other.totalTracks == totalTracks) && + (identical(other.albumType, albumType) || + other.albumType == albumType) && + (identical(other.recordLabel, recordLabel) || + other.recordLabel == recordLabel) && + const DeepCollectionEquality().equals(other._genres, _genres)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + id, + name, + const DeepCollectionEquality().hash(_artists), + const DeepCollectionEquality().hash(_images), + releaseDate, + externalUri, + totalTracks, + albumType, + recordLabel, + const DeepCollectionEquality().hash(_genres)); + + /// Create a copy of SpotubeFullAlbumObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeFullAlbumObjectImplCopyWith<_$SpotubeFullAlbumObjectImpl> + get copyWith => __$$SpotubeFullAlbumObjectImplCopyWithImpl< + _$SpotubeFullAlbumObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeFullAlbumObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeFullAlbumObject implements SpotubeFullAlbumObject { + factory _SpotubeFullAlbumObject( + {required final String id, + required final String name, + required final List artists, + final List images, + required final String releaseDate, + required final String externalUri, + required final int totalTracks, + required final SpotubeAlbumType albumType, + final String? recordLabel, + final List? genres}) = _$SpotubeFullAlbumObjectImpl; + + factory _SpotubeFullAlbumObject.fromJson(Map json) = + _$SpotubeFullAlbumObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + List get artists; + @override + List get images; + @override + String get releaseDate; + @override + String get externalUri; + @override + int get totalTracks; + @override + SpotubeAlbumType get albumType; + @override + String? get recordLabel; + @override + List? get genres; + + /// Create a copy of SpotubeFullAlbumObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeFullAlbumObjectImplCopyWith<_$SpotubeFullAlbumObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeSimpleAlbumObject _$SpotubeSimpleAlbumObjectFromJson( + Map json) { + return _SpotubeSimpleAlbumObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeSimpleAlbumObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + List get artists => + throw _privateConstructorUsedError; + List get images => throw _privateConstructorUsedError; + SpotubeAlbumType get albumType => throw _privateConstructorUsedError; + String? get releaseDate => throw _privateConstructorUsedError; + + /// Serializes this SpotubeSimpleAlbumObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeSimpleAlbumObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeSimpleAlbumObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeSimpleAlbumObjectCopyWith<$Res> { + factory $SpotubeSimpleAlbumObjectCopyWith(SpotubeSimpleAlbumObject value, + $Res Function(SpotubeSimpleAlbumObject) then) = + _$SpotubeSimpleAlbumObjectCopyWithImpl<$Res, SpotubeSimpleAlbumObject>; + @useResult + $Res call( + {String id, + String name, + String externalUri, + List artists, + List images, + SpotubeAlbumType albumType, + String? releaseDate}); +} + +/// @nodoc +class _$SpotubeSimpleAlbumObjectCopyWithImpl<$Res, + $Val extends SpotubeSimpleAlbumObject> + implements $SpotubeSimpleAlbumObjectCopyWith<$Res> { + _$SpotubeSimpleAlbumObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeSimpleAlbumObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? artists = null, + Object? images = null, + Object? albumType = null, + Object? releaseDate = freezed, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value.artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + images: null == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List, + albumType: null == albumType + ? _value.albumType + : albumType // ignore: cast_nullable_to_non_nullable + as SpotubeAlbumType, + releaseDate: freezed == releaseDate + ? _value.releaseDate + : releaseDate // ignore: cast_nullable_to_non_nullable + as String?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeSimpleAlbumObjectImplCopyWith<$Res> + implements $SpotubeSimpleAlbumObjectCopyWith<$Res> { + factory _$$SpotubeSimpleAlbumObjectImplCopyWith( + _$SpotubeSimpleAlbumObjectImpl value, + $Res Function(_$SpotubeSimpleAlbumObjectImpl) then) = + __$$SpotubeSimpleAlbumObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String externalUri, + List artists, + List images, + SpotubeAlbumType albumType, + String? releaseDate}); +} + +/// @nodoc +class __$$SpotubeSimpleAlbumObjectImplCopyWithImpl<$Res> + extends _$SpotubeSimpleAlbumObjectCopyWithImpl<$Res, + _$SpotubeSimpleAlbumObjectImpl> + implements _$$SpotubeSimpleAlbumObjectImplCopyWith<$Res> { + __$$SpotubeSimpleAlbumObjectImplCopyWithImpl( + _$SpotubeSimpleAlbumObjectImpl _value, + $Res Function(_$SpotubeSimpleAlbumObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeSimpleAlbumObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? artists = null, + Object? images = null, + Object? albumType = null, + Object? releaseDate = freezed, + }) { + return _then(_$SpotubeSimpleAlbumObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value._artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + images: null == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List, + albumType: null == albumType + ? _value.albumType + : albumType // ignore: cast_nullable_to_non_nullable + as SpotubeAlbumType, + releaseDate: freezed == releaseDate + ? _value.releaseDate + : releaseDate // ignore: cast_nullable_to_non_nullable + as String?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeSimpleAlbumObjectImpl implements _SpotubeSimpleAlbumObject { + _$SpotubeSimpleAlbumObjectImpl( + {required this.id, + required this.name, + required this.externalUri, + required final List artists, + final List images = const [], + required this.albumType, + this.releaseDate}) + : _artists = artists, + _images = images; + + factory _$SpotubeSimpleAlbumObjectImpl.fromJson(Map json) => + _$$SpotubeSimpleAlbumObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String externalUri; + final List _artists; + @override + List get artists { + if (_artists is EqualUnmodifiableListView) return _artists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_artists); + } + + final List _images; + @override + @JsonKey() + List get images { + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_images); + } + + @override + final SpotubeAlbumType albumType; + @override + final String? releaseDate; + + @override + String toString() { + return 'SpotubeSimpleAlbumObject(id: $id, name: $name, externalUri: $externalUri, artists: $artists, images: $images, albumType: $albumType, releaseDate: $releaseDate)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeSimpleAlbumObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + const DeepCollectionEquality().equals(other._artists, _artists) && + const DeepCollectionEquality().equals(other._images, _images) && + (identical(other.albumType, albumType) || + other.albumType == albumType) && + (identical(other.releaseDate, releaseDate) || + other.releaseDate == releaseDate)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + id, + name, + externalUri, + const DeepCollectionEquality().hash(_artists), + const DeepCollectionEquality().hash(_images), + albumType, + releaseDate); + + /// Create a copy of SpotubeSimpleAlbumObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeSimpleAlbumObjectImplCopyWith<_$SpotubeSimpleAlbumObjectImpl> + get copyWith => __$$SpotubeSimpleAlbumObjectImplCopyWithImpl< + _$SpotubeSimpleAlbumObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeSimpleAlbumObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeSimpleAlbumObject implements SpotubeSimpleAlbumObject { + factory _SpotubeSimpleAlbumObject( + {required final String id, + required final String name, + required final String externalUri, + required final List artists, + final List images, + required final SpotubeAlbumType albumType, + final String? releaseDate}) = _$SpotubeSimpleAlbumObjectImpl; + + factory _SpotubeSimpleAlbumObject.fromJson(Map json) = + _$SpotubeSimpleAlbumObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get externalUri; + @override + List get artists; + @override + List get images; + @override + SpotubeAlbumType get albumType; + @override + String? get releaseDate; + + /// Create a copy of SpotubeSimpleAlbumObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeSimpleAlbumObjectImplCopyWith<_$SpotubeSimpleAlbumObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeFullArtistObject _$SpotubeFullArtistObjectFromJson( + Map json) { + return _SpotubeFullArtistObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeFullArtistObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + List get images => throw _privateConstructorUsedError; + List? get genres => throw _privateConstructorUsedError; + int? get followers => throw _privateConstructorUsedError; + + /// Serializes this SpotubeFullArtistObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeFullArtistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeFullArtistObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeFullArtistObjectCopyWith<$Res> { + factory $SpotubeFullArtistObjectCopyWith(SpotubeFullArtistObject value, + $Res Function(SpotubeFullArtistObject) then) = + _$SpotubeFullArtistObjectCopyWithImpl<$Res, SpotubeFullArtistObject>; + @useResult + $Res call( + {String id, + String name, + String externalUri, + List images, + List? genres, + int? followers}); +} + +/// @nodoc +class _$SpotubeFullArtistObjectCopyWithImpl<$Res, + $Val extends SpotubeFullArtistObject> + implements $SpotubeFullArtistObjectCopyWith<$Res> { + _$SpotubeFullArtistObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeFullArtistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? images = null, + Object? genres = freezed, + Object? followers = freezed, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + images: null == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List, + genres: freezed == genres + ? _value.genres + : genres // ignore: cast_nullable_to_non_nullable + as List?, + followers: freezed == followers + ? _value.followers + : followers // ignore: cast_nullable_to_non_nullable + as int?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeFullArtistObjectImplCopyWith<$Res> + implements $SpotubeFullArtistObjectCopyWith<$Res> { + factory _$$SpotubeFullArtistObjectImplCopyWith( + _$SpotubeFullArtistObjectImpl value, + $Res Function(_$SpotubeFullArtistObjectImpl) then) = + __$$SpotubeFullArtistObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String externalUri, + List images, + List? genres, + int? followers}); +} + +/// @nodoc +class __$$SpotubeFullArtistObjectImplCopyWithImpl<$Res> + extends _$SpotubeFullArtistObjectCopyWithImpl<$Res, + _$SpotubeFullArtistObjectImpl> + implements _$$SpotubeFullArtistObjectImplCopyWith<$Res> { + __$$SpotubeFullArtistObjectImplCopyWithImpl( + _$SpotubeFullArtistObjectImpl _value, + $Res Function(_$SpotubeFullArtistObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeFullArtistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? images = null, + Object? genres = freezed, + Object? followers = freezed, + }) { + return _then(_$SpotubeFullArtistObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + images: null == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List, + genres: freezed == genres + ? _value._genres + : genres // ignore: cast_nullable_to_non_nullable + as List?, + followers: freezed == followers + ? _value.followers + : followers // ignore: cast_nullable_to_non_nullable + as int?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeFullArtistObjectImpl implements _SpotubeFullArtistObject { + _$SpotubeFullArtistObjectImpl( + {required this.id, + required this.name, + required this.externalUri, + final List images = const [], + final List? genres, + this.followers}) + : _images = images, + _genres = genres; + + factory _$SpotubeFullArtistObjectImpl.fromJson(Map json) => + _$$SpotubeFullArtistObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String externalUri; + final List _images; + @override + @JsonKey() + List get images { + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_images); + } + + final List? _genres; + @override + List? get genres { + final value = _genres; + if (value == null) return null; + if (_genres is EqualUnmodifiableListView) return _genres; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(value); + } + + @override + final int? followers; + + @override + String toString() { + return 'SpotubeFullArtistObject(id: $id, name: $name, externalUri: $externalUri, images: $images, genres: $genres, followers: $followers)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeFullArtistObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + const DeepCollectionEquality().equals(other._images, _images) && + const DeepCollectionEquality().equals(other._genres, _genres) && + (identical(other.followers, followers) || + other.followers == followers)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + id, + name, + externalUri, + const DeepCollectionEquality().hash(_images), + const DeepCollectionEquality().hash(_genres), + followers); + + /// Create a copy of SpotubeFullArtistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeFullArtistObjectImplCopyWith<_$SpotubeFullArtistObjectImpl> + get copyWith => __$$SpotubeFullArtistObjectImplCopyWithImpl< + _$SpotubeFullArtistObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeFullArtistObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeFullArtistObject implements SpotubeFullArtistObject { + factory _SpotubeFullArtistObject( + {required final String id, + required final String name, + required final String externalUri, + final List images, + final List? genres, + final int? followers}) = _$SpotubeFullArtistObjectImpl; + + factory _SpotubeFullArtistObject.fromJson(Map json) = + _$SpotubeFullArtistObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get externalUri; + @override + List get images; + @override + List? get genres; + @override + int? get followers; + + /// Create a copy of SpotubeFullArtistObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeFullArtistObjectImplCopyWith<_$SpotubeFullArtistObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeSimpleArtistObject _$SpotubeSimpleArtistObjectFromJson( + Map json) { + return _SpotubeSimpleArtistObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeSimpleArtistObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + List? get images => throw _privateConstructorUsedError; + + /// Serializes this SpotubeSimpleArtistObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeSimpleArtistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeSimpleArtistObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeSimpleArtistObjectCopyWith<$Res> { + factory $SpotubeSimpleArtistObjectCopyWith(SpotubeSimpleArtistObject value, + $Res Function(SpotubeSimpleArtistObject) then) = + _$SpotubeSimpleArtistObjectCopyWithImpl<$Res, SpotubeSimpleArtistObject>; + @useResult + $Res call( + {String id, + String name, + String externalUri, + List? images}); +} + +/// @nodoc +class _$SpotubeSimpleArtistObjectCopyWithImpl<$Res, + $Val extends SpotubeSimpleArtistObject> + implements $SpotubeSimpleArtistObjectCopyWith<$Res> { + _$SpotubeSimpleArtistObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeSimpleArtistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? images = freezed, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + images: freezed == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeSimpleArtistObjectImplCopyWith<$Res> + implements $SpotubeSimpleArtistObjectCopyWith<$Res> { + factory _$$SpotubeSimpleArtistObjectImplCopyWith( + _$SpotubeSimpleArtistObjectImpl value, + $Res Function(_$SpotubeSimpleArtistObjectImpl) then) = + __$$SpotubeSimpleArtistObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String externalUri, + List? images}); +} + +/// @nodoc +class __$$SpotubeSimpleArtistObjectImplCopyWithImpl<$Res> + extends _$SpotubeSimpleArtistObjectCopyWithImpl<$Res, + _$SpotubeSimpleArtistObjectImpl> + implements _$$SpotubeSimpleArtistObjectImplCopyWith<$Res> { + __$$SpotubeSimpleArtistObjectImplCopyWithImpl( + _$SpotubeSimpleArtistObjectImpl _value, + $Res Function(_$SpotubeSimpleArtistObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeSimpleArtistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? images = freezed, + }) { + return _then(_$SpotubeSimpleArtistObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + images: freezed == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeSimpleArtistObjectImpl implements _SpotubeSimpleArtistObject { + _$SpotubeSimpleArtistObjectImpl( + {required this.id, + required this.name, + required this.externalUri, + final List? images}) + : _images = images; + + factory _$SpotubeSimpleArtistObjectImpl.fromJson(Map json) => + _$$SpotubeSimpleArtistObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String externalUri; + final List? _images; + @override + List? get images { + final value = _images; + if (value == null) return null; + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(value); + } + + @override + String toString() { + return 'SpotubeSimpleArtistObject(id: $id, name: $name, externalUri: $externalUri, images: $images)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeSimpleArtistObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + const DeepCollectionEquality().equals(other._images, _images)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, id, name, externalUri, + const DeepCollectionEquality().hash(_images)); + + /// Create a copy of SpotubeSimpleArtistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeSimpleArtistObjectImplCopyWith<_$SpotubeSimpleArtistObjectImpl> + get copyWith => __$$SpotubeSimpleArtistObjectImplCopyWithImpl< + _$SpotubeSimpleArtistObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeSimpleArtistObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeSimpleArtistObject implements SpotubeSimpleArtistObject { + factory _SpotubeSimpleArtistObject( + {required final String id, + required final String name, + required final String externalUri, + final List? images}) = + _$SpotubeSimpleArtistObjectImpl; + + factory _SpotubeSimpleArtistObject.fromJson(Map json) = + _$SpotubeSimpleArtistObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get externalUri; + @override + List? get images; + + /// Create a copy of SpotubeSimpleArtistObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeSimpleArtistObjectImplCopyWith<_$SpotubeSimpleArtistObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeBrowseSectionObject _$SpotubeBrowseSectionObjectFromJson( + Map json, T Function(Object?) fromJsonT) { + return _SpotubeBrowseSectionObject.fromJson(json, fromJsonT); +} + +/// @nodoc +mixin _$SpotubeBrowseSectionObject { + String get id => throw _privateConstructorUsedError; + String get title => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + bool get browseMore => throw _privateConstructorUsedError; + List get items => throw _privateConstructorUsedError; + + /// Serializes this SpotubeBrowseSectionObject to a JSON map. + Map toJson(Object? Function(T) toJsonT) => + throw _privateConstructorUsedError; + + /// Create a copy of SpotubeBrowseSectionObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeBrowseSectionObjectCopyWith> + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeBrowseSectionObjectCopyWith { + factory $SpotubeBrowseSectionObjectCopyWith( + SpotubeBrowseSectionObject value, + $Res Function(SpotubeBrowseSectionObject) then) = + _$SpotubeBrowseSectionObjectCopyWithImpl>; + @useResult + $Res call( + {String id, + String title, + String externalUri, + bool browseMore, + List items}); +} + +/// @nodoc +class _$SpotubeBrowseSectionObjectCopyWithImpl> + implements $SpotubeBrowseSectionObjectCopyWith { + _$SpotubeBrowseSectionObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeBrowseSectionObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? title = null, + Object? externalUri = null, + Object? browseMore = null, + Object? items = null, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + title: null == title + ? _value.title + : title // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + browseMore: null == browseMore + ? _value.browseMore + : browseMore // ignore: cast_nullable_to_non_nullable + as bool, + items: null == items + ? _value.items + : items // ignore: cast_nullable_to_non_nullable + as List, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeBrowseSectionObjectImplCopyWith + implements $SpotubeBrowseSectionObjectCopyWith { + factory _$$SpotubeBrowseSectionObjectImplCopyWith( + _$SpotubeBrowseSectionObjectImpl value, + $Res Function(_$SpotubeBrowseSectionObjectImpl) then) = + __$$SpotubeBrowseSectionObjectImplCopyWithImpl; + @override + @useResult + $Res call( + {String id, + String title, + String externalUri, + bool browseMore, + List items}); +} + +/// @nodoc +class __$$SpotubeBrowseSectionObjectImplCopyWithImpl + extends _$SpotubeBrowseSectionObjectCopyWithImpl> + implements _$$SpotubeBrowseSectionObjectImplCopyWith { + __$$SpotubeBrowseSectionObjectImplCopyWithImpl( + _$SpotubeBrowseSectionObjectImpl _value, + $Res Function(_$SpotubeBrowseSectionObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeBrowseSectionObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? title = null, + Object? externalUri = null, + Object? browseMore = null, + Object? items = null, + }) { + return _then(_$SpotubeBrowseSectionObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + title: null == title + ? _value.title + : title // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + browseMore: null == browseMore + ? _value.browseMore + : browseMore // ignore: cast_nullable_to_non_nullable + as bool, + items: null == items + ? _value._items + : items // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable(genericArgumentFactories: true) +class _$SpotubeBrowseSectionObjectImpl + implements _SpotubeBrowseSectionObject { + _$SpotubeBrowseSectionObjectImpl( + {required this.id, + required this.title, + required this.externalUri, + required this.browseMore, + required final List items}) + : _items = items; + + factory _$SpotubeBrowseSectionObjectImpl.fromJson( + Map json, T Function(Object?) fromJsonT) => + _$$SpotubeBrowseSectionObjectImplFromJson(json, fromJsonT); + + @override + final String id; + @override + final String title; + @override + final String externalUri; + @override + final bool browseMore; + final List _items; + @override + List get items { + if (_items is EqualUnmodifiableListView) return _items; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_items); + } + + @override + String toString() { + return 'SpotubeBrowseSectionObject<$T>(id: $id, title: $title, externalUri: $externalUri, browseMore: $browseMore, items: $items)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeBrowseSectionObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.title, title) || other.title == title) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + (identical(other.browseMore, browseMore) || + other.browseMore == browseMore) && + const DeepCollectionEquality().equals(other._items, _items)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, id, title, externalUri, + browseMore, const DeepCollectionEquality().hash(_items)); + + /// Create a copy of SpotubeBrowseSectionObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeBrowseSectionObjectImplCopyWith> + get copyWith => __$$SpotubeBrowseSectionObjectImplCopyWithImpl>(this, _$identity); + + @override + Map toJson(Object? Function(T) toJsonT) { + return _$$SpotubeBrowseSectionObjectImplToJson(this, toJsonT); + } +} + +abstract class _SpotubeBrowseSectionObject + implements SpotubeBrowseSectionObject { + factory _SpotubeBrowseSectionObject( + {required final String id, + required final String title, + required final String externalUri, + required final bool browseMore, + required final List items}) = _$SpotubeBrowseSectionObjectImpl; + + factory _SpotubeBrowseSectionObject.fromJson( + Map json, T Function(Object?) fromJsonT) = + _$SpotubeBrowseSectionObjectImpl.fromJson; + + @override + String get id; + @override + String get title; + @override + String get externalUri; + @override + bool get browseMore; + @override + List get items; + + /// Create a copy of SpotubeBrowseSectionObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeBrowseSectionObjectImplCopyWith> + get copyWith => throw _privateConstructorUsedError; +} + +MetadataFormFieldObject _$MetadataFormFieldObjectFromJson( + Map json) { + switch (json['objectType']) { + case 'input': + return MetadataFormFieldInputObject.fromJson(json); + case 'text': + return MetadataFormFieldTextObject.fromJson(json); + + default: + throw CheckedFromJsonException( + json, + 'objectType', + 'MetadataFormFieldObject', + 'Invalid union type "${json['objectType']}"!'); + } +} + +/// @nodoc +mixin _$MetadataFormFieldObject { + String get objectType => throw _privateConstructorUsedError; + @optionalTypeArgs + TResult when({ + required TResult Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex) + input, + required TResult Function(String objectType, String text) text, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex)? + input, + TResult? Function(String objectType, String text)? text, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult maybeWhen({ + TResult Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex)? + input, + TResult Function(String objectType, String text)? text, + required TResult orElse(), + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult map({ + required TResult Function(MetadataFormFieldInputObject value) input, + required TResult Function(MetadataFormFieldTextObject value) text, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(MetadataFormFieldInputObject value)? input, + TResult? Function(MetadataFormFieldTextObject value)? text, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult maybeMap({ + TResult Function(MetadataFormFieldInputObject value)? input, + TResult Function(MetadataFormFieldTextObject value)? text, + required TResult orElse(), + }) => + throw _privateConstructorUsedError; + + /// Serializes this MetadataFormFieldObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $MetadataFormFieldObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $MetadataFormFieldObjectCopyWith<$Res> { + factory $MetadataFormFieldObjectCopyWith(MetadataFormFieldObject value, + $Res Function(MetadataFormFieldObject) then) = + _$MetadataFormFieldObjectCopyWithImpl<$Res, MetadataFormFieldObject>; + @useResult + $Res call({String objectType}); +} + +/// @nodoc +class _$MetadataFormFieldObjectCopyWithImpl<$Res, + $Val extends MetadataFormFieldObject> + implements $MetadataFormFieldObjectCopyWith<$Res> { + _$MetadataFormFieldObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? objectType = null, + }) { + return _then(_value.copyWith( + objectType: null == objectType + ? _value.objectType + : objectType // ignore: cast_nullable_to_non_nullable + as String, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$MetadataFormFieldInputObjectImplCopyWith<$Res> + implements $MetadataFormFieldObjectCopyWith<$Res> { + factory _$$MetadataFormFieldInputObjectImplCopyWith( + _$MetadataFormFieldInputObjectImpl value, + $Res Function(_$MetadataFormFieldInputObjectImpl) then) = + __$$MetadataFormFieldInputObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex}); +} + +/// @nodoc +class __$$MetadataFormFieldInputObjectImplCopyWithImpl<$Res> + extends _$MetadataFormFieldObjectCopyWithImpl<$Res, + _$MetadataFormFieldInputObjectImpl> + implements _$$MetadataFormFieldInputObjectImplCopyWith<$Res> { + __$$MetadataFormFieldInputObjectImplCopyWithImpl( + _$MetadataFormFieldInputObjectImpl _value, + $Res Function(_$MetadataFormFieldInputObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? objectType = null, + Object? id = null, + Object? variant = null, + Object? placeholder = freezed, + Object? defaultValue = freezed, + Object? required = freezed, + Object? regex = freezed, + }) { + return _then(_$MetadataFormFieldInputObjectImpl( + objectType: null == objectType + ? _value.objectType + : objectType // ignore: cast_nullable_to_non_nullable + as String, + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + variant: null == variant + ? _value.variant + : variant // ignore: cast_nullable_to_non_nullable + as FormFieldVariant, + placeholder: freezed == placeholder + ? _value.placeholder + : placeholder // ignore: cast_nullable_to_non_nullable + as String?, + defaultValue: freezed == defaultValue + ? _value.defaultValue + : defaultValue // ignore: cast_nullable_to_non_nullable + as String?, + required: freezed == required + ? _value.required + : required // ignore: cast_nullable_to_non_nullable + as bool?, + regex: freezed == regex + ? _value.regex + : regex // ignore: cast_nullable_to_non_nullable + as String?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$MetadataFormFieldInputObjectImpl + implements MetadataFormFieldInputObject { + _$MetadataFormFieldInputObjectImpl( + {required this.objectType, + required this.id, + this.variant = FormFieldVariant.text, + this.placeholder, + this.defaultValue, + this.required, + this.regex}); + + factory _$MetadataFormFieldInputObjectImpl.fromJson( + Map json) => + _$$MetadataFormFieldInputObjectImplFromJson(json); + + @override + final String objectType; + @override + final String id; + @override + @JsonKey() + final FormFieldVariant variant; + @override + final String? placeholder; + @override + final String? defaultValue; + @override + final bool? required; + @override + final String? regex; + + @override + String toString() { + return 'MetadataFormFieldObject.input(objectType: $objectType, id: $id, variant: $variant, placeholder: $placeholder, defaultValue: $defaultValue, required: $required, regex: $regex)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$MetadataFormFieldInputObjectImpl && + (identical(other.objectType, objectType) || + other.objectType == objectType) && + (identical(other.id, id) || other.id == id) && + (identical(other.variant, variant) || other.variant == variant) && + (identical(other.placeholder, placeholder) || + other.placeholder == placeholder) && + (identical(other.defaultValue, defaultValue) || + other.defaultValue == defaultValue) && + (identical(other.required, required) || + other.required == required) && + (identical(other.regex, regex) || other.regex == regex)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, objectType, id, variant, + placeholder, defaultValue, required, regex); + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$MetadataFormFieldInputObjectImplCopyWith< + _$MetadataFormFieldInputObjectImpl> + get copyWith => __$$MetadataFormFieldInputObjectImplCopyWithImpl< + _$MetadataFormFieldInputObjectImpl>(this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex) + input, + required TResult Function(String objectType, String text) text, + }) { + return input( + objectType, id, variant, placeholder, defaultValue, required, regex); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex)? + input, + TResult? Function(String objectType, String text)? text, + }) { + return input?.call( + objectType, id, variant, placeholder, defaultValue, required, regex); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex)? + input, + TResult Function(String objectType, String text)? text, + required TResult orElse(), + }) { + if (input != null) { + return input( + objectType, id, variant, placeholder, defaultValue, required, regex); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(MetadataFormFieldInputObject value) input, + required TResult Function(MetadataFormFieldTextObject value) text, + }) { + return input(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(MetadataFormFieldInputObject value)? input, + TResult? Function(MetadataFormFieldTextObject value)? text, + }) { + return input?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(MetadataFormFieldInputObject value)? input, + TResult Function(MetadataFormFieldTextObject value)? text, + required TResult orElse(), + }) { + if (input != null) { + return input(this); + } + return orElse(); + } + + @override + Map toJson() { + return _$$MetadataFormFieldInputObjectImplToJson( + this, + ); + } +} + +abstract class MetadataFormFieldInputObject implements MetadataFormFieldObject { + factory MetadataFormFieldInputObject( + {required final String objectType, + required final String id, + final FormFieldVariant variant, + final String? placeholder, + final String? defaultValue, + final bool? required, + final String? regex}) = _$MetadataFormFieldInputObjectImpl; + + factory MetadataFormFieldInputObject.fromJson(Map json) = + _$MetadataFormFieldInputObjectImpl.fromJson; + + @override + String get objectType; + String get id; + FormFieldVariant get variant; + String? get placeholder; + String? get defaultValue; + bool? get required; + String? get regex; + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$MetadataFormFieldInputObjectImplCopyWith< + _$MetadataFormFieldInputObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class _$$MetadataFormFieldTextObjectImplCopyWith<$Res> + implements $MetadataFormFieldObjectCopyWith<$Res> { + factory _$$MetadataFormFieldTextObjectImplCopyWith( + _$MetadataFormFieldTextObjectImpl value, + $Res Function(_$MetadataFormFieldTextObjectImpl) then) = + __$$MetadataFormFieldTextObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call({String objectType, String text}); +} + +/// @nodoc +class __$$MetadataFormFieldTextObjectImplCopyWithImpl<$Res> + extends _$MetadataFormFieldObjectCopyWithImpl<$Res, + _$MetadataFormFieldTextObjectImpl> + implements _$$MetadataFormFieldTextObjectImplCopyWith<$Res> { + __$$MetadataFormFieldTextObjectImplCopyWithImpl( + _$MetadataFormFieldTextObjectImpl _value, + $Res Function(_$MetadataFormFieldTextObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? objectType = null, + Object? text = null, + }) { + return _then(_$MetadataFormFieldTextObjectImpl( + objectType: null == objectType + ? _value.objectType + : objectType // ignore: cast_nullable_to_non_nullable + as String, + text: null == text + ? _value.text + : text // ignore: cast_nullable_to_non_nullable + as String, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$MetadataFormFieldTextObjectImpl implements MetadataFormFieldTextObject { + _$MetadataFormFieldTextObjectImpl( + {required this.objectType, required this.text}); + + factory _$MetadataFormFieldTextObjectImpl.fromJson( + Map json) => + _$$MetadataFormFieldTextObjectImplFromJson(json); + + @override + final String objectType; + @override + final String text; + + @override + String toString() { + return 'MetadataFormFieldObject.text(objectType: $objectType, text: $text)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$MetadataFormFieldTextObjectImpl && + (identical(other.objectType, objectType) || + other.objectType == objectType) && + (identical(other.text, text) || other.text == text)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, objectType, text); + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$MetadataFormFieldTextObjectImplCopyWith<_$MetadataFormFieldTextObjectImpl> + get copyWith => __$$MetadataFormFieldTextObjectImplCopyWithImpl< + _$MetadataFormFieldTextObjectImpl>(this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex) + input, + required TResult Function(String objectType, String text) text, + }) { + return text(objectType, this.text); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex)? + input, + TResult? Function(String objectType, String text)? text, + }) { + return text?.call(objectType, this.text); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function( + String objectType, + String id, + FormFieldVariant variant, + String? placeholder, + String? defaultValue, + bool? required, + String? regex)? + input, + TResult Function(String objectType, String text)? text, + required TResult orElse(), + }) { + if (text != null) { + return text(objectType, this.text); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(MetadataFormFieldInputObject value) input, + required TResult Function(MetadataFormFieldTextObject value) text, + }) { + return text(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(MetadataFormFieldInputObject value)? input, + TResult? Function(MetadataFormFieldTextObject value)? text, + }) { + return text?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(MetadataFormFieldInputObject value)? input, + TResult Function(MetadataFormFieldTextObject value)? text, + required TResult orElse(), + }) { + if (text != null) { + return text(this); + } + return orElse(); + } + + @override + Map toJson() { + return _$$MetadataFormFieldTextObjectImplToJson( + this, + ); + } +} + +abstract class MetadataFormFieldTextObject implements MetadataFormFieldObject { + factory MetadataFormFieldTextObject( + {required final String objectType, + required final String text}) = _$MetadataFormFieldTextObjectImpl; + + factory MetadataFormFieldTextObject.fromJson(Map json) = + _$MetadataFormFieldTextObjectImpl.fromJson; + + @override + String get objectType; + String get text; + + /// Create a copy of MetadataFormFieldObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$MetadataFormFieldTextObjectImplCopyWith<_$MetadataFormFieldTextObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeImageObject _$SpotubeImageObjectFromJson(Map json) { + return _SpotubeImageObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeImageObject { + String get url => throw _privateConstructorUsedError; + int? get width => throw _privateConstructorUsedError; + int? get height => throw _privateConstructorUsedError; + + /// Serializes this SpotubeImageObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeImageObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeImageObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeImageObjectCopyWith<$Res> { + factory $SpotubeImageObjectCopyWith( + SpotubeImageObject value, $Res Function(SpotubeImageObject) then) = + _$SpotubeImageObjectCopyWithImpl<$Res, SpotubeImageObject>; + @useResult + $Res call({String url, int? width, int? height}); +} + +/// @nodoc +class _$SpotubeImageObjectCopyWithImpl<$Res, $Val extends SpotubeImageObject> + implements $SpotubeImageObjectCopyWith<$Res> { + _$SpotubeImageObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeImageObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? url = null, + Object? width = freezed, + Object? height = freezed, + }) { + return _then(_value.copyWith( + url: null == url + ? _value.url + : url // ignore: cast_nullable_to_non_nullable + as String, + width: freezed == width + ? _value.width + : width // ignore: cast_nullable_to_non_nullable + as int?, + height: freezed == height + ? _value.height + : height // ignore: cast_nullable_to_non_nullable + as int?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeImageObjectImplCopyWith<$Res> + implements $SpotubeImageObjectCopyWith<$Res> { + factory _$$SpotubeImageObjectImplCopyWith(_$SpotubeImageObjectImpl value, + $Res Function(_$SpotubeImageObjectImpl) then) = + __$$SpotubeImageObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call({String url, int? width, int? height}); +} + +/// @nodoc +class __$$SpotubeImageObjectImplCopyWithImpl<$Res> + extends _$SpotubeImageObjectCopyWithImpl<$Res, _$SpotubeImageObjectImpl> + implements _$$SpotubeImageObjectImplCopyWith<$Res> { + __$$SpotubeImageObjectImplCopyWithImpl(_$SpotubeImageObjectImpl _value, + $Res Function(_$SpotubeImageObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeImageObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? url = null, + Object? width = freezed, + Object? height = freezed, + }) { + return _then(_$SpotubeImageObjectImpl( + url: null == url + ? _value.url + : url // ignore: cast_nullable_to_non_nullable + as String, + width: freezed == width + ? _value.width + : width // ignore: cast_nullable_to_non_nullable + as int?, + height: freezed == height + ? _value.height + : height // ignore: cast_nullable_to_non_nullable + as int?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeImageObjectImpl implements _SpotubeImageObject { + _$SpotubeImageObjectImpl({required this.url, this.width, this.height}); + + factory _$SpotubeImageObjectImpl.fromJson(Map json) => + _$$SpotubeImageObjectImplFromJson(json); + + @override + final String url; + @override + final int? width; + @override + final int? height; + + @override + String toString() { + return 'SpotubeImageObject(url: $url, width: $width, height: $height)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeImageObjectImpl && + (identical(other.url, url) || other.url == url) && + (identical(other.width, width) || other.width == width) && + (identical(other.height, height) || other.height == height)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, url, width, height); + + /// Create a copy of SpotubeImageObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeImageObjectImplCopyWith<_$SpotubeImageObjectImpl> get copyWith => + __$$SpotubeImageObjectImplCopyWithImpl<_$SpotubeImageObjectImpl>( + this, _$identity); + + @override + Map toJson() { + return _$$SpotubeImageObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeImageObject implements SpotubeImageObject { + factory _SpotubeImageObject( + {required final String url, + final int? width, + final int? height}) = _$SpotubeImageObjectImpl; + + factory _SpotubeImageObject.fromJson(Map json) = + _$SpotubeImageObjectImpl.fromJson; + + @override + String get url; + @override + int? get width; + @override + int? get height; + + /// Create a copy of SpotubeImageObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeImageObjectImplCopyWith<_$SpotubeImageObjectImpl> get copyWith => + throw _privateConstructorUsedError; +} + +SpotubePaginationResponseObject _$SpotubePaginationResponseObjectFromJson( + Map json, T Function(Object?) fromJsonT) { + return _SpotubePaginationResponseObject.fromJson(json, fromJsonT); +} + +/// @nodoc +mixin _$SpotubePaginationResponseObject { + int get limit => throw _privateConstructorUsedError; + int? get nextOffset => throw _privateConstructorUsedError; + int get total => throw _privateConstructorUsedError; + bool get hasMore => throw _privateConstructorUsedError; + List get items => throw _privateConstructorUsedError; + + /// Serializes this SpotubePaginationResponseObject to a JSON map. + Map toJson(Object? Function(T) toJsonT) => + throw _privateConstructorUsedError; + + /// Create a copy of SpotubePaginationResponseObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubePaginationResponseObjectCopyWith> + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubePaginationResponseObjectCopyWith { + factory $SpotubePaginationResponseObjectCopyWith( + SpotubePaginationResponseObject value, + $Res Function(SpotubePaginationResponseObject) then) = + _$SpotubePaginationResponseObjectCopyWithImpl>; + @useResult + $Res call( + {int limit, int? nextOffset, int total, bool hasMore, List items}); +} + +/// @nodoc +class _$SpotubePaginationResponseObjectCopyWithImpl> + implements $SpotubePaginationResponseObjectCopyWith { + _$SpotubePaginationResponseObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubePaginationResponseObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? limit = null, + Object? nextOffset = freezed, + Object? total = null, + Object? hasMore = null, + Object? items = null, + }) { + return _then(_value.copyWith( + limit: null == limit + ? _value.limit + : limit // ignore: cast_nullable_to_non_nullable + as int, + nextOffset: freezed == nextOffset + ? _value.nextOffset + : nextOffset // ignore: cast_nullable_to_non_nullable + as int?, + total: null == total + ? _value.total + : total // ignore: cast_nullable_to_non_nullable + as int, + hasMore: null == hasMore + ? _value.hasMore + : hasMore // ignore: cast_nullable_to_non_nullable + as bool, + items: null == items + ? _value.items + : items // ignore: cast_nullable_to_non_nullable + as List, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubePaginationResponseObjectImplCopyWith + implements $SpotubePaginationResponseObjectCopyWith { + factory _$$SpotubePaginationResponseObjectImplCopyWith( + _$SpotubePaginationResponseObjectImpl value, + $Res Function(_$SpotubePaginationResponseObjectImpl) then) = + __$$SpotubePaginationResponseObjectImplCopyWithImpl; + @override + @useResult + $Res call( + {int limit, int? nextOffset, int total, bool hasMore, List items}); +} + +/// @nodoc +class __$$SpotubePaginationResponseObjectImplCopyWithImpl + extends _$SpotubePaginationResponseObjectCopyWithImpl> + implements _$$SpotubePaginationResponseObjectImplCopyWith { + __$$SpotubePaginationResponseObjectImplCopyWithImpl( + _$SpotubePaginationResponseObjectImpl _value, + $Res Function(_$SpotubePaginationResponseObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubePaginationResponseObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? limit = null, + Object? nextOffset = freezed, + Object? total = null, + Object? hasMore = null, + Object? items = null, + }) { + return _then(_$SpotubePaginationResponseObjectImpl( + limit: null == limit + ? _value.limit + : limit // ignore: cast_nullable_to_non_nullable + as int, + nextOffset: freezed == nextOffset + ? _value.nextOffset + : nextOffset // ignore: cast_nullable_to_non_nullable + as int?, + total: null == total + ? _value.total + : total // ignore: cast_nullable_to_non_nullable + as int, + hasMore: null == hasMore + ? _value.hasMore + : hasMore // ignore: cast_nullable_to_non_nullable + as bool, + items: null == items + ? _value._items + : items // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable(genericArgumentFactories: true) +class _$SpotubePaginationResponseObjectImpl + implements _SpotubePaginationResponseObject { + _$SpotubePaginationResponseObjectImpl( + {required this.limit, + required this.nextOffset, + required this.total, + required this.hasMore, + required final List items}) + : _items = items; + + factory _$SpotubePaginationResponseObjectImpl.fromJson( + Map json, T Function(Object?) fromJsonT) => + _$$SpotubePaginationResponseObjectImplFromJson(json, fromJsonT); + + @override + final int limit; + @override + final int? nextOffset; + @override + final int total; + @override + final bool hasMore; + final List _items; + @override + List get items { + if (_items is EqualUnmodifiableListView) return _items; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_items); + } + + @override + String toString() { + return 'SpotubePaginationResponseObject<$T>(limit: $limit, nextOffset: $nextOffset, total: $total, hasMore: $hasMore, items: $items)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubePaginationResponseObjectImpl && + (identical(other.limit, limit) || other.limit == limit) && + (identical(other.nextOffset, nextOffset) || + other.nextOffset == nextOffset) && + (identical(other.total, total) || other.total == total) && + (identical(other.hasMore, hasMore) || other.hasMore == hasMore) && + const DeepCollectionEquality().equals(other._items, _items)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, limit, nextOffset, total, + hasMore, const DeepCollectionEquality().hash(_items)); + + /// Create a copy of SpotubePaginationResponseObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubePaginationResponseObjectImplCopyWith> + get copyWith => __$$SpotubePaginationResponseObjectImplCopyWithImpl>(this, _$identity); + + @override + Map toJson(Object? Function(T) toJsonT) { + return _$$SpotubePaginationResponseObjectImplToJson(this, toJsonT); + } +} + +abstract class _SpotubePaginationResponseObject + implements SpotubePaginationResponseObject { + factory _SpotubePaginationResponseObject( + {required final int limit, + required final int? nextOffset, + required final int total, + required final bool hasMore, + required final List items}) = _$SpotubePaginationResponseObjectImpl; + + factory _SpotubePaginationResponseObject.fromJson( + Map json, T Function(Object?) fromJsonT) = + _$SpotubePaginationResponseObjectImpl.fromJson; + + @override + int get limit; + @override + int? get nextOffset; + @override + int get total; + @override + bool get hasMore; + @override + List get items; + + /// Create a copy of SpotubePaginationResponseObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubePaginationResponseObjectImplCopyWith> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeFullPlaylistObject _$SpotubeFullPlaylistObjectFromJson( + Map json) { + return _SpotubeFullPlaylistObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeFullPlaylistObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + String get description => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + SpotubeUserObject get owner => throw _privateConstructorUsedError; + List get images => throw _privateConstructorUsedError; + List get collaborators => + throw _privateConstructorUsedError; + bool get collaborative => throw _privateConstructorUsedError; + bool get public => throw _privateConstructorUsedError; + + /// Serializes this SpotubeFullPlaylistObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeFullPlaylistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeFullPlaylistObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeFullPlaylistObjectCopyWith<$Res> { + factory $SpotubeFullPlaylistObjectCopyWith(SpotubeFullPlaylistObject value, + $Res Function(SpotubeFullPlaylistObject) then) = + _$SpotubeFullPlaylistObjectCopyWithImpl<$Res, SpotubeFullPlaylistObject>; + @useResult + $Res call( + {String id, + String name, + String description, + String externalUri, + SpotubeUserObject owner, + List images, + List collaborators, + bool collaborative, + bool public}); + + $SpotubeUserObjectCopyWith<$Res> get owner; +} + +/// @nodoc +class _$SpotubeFullPlaylistObjectCopyWithImpl<$Res, + $Val extends SpotubeFullPlaylistObject> + implements $SpotubeFullPlaylistObjectCopyWith<$Res> { + _$SpotubeFullPlaylistObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeFullPlaylistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? description = null, + Object? externalUri = null, + Object? owner = null, + Object? images = null, + Object? collaborators = null, + Object? collaborative = null, + Object? public = null, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + owner: null == owner + ? _value.owner + : owner // ignore: cast_nullable_to_non_nullable + as SpotubeUserObject, + images: null == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List, + collaborators: null == collaborators + ? _value.collaborators + : collaborators // ignore: cast_nullable_to_non_nullable + as List, + collaborative: null == collaborative + ? _value.collaborative + : collaborative // ignore: cast_nullable_to_non_nullable + as bool, + public: null == public + ? _value.public + : public // ignore: cast_nullable_to_non_nullable + as bool, + ) as $Val); + } + + /// Create a copy of SpotubeFullPlaylistObject + /// with the given fields replaced by the non-null parameter values. + @override + @pragma('vm:prefer-inline') + $SpotubeUserObjectCopyWith<$Res> get owner { + return $SpotubeUserObjectCopyWith<$Res>(_value.owner, (value) { + return _then(_value.copyWith(owner: value) as $Val); + }); + } +} + +/// @nodoc +abstract class _$$SpotubeFullPlaylistObjectImplCopyWith<$Res> + implements $SpotubeFullPlaylistObjectCopyWith<$Res> { + factory _$$SpotubeFullPlaylistObjectImplCopyWith( + _$SpotubeFullPlaylistObjectImpl value, + $Res Function(_$SpotubeFullPlaylistObjectImpl) then) = + __$$SpotubeFullPlaylistObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String description, + String externalUri, + SpotubeUserObject owner, + List images, + List collaborators, + bool collaborative, + bool public}); + + @override + $SpotubeUserObjectCopyWith<$Res> get owner; +} + +/// @nodoc +class __$$SpotubeFullPlaylistObjectImplCopyWithImpl<$Res> + extends _$SpotubeFullPlaylistObjectCopyWithImpl<$Res, + _$SpotubeFullPlaylistObjectImpl> + implements _$$SpotubeFullPlaylistObjectImplCopyWith<$Res> { + __$$SpotubeFullPlaylistObjectImplCopyWithImpl( + _$SpotubeFullPlaylistObjectImpl _value, + $Res Function(_$SpotubeFullPlaylistObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeFullPlaylistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? description = null, + Object? externalUri = null, + Object? owner = null, + Object? images = null, + Object? collaborators = null, + Object? collaborative = null, + Object? public = null, + }) { + return _then(_$SpotubeFullPlaylistObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + owner: null == owner + ? _value.owner + : owner // ignore: cast_nullable_to_non_nullable + as SpotubeUserObject, + images: null == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List, + collaborators: null == collaborators + ? _value._collaborators + : collaborators // ignore: cast_nullable_to_non_nullable + as List, + collaborative: null == collaborative + ? _value.collaborative + : collaborative // ignore: cast_nullable_to_non_nullable + as bool, + public: null == public + ? _value.public + : public // ignore: cast_nullable_to_non_nullable + as bool, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeFullPlaylistObjectImpl implements _SpotubeFullPlaylistObject { + _$SpotubeFullPlaylistObjectImpl( + {required this.id, + required this.name, + required this.description, + required this.externalUri, + required this.owner, + final List images = const [], + final List collaborators = const [], + this.collaborative = false, + this.public = false}) + : _images = images, + _collaborators = collaborators; + + factory _$SpotubeFullPlaylistObjectImpl.fromJson(Map json) => + _$$SpotubeFullPlaylistObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String description; + @override + final String externalUri; + @override + final SpotubeUserObject owner; + final List _images; + @override + @JsonKey() + List get images { + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_images); + } + + final List _collaborators; + @override + @JsonKey() + List get collaborators { + if (_collaborators is EqualUnmodifiableListView) return _collaborators; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_collaborators); + } + + @override + @JsonKey() + final bool collaborative; + @override + @JsonKey() + final bool public; + + @override + String toString() { + return 'SpotubeFullPlaylistObject(id: $id, name: $name, description: $description, externalUri: $externalUri, owner: $owner, images: $images, collaborators: $collaborators, collaborative: $collaborative, public: $public)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeFullPlaylistObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.description, description) || + other.description == description) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + (identical(other.owner, owner) || other.owner == owner) && + const DeepCollectionEquality().equals(other._images, _images) && + const DeepCollectionEquality() + .equals(other._collaborators, _collaborators) && + (identical(other.collaborative, collaborative) || + other.collaborative == collaborative) && + (identical(other.public, public) || other.public == public)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + id, + name, + description, + externalUri, + owner, + const DeepCollectionEquality().hash(_images), + const DeepCollectionEquality().hash(_collaborators), + collaborative, + public); + + /// Create a copy of SpotubeFullPlaylistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeFullPlaylistObjectImplCopyWith<_$SpotubeFullPlaylistObjectImpl> + get copyWith => __$$SpotubeFullPlaylistObjectImplCopyWithImpl< + _$SpotubeFullPlaylistObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeFullPlaylistObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeFullPlaylistObject implements SpotubeFullPlaylistObject { + factory _SpotubeFullPlaylistObject( + {required final String id, + required final String name, + required final String description, + required final String externalUri, + required final SpotubeUserObject owner, + final List images, + final List collaborators, + final bool collaborative, + final bool public}) = _$SpotubeFullPlaylistObjectImpl; + + factory _SpotubeFullPlaylistObject.fromJson(Map json) = + _$SpotubeFullPlaylistObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get description; + @override + String get externalUri; + @override + SpotubeUserObject get owner; + @override + List get images; + @override + List get collaborators; + @override + bool get collaborative; + @override + bool get public; + + /// Create a copy of SpotubeFullPlaylistObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeFullPlaylistObjectImplCopyWith<_$SpotubeFullPlaylistObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeSimplePlaylistObject _$SpotubeSimplePlaylistObjectFromJson( + Map json) { + return _SpotubeSimplePlaylistObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeSimplePlaylistObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + String get description => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + SpotubeUserObject get owner => throw _privateConstructorUsedError; + List get images => throw _privateConstructorUsedError; + + /// Serializes this SpotubeSimplePlaylistObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeSimplePlaylistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeSimplePlaylistObjectCopyWith + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeSimplePlaylistObjectCopyWith<$Res> { + factory $SpotubeSimplePlaylistObjectCopyWith( + SpotubeSimplePlaylistObject value, + $Res Function(SpotubeSimplePlaylistObject) then) = + _$SpotubeSimplePlaylistObjectCopyWithImpl<$Res, + SpotubeSimplePlaylistObject>; + @useResult + $Res call( + {String id, + String name, + String description, + String externalUri, + SpotubeUserObject owner, + List images}); + + $SpotubeUserObjectCopyWith<$Res> get owner; +} + +/// @nodoc +class _$SpotubeSimplePlaylistObjectCopyWithImpl<$Res, + $Val extends SpotubeSimplePlaylistObject> + implements $SpotubeSimplePlaylistObjectCopyWith<$Res> { + _$SpotubeSimplePlaylistObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeSimplePlaylistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? description = null, + Object? externalUri = null, + Object? owner = null, + Object? images = null, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + owner: null == owner + ? _value.owner + : owner // ignore: cast_nullable_to_non_nullable + as SpotubeUserObject, + images: null == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List, + ) as $Val); + } + + /// Create a copy of SpotubeSimplePlaylistObject + /// with the given fields replaced by the non-null parameter values. + @override + @pragma('vm:prefer-inline') + $SpotubeUserObjectCopyWith<$Res> get owner { + return $SpotubeUserObjectCopyWith<$Res>(_value.owner, (value) { + return _then(_value.copyWith(owner: value) as $Val); + }); + } +} + +/// @nodoc +abstract class _$$SpotubeSimplePlaylistObjectImplCopyWith<$Res> + implements $SpotubeSimplePlaylistObjectCopyWith<$Res> { + factory _$$SpotubeSimplePlaylistObjectImplCopyWith( + _$SpotubeSimplePlaylistObjectImpl value, + $Res Function(_$SpotubeSimplePlaylistObjectImpl) then) = + __$$SpotubeSimplePlaylistObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String description, + String externalUri, + SpotubeUserObject owner, + List images}); + + @override + $SpotubeUserObjectCopyWith<$Res> get owner; +} + +/// @nodoc +class __$$SpotubeSimplePlaylistObjectImplCopyWithImpl<$Res> + extends _$SpotubeSimplePlaylistObjectCopyWithImpl<$Res, + _$SpotubeSimplePlaylistObjectImpl> + implements _$$SpotubeSimplePlaylistObjectImplCopyWith<$Res> { + __$$SpotubeSimplePlaylistObjectImplCopyWithImpl( + _$SpotubeSimplePlaylistObjectImpl _value, + $Res Function(_$SpotubeSimplePlaylistObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeSimplePlaylistObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? description = null, + Object? externalUri = null, + Object? owner = null, + Object? images = null, + }) { + return _then(_$SpotubeSimplePlaylistObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + owner: null == owner + ? _value.owner + : owner // ignore: cast_nullable_to_non_nullable + as SpotubeUserObject, + images: null == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeSimplePlaylistObjectImpl + implements _SpotubeSimplePlaylistObject { + _$SpotubeSimplePlaylistObjectImpl( + {required this.id, + required this.name, + required this.description, + required this.externalUri, + required this.owner, + final List images = const []}) + : _images = images; + + factory _$SpotubeSimplePlaylistObjectImpl.fromJson( + Map json) => + _$$SpotubeSimplePlaylistObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String description; + @override + final String externalUri; + @override + final SpotubeUserObject owner; + final List _images; + @override + @JsonKey() + List get images { + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_images); + } + + @override + String toString() { + return 'SpotubeSimplePlaylistObject(id: $id, name: $name, description: $description, externalUri: $externalUri, owner: $owner, images: $images)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeSimplePlaylistObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.description, description) || + other.description == description) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + (identical(other.owner, owner) || other.owner == owner) && + const DeepCollectionEquality().equals(other._images, _images)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, id, name, description, + externalUri, owner, const DeepCollectionEquality().hash(_images)); + + /// Create a copy of SpotubeSimplePlaylistObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeSimplePlaylistObjectImplCopyWith<_$SpotubeSimplePlaylistObjectImpl> + get copyWith => __$$SpotubeSimplePlaylistObjectImplCopyWithImpl< + _$SpotubeSimplePlaylistObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeSimplePlaylistObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeSimplePlaylistObject + implements SpotubeSimplePlaylistObject { + factory _SpotubeSimplePlaylistObject( + {required final String id, + required final String name, + required final String description, + required final String externalUri, + required final SpotubeUserObject owner, + final List images}) = + _$SpotubeSimplePlaylistObjectImpl; + + factory _SpotubeSimplePlaylistObject.fromJson(Map json) = + _$SpotubeSimplePlaylistObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get description; + @override + String get externalUri; + @override + SpotubeUserObject get owner; + @override + List get images; + + /// Create a copy of SpotubeSimplePlaylistObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeSimplePlaylistObjectImplCopyWith<_$SpotubeSimplePlaylistObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeSearchResponseObject _$SpotubeSearchResponseObjectFromJson( + Map json) { + return _SpotubeSearchResponseObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeSearchResponseObject { + List get albums => + throw _privateConstructorUsedError; + List get artists => + throw _privateConstructorUsedError; + List get playlists => + throw _privateConstructorUsedError; + List get tracks => throw _privateConstructorUsedError; + + /// Serializes this SpotubeSearchResponseObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeSearchResponseObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeSearchResponseObjectCopyWith + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeSearchResponseObjectCopyWith<$Res> { + factory $SpotubeSearchResponseObjectCopyWith( + SpotubeSearchResponseObject value, + $Res Function(SpotubeSearchResponseObject) then) = + _$SpotubeSearchResponseObjectCopyWithImpl<$Res, + SpotubeSearchResponseObject>; + @useResult + $Res call( + {List albums, + List artists, + List playlists, + List tracks}); +} + +/// @nodoc +class _$SpotubeSearchResponseObjectCopyWithImpl<$Res, + $Val extends SpotubeSearchResponseObject> + implements $SpotubeSearchResponseObjectCopyWith<$Res> { + _$SpotubeSearchResponseObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeSearchResponseObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? albums = null, + Object? artists = null, + Object? playlists = null, + Object? tracks = null, + }) { + return _then(_value.copyWith( + albums: null == albums + ? _value.albums + : albums // ignore: cast_nullable_to_non_nullable + as List, + artists: null == artists + ? _value.artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + playlists: null == playlists + ? _value.playlists + : playlists // ignore: cast_nullable_to_non_nullable + as List, + tracks: null == tracks + ? _value.tracks + : tracks // ignore: cast_nullable_to_non_nullable + as List, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeSearchResponseObjectImplCopyWith<$Res> + implements $SpotubeSearchResponseObjectCopyWith<$Res> { + factory _$$SpotubeSearchResponseObjectImplCopyWith( + _$SpotubeSearchResponseObjectImpl value, + $Res Function(_$SpotubeSearchResponseObjectImpl) then) = + __$$SpotubeSearchResponseObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {List albums, + List artists, + List playlists, + List tracks}); +} + +/// @nodoc +class __$$SpotubeSearchResponseObjectImplCopyWithImpl<$Res> + extends _$SpotubeSearchResponseObjectCopyWithImpl<$Res, + _$SpotubeSearchResponseObjectImpl> + implements _$$SpotubeSearchResponseObjectImplCopyWith<$Res> { + __$$SpotubeSearchResponseObjectImplCopyWithImpl( + _$SpotubeSearchResponseObjectImpl _value, + $Res Function(_$SpotubeSearchResponseObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeSearchResponseObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? albums = null, + Object? artists = null, + Object? playlists = null, + Object? tracks = null, + }) { + return _then(_$SpotubeSearchResponseObjectImpl( + albums: null == albums + ? _value._albums + : albums // ignore: cast_nullable_to_non_nullable + as List, + artists: null == artists + ? _value._artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + playlists: null == playlists + ? _value._playlists + : playlists // ignore: cast_nullable_to_non_nullable + as List, + tracks: null == tracks + ? _value._tracks + : tracks // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeSearchResponseObjectImpl + implements _SpotubeSearchResponseObject { + _$SpotubeSearchResponseObjectImpl( + {required final List albums, + required final List artists, + required final List playlists, + required final List tracks}) + : _albums = albums, + _artists = artists, + _playlists = playlists, + _tracks = tracks; + + factory _$SpotubeSearchResponseObjectImpl.fromJson( + Map json) => + _$$SpotubeSearchResponseObjectImplFromJson(json); + + final List _albums; + @override + List get albums { + if (_albums is EqualUnmodifiableListView) return _albums; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_albums); + } + + final List _artists; + @override + List get artists { + if (_artists is EqualUnmodifiableListView) return _artists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_artists); + } + + final List _playlists; + @override + List get playlists { + if (_playlists is EqualUnmodifiableListView) return _playlists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_playlists); + } + + final List _tracks; + @override + List get tracks { + if (_tracks is EqualUnmodifiableListView) return _tracks; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_tracks); + } + + @override + String toString() { + return 'SpotubeSearchResponseObject(albums: $albums, artists: $artists, playlists: $playlists, tracks: $tracks)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeSearchResponseObjectImpl && + const DeepCollectionEquality().equals(other._albums, _albums) && + const DeepCollectionEquality().equals(other._artists, _artists) && + const DeepCollectionEquality() + .equals(other._playlists, _playlists) && + const DeepCollectionEquality().equals(other._tracks, _tracks)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + const DeepCollectionEquality().hash(_albums), + const DeepCollectionEquality().hash(_artists), + const DeepCollectionEquality().hash(_playlists), + const DeepCollectionEquality().hash(_tracks)); + + /// Create a copy of SpotubeSearchResponseObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeSearchResponseObjectImplCopyWith<_$SpotubeSearchResponseObjectImpl> + get copyWith => __$$SpotubeSearchResponseObjectImplCopyWithImpl< + _$SpotubeSearchResponseObjectImpl>(this, _$identity); + + @override + Map toJson() { + return _$$SpotubeSearchResponseObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeSearchResponseObject + implements SpotubeSearchResponseObject { + factory _SpotubeSearchResponseObject( + {required final List albums, + required final List artists, + required final List playlists, + required final List tracks}) = + _$SpotubeSearchResponseObjectImpl; + + factory _SpotubeSearchResponseObject.fromJson(Map json) = + _$SpotubeSearchResponseObjectImpl.fromJson; + + @override + List get albums; + @override + List get artists; + @override + List get playlists; + @override + List get tracks; + + /// Create a copy of SpotubeSearchResponseObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeSearchResponseObjectImplCopyWith<_$SpotubeSearchResponseObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeTrackObject _$SpotubeTrackObjectFromJson(Map json) { + switch (json['runtimeType']) { + case 'local': + return SpotubeLocalTrackObject.fromJson(json); + case 'full': + return SpotubeFullTrackObject.fromJson(json); + + default: + throw CheckedFromJsonException(json, 'runtimeType', 'SpotubeTrackObject', + 'Invalid union type "${json['runtimeType']}"!'); + } +} + +/// @nodoc +mixin _$SpotubeTrackObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + List get artists => + throw _privateConstructorUsedError; + SpotubeSimpleAlbumObject get album => throw _privateConstructorUsedError; + int get durationMs => throw _privateConstructorUsedError; + @optionalTypeArgs + TResult when({ + required TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path) + local, + required TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit) + full, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path)? + local, + TResult? Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit)? + full, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult maybeWhen({ + TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path)? + local, + TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit)? + full, + required TResult orElse(), + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult map({ + required TResult Function(SpotubeLocalTrackObject value) local, + required TResult Function(SpotubeFullTrackObject value) full, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(SpotubeLocalTrackObject value)? local, + TResult? Function(SpotubeFullTrackObject value)? full, + }) => + throw _privateConstructorUsedError; + @optionalTypeArgs + TResult maybeMap({ + TResult Function(SpotubeLocalTrackObject value)? local, + TResult Function(SpotubeFullTrackObject value)? full, + required TResult orElse(), + }) => + throw _privateConstructorUsedError; + + /// Serializes this SpotubeTrackObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeTrackObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeTrackObjectCopyWith<$Res> { + factory $SpotubeTrackObjectCopyWith( + SpotubeTrackObject value, $Res Function(SpotubeTrackObject) then) = + _$SpotubeTrackObjectCopyWithImpl<$Res, SpotubeTrackObject>; + @useResult + $Res call( + {String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs}); + + $SpotubeSimpleAlbumObjectCopyWith<$Res> get album; +} + +/// @nodoc +class _$SpotubeTrackObjectCopyWithImpl<$Res, $Val extends SpotubeTrackObject> + implements $SpotubeTrackObjectCopyWith<$Res> { + _$SpotubeTrackObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? artists = null, + Object? album = null, + Object? durationMs = null, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value.artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + album: null == album + ? _value.album + : album // ignore: cast_nullable_to_non_nullable + as SpotubeSimpleAlbumObject, + durationMs: null == durationMs + ? _value.durationMs + : durationMs // ignore: cast_nullable_to_non_nullable + as int, + ) as $Val); + } + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @override + @pragma('vm:prefer-inline') + $SpotubeSimpleAlbumObjectCopyWith<$Res> get album { + return $SpotubeSimpleAlbumObjectCopyWith<$Res>(_value.album, (value) { + return _then(_value.copyWith(album: value) as $Val); + }); + } +} + +/// @nodoc +abstract class _$$SpotubeLocalTrackObjectImplCopyWith<$Res> + implements $SpotubeTrackObjectCopyWith<$Res> { + factory _$$SpotubeLocalTrackObjectImplCopyWith( + _$SpotubeLocalTrackObjectImpl value, + $Res Function(_$SpotubeLocalTrackObjectImpl) then) = + __$$SpotubeLocalTrackObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path}); + + @override + $SpotubeSimpleAlbumObjectCopyWith<$Res> get album; +} + +/// @nodoc +class __$$SpotubeLocalTrackObjectImplCopyWithImpl<$Res> + extends _$SpotubeTrackObjectCopyWithImpl<$Res, + _$SpotubeLocalTrackObjectImpl> + implements _$$SpotubeLocalTrackObjectImplCopyWith<$Res> { + __$$SpotubeLocalTrackObjectImplCopyWithImpl( + _$SpotubeLocalTrackObjectImpl _value, + $Res Function(_$SpotubeLocalTrackObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? artists = null, + Object? album = null, + Object? durationMs = null, + Object? path = null, + }) { + return _then(_$SpotubeLocalTrackObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value._artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + album: null == album + ? _value.album + : album // ignore: cast_nullable_to_non_nullable + as SpotubeSimpleAlbumObject, + durationMs: null == durationMs + ? _value.durationMs + : durationMs // ignore: cast_nullable_to_non_nullable + as int, + path: null == path + ? _value.path + : path // ignore: cast_nullable_to_non_nullable + as String, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeLocalTrackObjectImpl implements SpotubeLocalTrackObject { + _$SpotubeLocalTrackObjectImpl( + {required this.id, + required this.name, + required this.externalUri, + final List artists = const [], + required this.album, + required this.durationMs, + required this.path, + final String? $type}) + : _artists = artists, + $type = $type ?? 'local'; + + factory _$SpotubeLocalTrackObjectImpl.fromJson(Map json) => + _$$SpotubeLocalTrackObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String externalUri; + final List _artists; + @override + @JsonKey() + List get artists { + if (_artists is EqualUnmodifiableListView) return _artists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_artists); + } + + @override + final SpotubeSimpleAlbumObject album; + @override + final int durationMs; + @override + final String path; + + @JsonKey(name: 'runtimeType') + final String $type; + + @override + String toString() { + return 'SpotubeTrackObject.local(id: $id, name: $name, externalUri: $externalUri, artists: $artists, album: $album, durationMs: $durationMs, path: $path)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeLocalTrackObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + const DeepCollectionEquality().equals(other._artists, _artists) && + (identical(other.album, album) || other.album == album) && + (identical(other.durationMs, durationMs) || + other.durationMs == durationMs) && + (identical(other.path, path) || other.path == path)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, id, name, externalUri, + const DeepCollectionEquality().hash(_artists), album, durationMs, path); + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeLocalTrackObjectImplCopyWith<_$SpotubeLocalTrackObjectImpl> + get copyWith => __$$SpotubeLocalTrackObjectImplCopyWithImpl< + _$SpotubeLocalTrackObjectImpl>(this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path) + local, + required TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit) + full, + }) { + return local(id, name, externalUri, artists, album, durationMs, path); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path)? + local, + TResult? Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit)? + full, + }) { + return local?.call(id, name, externalUri, artists, album, durationMs, path); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path)? + local, + TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit)? + full, + required TResult orElse(), + }) { + if (local != null) { + return local(id, name, externalUri, artists, album, durationMs, path); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(SpotubeLocalTrackObject value) local, + required TResult Function(SpotubeFullTrackObject value) full, + }) { + return local(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(SpotubeLocalTrackObject value)? local, + TResult? Function(SpotubeFullTrackObject value)? full, + }) { + return local?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(SpotubeLocalTrackObject value)? local, + TResult Function(SpotubeFullTrackObject value)? full, + required TResult orElse(), + }) { + if (local != null) { + return local(this); + } + return orElse(); + } + + @override + Map toJson() { + return _$$SpotubeLocalTrackObjectImplToJson( + this, + ); + } +} + +abstract class SpotubeLocalTrackObject implements SpotubeTrackObject { + factory SpotubeLocalTrackObject( + {required final String id, + required final String name, + required final String externalUri, + final List artists, + required final SpotubeSimpleAlbumObject album, + required final int durationMs, + required final String path}) = _$SpotubeLocalTrackObjectImpl; + + factory SpotubeLocalTrackObject.fromJson(Map json) = + _$SpotubeLocalTrackObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get externalUri; + @override + List get artists; + @override + SpotubeSimpleAlbumObject get album; + @override + int get durationMs; + String get path; + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeLocalTrackObjectImplCopyWith<_$SpotubeLocalTrackObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class _$$SpotubeFullTrackObjectImplCopyWith<$Res> + implements $SpotubeTrackObjectCopyWith<$Res> { + factory _$$SpotubeFullTrackObjectImplCopyWith( + _$SpotubeFullTrackObjectImpl value, + $Res Function(_$SpotubeFullTrackObjectImpl) then) = + __$$SpotubeFullTrackObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit}); + + @override + $SpotubeSimpleAlbumObjectCopyWith<$Res> get album; +} + +/// @nodoc +class __$$SpotubeFullTrackObjectImplCopyWithImpl<$Res> + extends _$SpotubeTrackObjectCopyWithImpl<$Res, _$SpotubeFullTrackObjectImpl> + implements _$$SpotubeFullTrackObjectImplCopyWith<$Res> { + __$$SpotubeFullTrackObjectImplCopyWithImpl( + _$SpotubeFullTrackObjectImpl _value, + $Res Function(_$SpotubeFullTrackObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? externalUri = null, + Object? artists = null, + Object? album = null, + Object? durationMs = null, + Object? isrc = null, + Object? explicit = null, + }) { + return _then(_$SpotubeFullTrackObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + artists: null == artists + ? _value._artists + : artists // ignore: cast_nullable_to_non_nullable + as List, + album: null == album + ? _value.album + : album // ignore: cast_nullable_to_non_nullable + as SpotubeSimpleAlbumObject, + durationMs: null == durationMs + ? _value.durationMs + : durationMs // ignore: cast_nullable_to_non_nullable + as int, + isrc: null == isrc + ? _value.isrc + : isrc // ignore: cast_nullable_to_non_nullable + as String, + explicit: null == explicit + ? _value.explicit + : explicit // ignore: cast_nullable_to_non_nullable + as bool, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeFullTrackObjectImpl implements SpotubeFullTrackObject { + _$SpotubeFullTrackObjectImpl( + {required this.id, + required this.name, + required this.externalUri, + final List artists = const [], + required this.album, + required this.durationMs, + required this.isrc, + required this.explicit, + final String? $type}) + : _artists = artists, + $type = $type ?? 'full'; + + factory _$SpotubeFullTrackObjectImpl.fromJson(Map json) => + _$$SpotubeFullTrackObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + @override + final String externalUri; + final List _artists; + @override + @JsonKey() + List get artists { + if (_artists is EqualUnmodifiableListView) return _artists; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_artists); + } + + @override + final SpotubeSimpleAlbumObject album; + @override + final int durationMs; + @override + final String isrc; + @override + final bool explicit; + + @JsonKey(name: 'runtimeType') + final String $type; + + @override + String toString() { + return 'SpotubeTrackObject.full(id: $id, name: $name, externalUri: $externalUri, artists: $artists, album: $album, durationMs: $durationMs, isrc: $isrc, explicit: $explicit)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeFullTrackObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri) && + const DeepCollectionEquality().equals(other._artists, _artists) && + (identical(other.album, album) || other.album == album) && + (identical(other.durationMs, durationMs) || + other.durationMs == durationMs) && + (identical(other.isrc, isrc) || other.isrc == isrc) && + (identical(other.explicit, explicit) || + other.explicit == explicit)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + id, + name, + externalUri, + const DeepCollectionEquality().hash(_artists), + album, + durationMs, + isrc, + explicit); + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeFullTrackObjectImplCopyWith<_$SpotubeFullTrackObjectImpl> + get copyWith => __$$SpotubeFullTrackObjectImplCopyWithImpl< + _$SpotubeFullTrackObjectImpl>(this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path) + local, + required TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit) + full, + }) { + return full( + id, name, externalUri, artists, album, durationMs, isrc, explicit); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path)? + local, + TResult? Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit)? + full, + }) { + return full?.call( + id, name, externalUri, artists, album, durationMs, isrc, explicit); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String path)? + local, + TResult Function( + String id, + String name, + String externalUri, + List artists, + SpotubeSimpleAlbumObject album, + int durationMs, + String isrc, + bool explicit)? + full, + required TResult orElse(), + }) { + if (full != null) { + return full( + id, name, externalUri, artists, album, durationMs, isrc, explicit); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(SpotubeLocalTrackObject value) local, + required TResult Function(SpotubeFullTrackObject value) full, + }) { + return full(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(SpotubeLocalTrackObject value)? local, + TResult? Function(SpotubeFullTrackObject value)? full, + }) { + return full?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(SpotubeLocalTrackObject value)? local, + TResult Function(SpotubeFullTrackObject value)? full, + required TResult orElse(), + }) { + if (full != null) { + return full(this); + } + return orElse(); + } + + @override + Map toJson() { + return _$$SpotubeFullTrackObjectImplToJson( + this, + ); + } +} + +abstract class SpotubeFullTrackObject implements SpotubeTrackObject { + factory SpotubeFullTrackObject( + {required final String id, + required final String name, + required final String externalUri, + final List artists, + required final SpotubeSimpleAlbumObject album, + required final int durationMs, + required final String isrc, + required final bool explicit}) = _$SpotubeFullTrackObjectImpl; + + factory SpotubeFullTrackObject.fromJson(Map json) = + _$SpotubeFullTrackObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + String get externalUri; + @override + List get artists; + @override + SpotubeSimpleAlbumObject get album; + @override + int get durationMs; + String get isrc; + bool get explicit; + + /// Create a copy of SpotubeTrackObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeFullTrackObjectImplCopyWith<_$SpotubeFullTrackObjectImpl> + get copyWith => throw _privateConstructorUsedError; +} + +SpotubeUserObject _$SpotubeUserObjectFromJson(Map json) { + return _SpotubeUserObject.fromJson(json); +} + +/// @nodoc +mixin _$SpotubeUserObject { + String get id => throw _privateConstructorUsedError; + String get name => throw _privateConstructorUsedError; + List get images => throw _privateConstructorUsedError; + String get externalUri => throw _privateConstructorUsedError; + + /// Serializes this SpotubeUserObject to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of SpotubeUserObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $SpotubeUserObjectCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $SpotubeUserObjectCopyWith<$Res> { + factory $SpotubeUserObjectCopyWith( + SpotubeUserObject value, $Res Function(SpotubeUserObject) then) = + _$SpotubeUserObjectCopyWithImpl<$Res, SpotubeUserObject>; + @useResult + $Res call( + {String id, + String name, + List images, + String externalUri}); +} + +/// @nodoc +class _$SpotubeUserObjectCopyWithImpl<$Res, $Val extends SpotubeUserObject> + implements $SpotubeUserObjectCopyWith<$Res> { + _$SpotubeUserObjectCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of SpotubeUserObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? images = null, + Object? externalUri = null, + }) { + return _then(_value.copyWith( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + images: null == images + ? _value.images + : images // ignore: cast_nullable_to_non_nullable + as List, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$SpotubeUserObjectImplCopyWith<$Res> + implements $SpotubeUserObjectCopyWith<$Res> { + factory _$$SpotubeUserObjectImplCopyWith(_$SpotubeUserObjectImpl value, + $Res Function(_$SpotubeUserObjectImpl) then) = + __$$SpotubeUserObjectImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String id, + String name, + List images, + String externalUri}); +} + +/// @nodoc +class __$$SpotubeUserObjectImplCopyWithImpl<$Res> + extends _$SpotubeUserObjectCopyWithImpl<$Res, _$SpotubeUserObjectImpl> + implements _$$SpotubeUserObjectImplCopyWith<$Res> { + __$$SpotubeUserObjectImplCopyWithImpl(_$SpotubeUserObjectImpl _value, + $Res Function(_$SpotubeUserObjectImpl) _then) + : super(_value, _then); + + /// Create a copy of SpotubeUserObject + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? id = null, + Object? name = null, + Object? images = null, + Object? externalUri = null, + }) { + return _then(_$SpotubeUserObjectImpl( + id: null == id + ? _value.id + : id // ignore: cast_nullable_to_non_nullable + as String, + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + images: null == images + ? _value._images + : images // ignore: cast_nullable_to_non_nullable + as List, + externalUri: null == externalUri + ? _value.externalUri + : externalUri // ignore: cast_nullable_to_non_nullable + as String, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$SpotubeUserObjectImpl implements _SpotubeUserObject { + _$SpotubeUserObjectImpl( + {required this.id, + required this.name, + final List images = const [], + required this.externalUri}) + : _images = images; + + factory _$SpotubeUserObjectImpl.fromJson(Map json) => + _$$SpotubeUserObjectImplFromJson(json); + + @override + final String id; + @override + final String name; + final List _images; + @override + @JsonKey() + List get images { + if (_images is EqualUnmodifiableListView) return _images; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_images); + } + + @override + final String externalUri; + + @override + String toString() { + return 'SpotubeUserObject(id: $id, name: $name, images: $images, externalUri: $externalUri)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$SpotubeUserObjectImpl && + (identical(other.id, id) || other.id == id) && + (identical(other.name, name) || other.name == name) && + const DeepCollectionEquality().equals(other._images, _images) && + (identical(other.externalUri, externalUri) || + other.externalUri == externalUri)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, id, name, + const DeepCollectionEquality().hash(_images), externalUri); + + /// Create a copy of SpotubeUserObject + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$SpotubeUserObjectImplCopyWith<_$SpotubeUserObjectImpl> get copyWith => + __$$SpotubeUserObjectImplCopyWithImpl<_$SpotubeUserObjectImpl>( + this, _$identity); + + @override + Map toJson() { + return _$$SpotubeUserObjectImplToJson( + this, + ); + } +} + +abstract class _SpotubeUserObject implements SpotubeUserObject { + factory _SpotubeUserObject( + {required final String id, + required final String name, + final List images, + required final String externalUri}) = _$SpotubeUserObjectImpl; + + factory _SpotubeUserObject.fromJson(Map json) = + _$SpotubeUserObjectImpl.fromJson; + + @override + String get id; + @override + String get name; + @override + List get images; + @override + String get externalUri; + + /// Create a copy of SpotubeUserObject + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$SpotubeUserObjectImplCopyWith<_$SpotubeUserObjectImpl> get copyWith => + throw _privateConstructorUsedError; +} + +PluginConfiguration _$PluginConfigurationFromJson(Map json) { + return _PluginConfiguration.fromJson(json); +} + +/// @nodoc +mixin _$PluginConfiguration { + String get name => throw _privateConstructorUsedError; + String get description => throw _privateConstructorUsedError; + String get version => throw _privateConstructorUsedError; + String get author => throw _privateConstructorUsedError; + String get entryPoint => throw _privateConstructorUsedError; + String get pluginApiVersion => throw _privateConstructorUsedError; + List get apis => throw _privateConstructorUsedError; + List get abilities => throw _privateConstructorUsedError; + String? get repository => throw _privateConstructorUsedError; + + /// Serializes this PluginConfiguration to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of PluginConfiguration + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $PluginConfigurationCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $PluginConfigurationCopyWith<$Res> { + factory $PluginConfigurationCopyWith( + PluginConfiguration value, $Res Function(PluginConfiguration) then) = + _$PluginConfigurationCopyWithImpl<$Res, PluginConfiguration>; + @useResult + $Res call( + {String name, + String description, + String version, + String author, + String entryPoint, + String pluginApiVersion, + List apis, + List abilities, + String? repository}); +} + +/// @nodoc +class _$PluginConfigurationCopyWithImpl<$Res, $Val extends PluginConfiguration> + implements $PluginConfigurationCopyWith<$Res> { + _$PluginConfigurationCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of PluginConfiguration + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? name = null, + Object? description = null, + Object? version = null, + Object? author = null, + Object? entryPoint = null, + Object? pluginApiVersion = null, + Object? apis = null, + Object? abilities = null, + Object? repository = freezed, + }) { + return _then(_value.copyWith( + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + version: null == version + ? _value.version + : version // ignore: cast_nullable_to_non_nullable + as String, + author: null == author + ? _value.author + : author // ignore: cast_nullable_to_non_nullable + as String, + entryPoint: null == entryPoint + ? _value.entryPoint + : entryPoint // ignore: cast_nullable_to_non_nullable + as String, + pluginApiVersion: null == pluginApiVersion + ? _value.pluginApiVersion + : pluginApiVersion // ignore: cast_nullable_to_non_nullable + as String, + apis: null == apis + ? _value.apis + : apis // ignore: cast_nullable_to_non_nullable + as List, + abilities: null == abilities + ? _value.abilities + : abilities // ignore: cast_nullable_to_non_nullable + as List, + repository: freezed == repository + ? _value.repository + : repository // ignore: cast_nullable_to_non_nullable + as String?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$PluginConfigurationImplCopyWith<$Res> + implements $PluginConfigurationCopyWith<$Res> { + factory _$$PluginConfigurationImplCopyWith(_$PluginConfigurationImpl value, + $Res Function(_$PluginConfigurationImpl) then) = + __$$PluginConfigurationImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String name, + String description, + String version, + String author, + String entryPoint, + String pluginApiVersion, + List apis, + List abilities, + String? repository}); +} + +/// @nodoc +class __$$PluginConfigurationImplCopyWithImpl<$Res> + extends _$PluginConfigurationCopyWithImpl<$Res, _$PluginConfigurationImpl> + implements _$$PluginConfigurationImplCopyWith<$Res> { + __$$PluginConfigurationImplCopyWithImpl(_$PluginConfigurationImpl _value, + $Res Function(_$PluginConfigurationImpl) _then) + : super(_value, _then); + + /// Create a copy of PluginConfiguration + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? name = null, + Object? description = null, + Object? version = null, + Object? author = null, + Object? entryPoint = null, + Object? pluginApiVersion = null, + Object? apis = null, + Object? abilities = null, + Object? repository = freezed, + }) { + return _then(_$PluginConfigurationImpl( + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + version: null == version + ? _value.version + : version // ignore: cast_nullable_to_non_nullable + as String, + author: null == author + ? _value.author + : author // ignore: cast_nullable_to_non_nullable + as String, + entryPoint: null == entryPoint + ? _value.entryPoint + : entryPoint // ignore: cast_nullable_to_non_nullable + as String, + pluginApiVersion: null == pluginApiVersion + ? _value.pluginApiVersion + : pluginApiVersion // ignore: cast_nullable_to_non_nullable + as String, + apis: null == apis + ? _value._apis + : apis // ignore: cast_nullable_to_non_nullable + as List, + abilities: null == abilities + ? _value._abilities + : abilities // ignore: cast_nullable_to_non_nullable + as List, + repository: freezed == repository + ? _value.repository + : repository // ignore: cast_nullable_to_non_nullable + as String?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$PluginConfigurationImpl extends _PluginConfiguration { + _$PluginConfigurationImpl( + {required this.name, + required this.description, + required this.version, + required this.author, + required this.entryPoint, + required this.pluginApiVersion, + final List apis = const [], + final List abilities = const [], + this.repository}) + : _apis = apis, + _abilities = abilities, + super._(); + + factory _$PluginConfigurationImpl.fromJson(Map json) => + _$$PluginConfigurationImplFromJson(json); + + @override + final String name; + @override + final String description; + @override + final String version; + @override + final String author; + @override + final String entryPoint; + @override + final String pluginApiVersion; + final List _apis; + @override + @JsonKey() + List get apis { + if (_apis is EqualUnmodifiableListView) return _apis; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_apis); + } + + final List _abilities; + @override + @JsonKey() + List get abilities { + if (_abilities is EqualUnmodifiableListView) return _abilities; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_abilities); + } + + @override + final String? repository; + + @override + String toString() { + return 'PluginConfiguration(name: $name, description: $description, version: $version, author: $author, entryPoint: $entryPoint, pluginApiVersion: $pluginApiVersion, apis: $apis, abilities: $abilities, repository: $repository)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$PluginConfigurationImpl && + (identical(other.name, name) || other.name == name) && + (identical(other.description, description) || + other.description == description) && + (identical(other.version, version) || other.version == version) && + (identical(other.author, author) || other.author == author) && + (identical(other.entryPoint, entryPoint) || + other.entryPoint == entryPoint) && + (identical(other.pluginApiVersion, pluginApiVersion) || + other.pluginApiVersion == pluginApiVersion) && + const DeepCollectionEquality().equals(other._apis, _apis) && + const DeepCollectionEquality() + .equals(other._abilities, _abilities) && + (identical(other.repository, repository) || + other.repository == repository)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + name, + description, + version, + author, + entryPoint, + pluginApiVersion, + const DeepCollectionEquality().hash(_apis), + const DeepCollectionEquality().hash(_abilities), + repository); + + /// Create a copy of PluginConfiguration + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$PluginConfigurationImplCopyWith<_$PluginConfigurationImpl> get copyWith => + __$$PluginConfigurationImplCopyWithImpl<_$PluginConfigurationImpl>( + this, _$identity); + + @override + Map toJson() { + return _$$PluginConfigurationImplToJson( + this, + ); + } +} + +abstract class _PluginConfiguration extends PluginConfiguration { + factory _PluginConfiguration( + {required final String name, + required final String description, + required final String version, + required final String author, + required final String entryPoint, + required final String pluginApiVersion, + final List apis, + final List abilities, + final String? repository}) = _$PluginConfigurationImpl; + _PluginConfiguration._() : super._(); + + factory _PluginConfiguration.fromJson(Map json) = + _$PluginConfigurationImpl.fromJson; + + @override + String get name; + @override + String get description; + @override + String get version; + @override + String get author; + @override + String get entryPoint; + @override + String get pluginApiVersion; + @override + List get apis; + @override + List get abilities; + @override + String? get repository; + + /// Create a copy of PluginConfiguration + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$PluginConfigurationImplCopyWith<_$PluginConfigurationImpl> get copyWith => + throw _privateConstructorUsedError; +} + +PluginUpdateAvailable _$PluginUpdateAvailableFromJson( + Map json) { + return _PluginUpdateAvailable.fromJson(json); +} + +/// @nodoc +mixin _$PluginUpdateAvailable { + String get downloadUrl => throw _privateConstructorUsedError; + String get version => throw _privateConstructorUsedError; + String? get changelog => throw _privateConstructorUsedError; + + /// Serializes this PluginUpdateAvailable to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of PluginUpdateAvailable + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $PluginUpdateAvailableCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $PluginUpdateAvailableCopyWith<$Res> { + factory $PluginUpdateAvailableCopyWith(PluginUpdateAvailable value, + $Res Function(PluginUpdateAvailable) then) = + _$PluginUpdateAvailableCopyWithImpl<$Res, PluginUpdateAvailable>; + @useResult + $Res call({String downloadUrl, String version, String? changelog}); +} + +/// @nodoc +class _$PluginUpdateAvailableCopyWithImpl<$Res, + $Val extends PluginUpdateAvailable> + implements $PluginUpdateAvailableCopyWith<$Res> { + _$PluginUpdateAvailableCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of PluginUpdateAvailable + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? downloadUrl = null, + Object? version = null, + Object? changelog = freezed, + }) { + return _then(_value.copyWith( + downloadUrl: null == downloadUrl + ? _value.downloadUrl + : downloadUrl // ignore: cast_nullable_to_non_nullable + as String, + version: null == version + ? _value.version + : version // ignore: cast_nullable_to_non_nullable + as String, + changelog: freezed == changelog + ? _value.changelog + : changelog // ignore: cast_nullable_to_non_nullable + as String?, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$PluginUpdateAvailableImplCopyWith<$Res> + implements $PluginUpdateAvailableCopyWith<$Res> { + factory _$$PluginUpdateAvailableImplCopyWith( + _$PluginUpdateAvailableImpl value, + $Res Function(_$PluginUpdateAvailableImpl) then) = + __$$PluginUpdateAvailableImplCopyWithImpl<$Res>; + @override + @useResult + $Res call({String downloadUrl, String version, String? changelog}); +} + +/// @nodoc +class __$$PluginUpdateAvailableImplCopyWithImpl<$Res> + extends _$PluginUpdateAvailableCopyWithImpl<$Res, + _$PluginUpdateAvailableImpl> + implements _$$PluginUpdateAvailableImplCopyWith<$Res> { + __$$PluginUpdateAvailableImplCopyWithImpl(_$PluginUpdateAvailableImpl _value, + $Res Function(_$PluginUpdateAvailableImpl) _then) + : super(_value, _then); + + /// Create a copy of PluginUpdateAvailable + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? downloadUrl = null, + Object? version = null, + Object? changelog = freezed, + }) { + return _then(_$PluginUpdateAvailableImpl( + downloadUrl: null == downloadUrl + ? _value.downloadUrl + : downloadUrl // ignore: cast_nullable_to_non_nullable + as String, + version: null == version + ? _value.version + : version // ignore: cast_nullable_to_non_nullable + as String, + changelog: freezed == changelog + ? _value.changelog + : changelog // ignore: cast_nullable_to_non_nullable + as String?, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$PluginUpdateAvailableImpl implements _PluginUpdateAvailable { + _$PluginUpdateAvailableImpl( + {required this.downloadUrl, required this.version, this.changelog}); + + factory _$PluginUpdateAvailableImpl.fromJson(Map json) => + _$$PluginUpdateAvailableImplFromJson(json); + + @override + final String downloadUrl; + @override + final String version; + @override + final String? changelog; + + @override + String toString() { + return 'PluginUpdateAvailable(downloadUrl: $downloadUrl, version: $version, changelog: $changelog)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$PluginUpdateAvailableImpl && + (identical(other.downloadUrl, downloadUrl) || + other.downloadUrl == downloadUrl) && + (identical(other.version, version) || other.version == version) && + (identical(other.changelog, changelog) || + other.changelog == changelog)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, downloadUrl, version, changelog); + + /// Create a copy of PluginUpdateAvailable + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$PluginUpdateAvailableImplCopyWith<_$PluginUpdateAvailableImpl> + get copyWith => __$$PluginUpdateAvailableImplCopyWithImpl< + _$PluginUpdateAvailableImpl>(this, _$identity); + + @override + Map toJson() { + return _$$PluginUpdateAvailableImplToJson( + this, + ); + } +} + +abstract class _PluginUpdateAvailable implements PluginUpdateAvailable { + factory _PluginUpdateAvailable( + {required final String downloadUrl, + required final String version, + final String? changelog}) = _$PluginUpdateAvailableImpl; + + factory _PluginUpdateAvailable.fromJson(Map json) = + _$PluginUpdateAvailableImpl.fromJson; + + @override + String get downloadUrl; + @override + String get version; + @override + String? get changelog; + + /// Create a copy of PluginUpdateAvailable + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$PluginUpdateAvailableImplCopyWith<_$PluginUpdateAvailableImpl> + get copyWith => throw _privateConstructorUsedError; +} + +MetadataPluginRepository _$MetadataPluginRepositoryFromJson( + Map json) { + return _MetadataPluginRepository.fromJson(json); +} + +/// @nodoc +mixin _$MetadataPluginRepository { + String get name => throw _privateConstructorUsedError; + String get owner => throw _privateConstructorUsedError; + String get description => throw _privateConstructorUsedError; + String get repoUrl => throw _privateConstructorUsedError; + List get topics => throw _privateConstructorUsedError; + + /// Serializes this MetadataPluginRepository to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of MetadataPluginRepository + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $MetadataPluginRepositoryCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $MetadataPluginRepositoryCopyWith<$Res> { + factory $MetadataPluginRepositoryCopyWith(MetadataPluginRepository value, + $Res Function(MetadataPluginRepository) then) = + _$MetadataPluginRepositoryCopyWithImpl<$Res, MetadataPluginRepository>; + @useResult + $Res call( + {String name, + String owner, + String description, + String repoUrl, + List topics}); +} + +/// @nodoc +class _$MetadataPluginRepositoryCopyWithImpl<$Res, + $Val extends MetadataPluginRepository> + implements $MetadataPluginRepositoryCopyWith<$Res> { + _$MetadataPluginRepositoryCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of MetadataPluginRepository + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? name = null, + Object? owner = null, + Object? description = null, + Object? repoUrl = null, + Object? topics = null, + }) { + return _then(_value.copyWith( + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + owner: null == owner + ? _value.owner + : owner // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + repoUrl: null == repoUrl + ? _value.repoUrl + : repoUrl // ignore: cast_nullable_to_non_nullable + as String, + topics: null == topics + ? _value.topics + : topics // ignore: cast_nullable_to_non_nullable + as List, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$MetadataPluginRepositoryImplCopyWith<$Res> + implements $MetadataPluginRepositoryCopyWith<$Res> { + factory _$$MetadataPluginRepositoryImplCopyWith( + _$MetadataPluginRepositoryImpl value, + $Res Function(_$MetadataPluginRepositoryImpl) then) = + __$$MetadataPluginRepositoryImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {String name, + String owner, + String description, + String repoUrl, + List topics}); +} + +/// @nodoc +class __$$MetadataPluginRepositoryImplCopyWithImpl<$Res> + extends _$MetadataPluginRepositoryCopyWithImpl<$Res, + _$MetadataPluginRepositoryImpl> + implements _$$MetadataPluginRepositoryImplCopyWith<$Res> { + __$$MetadataPluginRepositoryImplCopyWithImpl( + _$MetadataPluginRepositoryImpl _value, + $Res Function(_$MetadataPluginRepositoryImpl) _then) + : super(_value, _then); + + /// Create a copy of MetadataPluginRepository + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? name = null, + Object? owner = null, + Object? description = null, + Object? repoUrl = null, + Object? topics = null, + }) { + return _then(_$MetadataPluginRepositoryImpl( + name: null == name + ? _value.name + : name // ignore: cast_nullable_to_non_nullable + as String, + owner: null == owner + ? _value.owner + : owner // ignore: cast_nullable_to_non_nullable + as String, + description: null == description + ? _value.description + : description // ignore: cast_nullable_to_non_nullable + as String, + repoUrl: null == repoUrl + ? _value.repoUrl + : repoUrl // ignore: cast_nullable_to_non_nullable + as String, + topics: null == topics + ? _value._topics + : topics // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$MetadataPluginRepositoryImpl implements _MetadataPluginRepository { + _$MetadataPluginRepositoryImpl( + {required this.name, + required this.owner, + required this.description, + required this.repoUrl, + required final List topics}) + : _topics = topics; + + factory _$MetadataPluginRepositoryImpl.fromJson(Map json) => + _$$MetadataPluginRepositoryImplFromJson(json); + + @override + final String name; + @override + final String owner; + @override + final String description; + @override + final String repoUrl; + final List _topics; + @override + List get topics { + if (_topics is EqualUnmodifiableListView) return _topics; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_topics); + } + + @override + String toString() { + return 'MetadataPluginRepository(name: $name, owner: $owner, description: $description, repoUrl: $repoUrl, topics: $topics)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$MetadataPluginRepositoryImpl && + (identical(other.name, name) || other.name == name) && + (identical(other.owner, owner) || other.owner == owner) && + (identical(other.description, description) || + other.description == description) && + (identical(other.repoUrl, repoUrl) || other.repoUrl == repoUrl) && + const DeepCollectionEquality().equals(other._topics, _topics)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash(runtimeType, name, owner, description, + repoUrl, const DeepCollectionEquality().hash(_topics)); + + /// Create a copy of MetadataPluginRepository + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$MetadataPluginRepositoryImplCopyWith<_$MetadataPluginRepositoryImpl> + get copyWith => __$$MetadataPluginRepositoryImplCopyWithImpl< + _$MetadataPluginRepositoryImpl>(this, _$identity); + + @override + Map toJson() { + return _$$MetadataPluginRepositoryImplToJson( + this, + ); + } +} + +abstract class _MetadataPluginRepository implements MetadataPluginRepository { + factory _MetadataPluginRepository( + {required final String name, + required final String owner, + required final String description, + required final String repoUrl, + required final List topics}) = _$MetadataPluginRepositoryImpl; + + factory _MetadataPluginRepository.fromJson(Map json) = + _$MetadataPluginRepositoryImpl.fromJson; + + @override + String get name; + @override + String get owner; + @override + String get description; + @override + String get repoUrl; + @override + List get topics; + + /// Create a copy of MetadataPluginRepository + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$MetadataPluginRepositoryImplCopyWith<_$MetadataPluginRepositoryImpl> + get copyWith => throw _privateConstructorUsedError; +} diff --git a/lib/models/metadata/metadata.g.dart b/lib/models/metadata/metadata.g.dart new file mode 100644 index 00000000..56783d80 --- /dev/null +++ b/lib/models/metadata/metadata.g.dart @@ -0,0 +1,618 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'metadata.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +_$SpotubeAudioSourceContainerPresetLossyImpl + _$$SpotubeAudioSourceContainerPresetLossyImplFromJson(Map json) => + _$SpotubeAudioSourceContainerPresetLossyImpl( + type: $enumDecode(_$SpotubeMediaCompressionTypeEnumMap, json['type']), + name: json['name'] as String, + qualities: (json['qualities'] as List) + .map((e) => SpotubeAudioLossyContainerQuality.fromJson( + Map.from(e as Map))) + .toList(), + ); + +Map _$$SpotubeAudioSourceContainerPresetLossyImplToJson( + _$SpotubeAudioSourceContainerPresetLossyImpl instance) => + { + 'type': _$SpotubeMediaCompressionTypeEnumMap[instance.type]!, + 'name': instance.name, + 'qualities': instance.qualities.map((e) => e.toJson()).toList(), + }; + +const _$SpotubeMediaCompressionTypeEnumMap = { + SpotubeMediaCompressionType.lossy: 'lossy', + SpotubeMediaCompressionType.lossless: 'lossless', +}; + +_$SpotubeAudioSourceContainerPresetLosslessImpl + _$$SpotubeAudioSourceContainerPresetLosslessImplFromJson(Map json) => + _$SpotubeAudioSourceContainerPresetLosslessImpl( + type: $enumDecode(_$SpotubeMediaCompressionTypeEnumMap, json['type']), + name: json['name'] as String, + qualities: (json['qualities'] as List) + .map((e) => SpotubeAudioLosslessContainerQuality.fromJson( + Map.from(e as Map))) + .toList(), + ); + +Map _$$SpotubeAudioSourceContainerPresetLosslessImplToJson( + _$SpotubeAudioSourceContainerPresetLosslessImpl instance) => + { + 'type': _$SpotubeMediaCompressionTypeEnumMap[instance.type]!, + 'name': instance.name, + 'qualities': instance.qualities.map((e) => e.toJson()).toList(), + }; + +_$SpotubeAudioLossyContainerQualityImpl + _$$SpotubeAudioLossyContainerQualityImplFromJson(Map json) => + _$SpotubeAudioLossyContainerQualityImpl( + bitrate: (json['bitrate'] as num).toInt(), + ); + +Map _$$SpotubeAudioLossyContainerQualityImplToJson( + _$SpotubeAudioLossyContainerQualityImpl instance) => + { + 'bitrate': instance.bitrate, + }; + +_$SpotubeAudioLosslessContainerQualityImpl + _$$SpotubeAudioLosslessContainerQualityImplFromJson(Map json) => + _$SpotubeAudioLosslessContainerQualityImpl( + bitDepth: (json['bitDepth'] as num).toInt(), + sampleRate: (json['sampleRate'] as num).toInt(), + ); + +Map _$$SpotubeAudioLosslessContainerQualityImplToJson( + _$SpotubeAudioLosslessContainerQualityImpl instance) => + { + 'bitDepth': instance.bitDepth, + 'sampleRate': instance.sampleRate, + }; + +_$SpotubeAudioSourceMatchObjectImpl + _$$SpotubeAudioSourceMatchObjectImplFromJson(Map json) => + _$SpotubeAudioSourceMatchObjectImpl( + id: json['id'] as String, + title: json['title'] as String, + artists: (json['artists'] as List) + .map((e) => e as String) + .toList(), + duration: Duration(microseconds: (json['duration'] as num).toInt()), + thumbnail: json['thumbnail'] as String?, + externalUri: json['externalUri'] as String, + ); + +Map _$$SpotubeAudioSourceMatchObjectImplToJson( + _$SpotubeAudioSourceMatchObjectImpl instance) => + { + 'id': instance.id, + 'title': instance.title, + 'artists': instance.artists, + 'duration': instance.duration.inMicroseconds, + 'thumbnail': instance.thumbnail, + 'externalUri': instance.externalUri, + }; + +_$SpotubeAudioSourceStreamObjectImpl + _$$SpotubeAudioSourceStreamObjectImplFromJson(Map json) => + _$SpotubeAudioSourceStreamObjectImpl( + url: json['url'] as String, + container: json['container'] as String, + type: $enumDecode(_$SpotubeMediaCompressionTypeEnumMap, json['type']), + codec: json['codec'] as String?, + bitrate: (json['bitrate'] as num?)?.toDouble(), + bitDepth: (json['bitDepth'] as num?)?.toInt(), + sampleRate: (json['sampleRate'] as num?)?.toDouble(), + ); + +Map _$$SpotubeAudioSourceStreamObjectImplToJson( + _$SpotubeAudioSourceStreamObjectImpl instance) => + { + 'url': instance.url, + 'container': instance.container, + 'type': _$SpotubeMediaCompressionTypeEnumMap[instance.type]!, + 'codec': instance.codec, + 'bitrate': instance.bitrate, + 'bitDepth': instance.bitDepth, + 'sampleRate': instance.sampleRate, + }; + +_$SpotubeFullAlbumObjectImpl _$$SpotubeFullAlbumObjectImplFromJson(Map json) => + _$SpotubeFullAlbumObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + artists: (json['artists'] as List) + .map((e) => SpotubeSimpleArtistObject.fromJson( + Map.from(e as Map))) + .toList(), + images: (json['images'] as List?) + ?.map((e) => SpotubeImageObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + releaseDate: json['releaseDate'] as String, + externalUri: json['externalUri'] as String, + totalTracks: (json['totalTracks'] as num).toInt(), + albumType: $enumDecode(_$SpotubeAlbumTypeEnumMap, json['albumType']), + recordLabel: json['recordLabel'] as String?, + genres: + (json['genres'] as List?)?.map((e) => e as String).toList(), + ); + +Map _$$SpotubeFullAlbumObjectImplToJson( + _$SpotubeFullAlbumObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'artists': instance.artists.map((e) => e.toJson()).toList(), + 'images': instance.images.map((e) => e.toJson()).toList(), + 'releaseDate': instance.releaseDate, + 'externalUri': instance.externalUri, + 'totalTracks': instance.totalTracks, + 'albumType': _$SpotubeAlbumTypeEnumMap[instance.albumType]!, + 'recordLabel': instance.recordLabel, + 'genres': instance.genres, + }; + +const _$SpotubeAlbumTypeEnumMap = { + SpotubeAlbumType.album: 'album', + SpotubeAlbumType.single: 'single', + SpotubeAlbumType.compilation: 'compilation', +}; + +_$SpotubeSimpleAlbumObjectImpl _$$SpotubeSimpleAlbumObjectImplFromJson( + Map json) => + _$SpotubeSimpleAlbumObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + externalUri: json['externalUri'] as String, + artists: (json['artists'] as List) + .map((e) => SpotubeSimpleArtistObject.fromJson( + Map.from(e as Map))) + .toList(), + images: (json['images'] as List?) + ?.map((e) => SpotubeImageObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + albumType: $enumDecode(_$SpotubeAlbumTypeEnumMap, json['albumType']), + releaseDate: json['releaseDate'] as String?, + ); + +Map _$$SpotubeSimpleAlbumObjectImplToJson( + _$SpotubeSimpleAlbumObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'externalUri': instance.externalUri, + 'artists': instance.artists.map((e) => e.toJson()).toList(), + 'images': instance.images.map((e) => e.toJson()).toList(), + 'albumType': _$SpotubeAlbumTypeEnumMap[instance.albumType]!, + 'releaseDate': instance.releaseDate, + }; + +_$SpotubeFullArtistObjectImpl _$$SpotubeFullArtistObjectImplFromJson( + Map json) => + _$SpotubeFullArtistObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + externalUri: json['externalUri'] as String, + images: (json['images'] as List?) + ?.map((e) => SpotubeImageObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + genres: + (json['genres'] as List?)?.map((e) => e as String).toList(), + followers: (json['followers'] as num?)?.toInt(), + ); + +Map _$$SpotubeFullArtistObjectImplToJson( + _$SpotubeFullArtistObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'externalUri': instance.externalUri, + 'images': instance.images.map((e) => e.toJson()).toList(), + 'genres': instance.genres, + 'followers': instance.followers, + }; + +_$SpotubeSimpleArtistObjectImpl _$$SpotubeSimpleArtistObjectImplFromJson( + Map json) => + _$SpotubeSimpleArtistObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + externalUri: json['externalUri'] as String, + images: (json['images'] as List?) + ?.map((e) => + SpotubeImageObject.fromJson(Map.from(e as Map))) + .toList(), + ); + +Map _$$SpotubeSimpleArtistObjectImplToJson( + _$SpotubeSimpleArtistObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'externalUri': instance.externalUri, + 'images': instance.images?.map((e) => e.toJson()).toList(), + }; + +_$SpotubeBrowseSectionObjectImpl + _$$SpotubeBrowseSectionObjectImplFromJson( + Map json, + T Function(Object? json) fromJsonT, +) => + _$SpotubeBrowseSectionObjectImpl( + id: json['id'] as String, + title: json['title'] as String, + externalUri: json['externalUri'] as String, + browseMore: json['browseMore'] as bool, + items: (json['items'] as List).map(fromJsonT).toList(), + ); + +Map _$$SpotubeBrowseSectionObjectImplToJson( + _$SpotubeBrowseSectionObjectImpl instance, + Object? Function(T value) toJsonT, +) => + { + 'id': instance.id, + 'title': instance.title, + 'externalUri': instance.externalUri, + 'browseMore': instance.browseMore, + 'items': instance.items.map(toJsonT).toList(), + }; + +_$MetadataFormFieldInputObjectImpl _$$MetadataFormFieldInputObjectImplFromJson( + Map json) => + _$MetadataFormFieldInputObjectImpl( + objectType: json['objectType'] as String, + id: json['id'] as String, + variant: + $enumDecodeNullable(_$FormFieldVariantEnumMap, json['variant']) ?? + FormFieldVariant.text, + placeholder: json['placeholder'] as String?, + defaultValue: json['defaultValue'] as String?, + required: json['required'] as bool?, + regex: json['regex'] as String?, + ); + +Map _$$MetadataFormFieldInputObjectImplToJson( + _$MetadataFormFieldInputObjectImpl instance) => + { + 'objectType': instance.objectType, + 'id': instance.id, + 'variant': _$FormFieldVariantEnumMap[instance.variant]!, + 'placeholder': instance.placeholder, + 'defaultValue': instance.defaultValue, + 'required': instance.required, + 'regex': instance.regex, + }; + +const _$FormFieldVariantEnumMap = { + FormFieldVariant.text: 'text', + FormFieldVariant.password: 'password', + FormFieldVariant.number: 'number', +}; + +_$MetadataFormFieldTextObjectImpl _$$MetadataFormFieldTextObjectImplFromJson( + Map json) => + _$MetadataFormFieldTextObjectImpl( + objectType: json['objectType'] as String, + text: json['text'] as String, + ); + +Map _$$MetadataFormFieldTextObjectImplToJson( + _$MetadataFormFieldTextObjectImpl instance) => + { + 'objectType': instance.objectType, + 'text': instance.text, + }; + +_$SpotubeImageObjectImpl _$$SpotubeImageObjectImplFromJson(Map json) => + _$SpotubeImageObjectImpl( + url: json['url'] as String, + width: (json['width'] as num?)?.toInt(), + height: (json['height'] as num?)?.toInt(), + ); + +Map _$$SpotubeImageObjectImplToJson( + _$SpotubeImageObjectImpl instance) => + { + 'url': instance.url, + 'width': instance.width, + 'height': instance.height, + }; + +_$SpotubePaginationResponseObjectImpl + _$$SpotubePaginationResponseObjectImplFromJson( + Map json, + T Function(Object? json) fromJsonT, +) => + _$SpotubePaginationResponseObjectImpl( + limit: (json['limit'] as num).toInt(), + nextOffset: (json['nextOffset'] as num?)?.toInt(), + total: (json['total'] as num).toInt(), + hasMore: json['hasMore'] as bool, + items: (json['items'] as List).map(fromJsonT).toList(), + ); + +Map _$$SpotubePaginationResponseObjectImplToJson( + _$SpotubePaginationResponseObjectImpl instance, + Object? Function(T value) toJsonT, +) => + { + 'limit': instance.limit, + 'nextOffset': instance.nextOffset, + 'total': instance.total, + 'hasMore': instance.hasMore, + 'items': instance.items.map(toJsonT).toList(), + }; + +_$SpotubeFullPlaylistObjectImpl _$$SpotubeFullPlaylistObjectImplFromJson( + Map json) => + _$SpotubeFullPlaylistObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + description: json['description'] as String, + externalUri: json['externalUri'] as String, + owner: SpotubeUserObject.fromJson( + Map.from(json['owner'] as Map)), + images: (json['images'] as List?) + ?.map((e) => SpotubeImageObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + collaborators: (json['collaborators'] as List?) + ?.map((e) => SpotubeUserObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + collaborative: json['collaborative'] as bool? ?? false, + public: json['public'] as bool? ?? false, + ); + +Map _$$SpotubeFullPlaylistObjectImplToJson( + _$SpotubeFullPlaylistObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'description': instance.description, + 'externalUri': instance.externalUri, + 'owner': instance.owner.toJson(), + 'images': instance.images.map((e) => e.toJson()).toList(), + 'collaborators': instance.collaborators.map((e) => e.toJson()).toList(), + 'collaborative': instance.collaborative, + 'public': instance.public, + }; + +_$SpotubeSimplePlaylistObjectImpl _$$SpotubeSimplePlaylistObjectImplFromJson( + Map json) => + _$SpotubeSimplePlaylistObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + description: json['description'] as String, + externalUri: json['externalUri'] as String, + owner: SpotubeUserObject.fromJson( + Map.from(json['owner'] as Map)), + images: (json['images'] as List?) + ?.map((e) => SpotubeImageObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + ); + +Map _$$SpotubeSimplePlaylistObjectImplToJson( + _$SpotubeSimplePlaylistObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'description': instance.description, + 'externalUri': instance.externalUri, + 'owner': instance.owner.toJson(), + 'images': instance.images.map((e) => e.toJson()).toList(), + }; + +_$SpotubeSearchResponseObjectImpl _$$SpotubeSearchResponseObjectImplFromJson( + Map json) => + _$SpotubeSearchResponseObjectImpl( + albums: (json['albums'] as List) + .map((e) => SpotubeSimpleAlbumObject.fromJson( + Map.from(e as Map))) + .toList(), + artists: (json['artists'] as List) + .map((e) => SpotubeFullArtistObject.fromJson( + Map.from(e as Map))) + .toList(), + playlists: (json['playlists'] as List) + .map((e) => SpotubeSimplePlaylistObject.fromJson( + Map.from(e as Map))) + .toList(), + tracks: (json['tracks'] as List) + .map((e) => SpotubeFullTrackObject.fromJson( + Map.from(e as Map))) + .toList(), + ); + +Map _$$SpotubeSearchResponseObjectImplToJson( + _$SpotubeSearchResponseObjectImpl instance) => + { + 'albums': instance.albums.map((e) => e.toJson()).toList(), + 'artists': instance.artists.map((e) => e.toJson()).toList(), + 'playlists': instance.playlists.map((e) => e.toJson()).toList(), + 'tracks': instance.tracks.map((e) => e.toJson()).toList(), + }; + +_$SpotubeLocalTrackObjectImpl _$$SpotubeLocalTrackObjectImplFromJson( + Map json) => + _$SpotubeLocalTrackObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + externalUri: json['externalUri'] as String, + artists: (json['artists'] as List?) + ?.map((e) => SpotubeSimpleArtistObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + album: SpotubeSimpleAlbumObject.fromJson( + Map.from(json['album'] as Map)), + durationMs: (json['durationMs'] as num).toInt(), + path: json['path'] as String, + $type: json['runtimeType'] as String?, + ); + +Map _$$SpotubeLocalTrackObjectImplToJson( + _$SpotubeLocalTrackObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'externalUri': instance.externalUri, + 'artists': instance.artists.map((e) => e.toJson()).toList(), + 'album': instance.album.toJson(), + 'durationMs': instance.durationMs, + 'path': instance.path, + 'runtimeType': instance.$type, + }; + +_$SpotubeFullTrackObjectImpl _$$SpotubeFullTrackObjectImplFromJson(Map json) => + _$SpotubeFullTrackObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + externalUri: json['externalUri'] as String, + artists: (json['artists'] as List?) + ?.map((e) => SpotubeSimpleArtistObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + album: SpotubeSimpleAlbumObject.fromJson( + Map.from(json['album'] as Map)), + durationMs: (json['durationMs'] as num).toInt(), + isrc: json['isrc'] as String, + explicit: json['explicit'] as bool, + $type: json['runtimeType'] as String?, + ); + +Map _$$SpotubeFullTrackObjectImplToJson( + _$SpotubeFullTrackObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'externalUri': instance.externalUri, + 'artists': instance.artists.map((e) => e.toJson()).toList(), + 'album': instance.album.toJson(), + 'durationMs': instance.durationMs, + 'isrc': instance.isrc, + 'explicit': instance.explicit, + 'runtimeType': instance.$type, + }; + +_$SpotubeUserObjectImpl _$$SpotubeUserObjectImplFromJson(Map json) => + _$SpotubeUserObjectImpl( + id: json['id'] as String, + name: json['name'] as String, + images: (json['images'] as List?) + ?.map((e) => SpotubeImageObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + externalUri: json['externalUri'] as String, + ); + +Map _$$SpotubeUserObjectImplToJson( + _$SpotubeUserObjectImpl instance) => + { + 'id': instance.id, + 'name': instance.name, + 'images': instance.images.map((e) => e.toJson()).toList(), + 'externalUri': instance.externalUri, + }; + +_$PluginConfigurationImpl _$$PluginConfigurationImplFromJson(Map json) => + _$PluginConfigurationImpl( + name: json['name'] as String, + description: json['description'] as String, + version: json['version'] as String, + author: json['author'] as String, + entryPoint: json['entryPoint'] as String, + pluginApiVersion: json['pluginApiVersion'] as String, + apis: (json['apis'] as List?) + ?.map((e) => $enumDecode(_$PluginApisEnumMap, e)) + .toList() ?? + const [], + abilities: (json['abilities'] as List?) + ?.map((e) => $enumDecode(_$PluginAbilitiesEnumMap, e)) + .toList() ?? + const [], + repository: json['repository'] as String?, + ); + +Map _$$PluginConfigurationImplToJson( + _$PluginConfigurationImpl instance) => + { + 'name': instance.name, + 'description': instance.description, + 'version': instance.version, + 'author': instance.author, + 'entryPoint': instance.entryPoint, + 'pluginApiVersion': instance.pluginApiVersion, + 'apis': instance.apis.map((e) => _$PluginApisEnumMap[e]!).toList(), + 'abilities': + instance.abilities.map((e) => _$PluginAbilitiesEnumMap[e]!).toList(), + 'repository': instance.repository, + }; + +const _$PluginApisEnumMap = { + PluginApis.webview: 'webview', + PluginApis.localstorage: 'localstorage', + PluginApis.timezone: 'timezone', +}; + +const _$PluginAbilitiesEnumMap = { + PluginAbilities.authentication: 'authentication', + PluginAbilities.scrobbling: 'scrobbling', + PluginAbilities.metadata: 'metadata', + PluginAbilities.audioSource: 'audio-source', +}; + +_$PluginUpdateAvailableImpl _$$PluginUpdateAvailableImplFromJson(Map json) => + _$PluginUpdateAvailableImpl( + downloadUrl: json['downloadUrl'] as String, + version: json['version'] as String, + changelog: json['changelog'] as String?, + ); + +Map _$$PluginUpdateAvailableImplToJson( + _$PluginUpdateAvailableImpl instance) => + { + 'downloadUrl': instance.downloadUrl, + 'version': instance.version, + 'changelog': instance.changelog, + }; + +_$MetadataPluginRepositoryImpl _$$MetadataPluginRepositoryImplFromJson( + Map json) => + _$MetadataPluginRepositoryImpl( + name: json['name'] as String, + owner: json['owner'] as String, + description: json['description'] as String, + repoUrl: json['repoUrl'] as String, + topics: + (json['topics'] as List).map((e) => e as String).toList(), + ); + +Map _$$MetadataPluginRepositoryImplToJson( + _$MetadataPluginRepositoryImpl instance) => + { + 'name': instance.name, + 'owner': instance.owner, + 'description': instance.description, + 'repoUrl': instance.repoUrl, + 'topics': instance.topics, + }; diff --git a/lib/models/metadata/pagination.dart b/lib/models/metadata/pagination.dart new file mode 100644 index 00000000..093c1d2b --- /dev/null +++ b/lib/models/metadata/pagination.dart @@ -0,0 +1,22 @@ +part of 'metadata.dart'; + +@Freezed(genericArgumentFactories: true) +class SpotubePaginationResponseObject + with _$SpotubePaginationResponseObject { + factory SpotubePaginationResponseObject({ + required int limit, + required int? nextOffset, + required int total, + required bool hasMore, + required List items, + }) = _SpotubePaginationResponseObject; + + factory SpotubePaginationResponseObject.fromJson( + Map json, + T Function(Map json) fromJsonT, + ) => + _$SpotubePaginationResponseObjectFromJson( + json, + (json) => fromJsonT(json as Map), + ); +} diff --git a/lib/models/metadata/playlist.dart b/lib/models/metadata/playlist.dart new file mode 100644 index 00000000..5bb8f1ae --- /dev/null +++ b/lib/models/metadata/playlist.dart @@ -0,0 +1,34 @@ +part of 'metadata.dart'; + +@freezed +class SpotubeFullPlaylistObject with _$SpotubeFullPlaylistObject { + factory SpotubeFullPlaylistObject({ + required String id, + required String name, + required String description, + required String externalUri, + required SpotubeUserObject owner, + @Default([]) List images, + @Default([]) List collaborators, + @Default(false) bool collaborative, + @Default(false) bool public, + }) = _SpotubeFullPlaylistObject; + + factory SpotubeFullPlaylistObject.fromJson(Map json) => + _$SpotubeFullPlaylistObjectFromJson(json); +} + +@freezed +class SpotubeSimplePlaylistObject with _$SpotubeSimplePlaylistObject { + factory SpotubeSimplePlaylistObject({ + required String id, + required String name, + required String description, + required String externalUri, + required SpotubeUserObject owner, + @Default([]) List images, + }) = _SpotubeSimplePlaylistObject; + + factory SpotubeSimplePlaylistObject.fromJson(Map json) => + _$SpotubeSimplePlaylistObjectFromJson(json); +} diff --git a/lib/models/metadata/plugin.dart b/lib/models/metadata/plugin.dart new file mode 100644 index 00000000..6bc84160 --- /dev/null +++ b/lib/models/metadata/plugin.dart @@ -0,0 +1,45 @@ +part of 'metadata.dart'; + +enum PluginApis { webview, localstorage, timezone } + +enum PluginAbilities { + authentication, + scrobbling, + metadata, + @JsonValue('audio-source') + audioSource, +} + +@freezed +class PluginConfiguration with _$PluginConfiguration { + const PluginConfiguration._(); + + factory PluginConfiguration({ + required String name, + required String description, + required String version, + required String author, + required String entryPoint, + required String pluginApiVersion, + @Default([]) List apis, + @Default([]) List abilities, + String? repository, + }) = _PluginConfiguration; + + factory PluginConfiguration.fromJson(Map json) => + _$PluginConfigurationFromJson(json); + + String get slug => name.toLowerCase().replaceAll(RegExp(r'[^a-z0-9]+'), '-'); +} + +@freezed +class PluginUpdateAvailable with _$PluginUpdateAvailable { + factory PluginUpdateAvailable({ + required String downloadUrl, + required String version, + String? changelog, + }) = _PluginUpdateAvailable; + + factory PluginUpdateAvailable.fromJson(Map json) => + _$PluginUpdateAvailableFromJson(json); +} diff --git a/lib/models/metadata/repository.dart b/lib/models/metadata/repository.dart new file mode 100644 index 00000000..2a83f791 --- /dev/null +++ b/lib/models/metadata/repository.dart @@ -0,0 +1,15 @@ +part of './metadata.dart'; + +@freezed +class MetadataPluginRepository with _$MetadataPluginRepository { + factory MetadataPluginRepository({ + required String name, + required String owner, + required String description, + required String repoUrl, + required List topics, + }) = _MetadataPluginRepository; + + factory MetadataPluginRepository.fromJson(Map json) => + _$MetadataPluginRepositoryFromJson(json); +} diff --git a/lib/models/metadata/search.dart b/lib/models/metadata/search.dart new file mode 100644 index 00000000..b39f063a --- /dev/null +++ b/lib/models/metadata/search.dart @@ -0,0 +1,14 @@ +part of 'metadata.dart'; + +@freezed +class SpotubeSearchResponseObject with _$SpotubeSearchResponseObject { + factory SpotubeSearchResponseObject({ + required List albums, + required List artists, + required List playlists, + required List tracks, + }) = _SpotubeSearchResponseObject; + + factory SpotubeSearchResponseObject.fromJson(Map json) => + _$SpotubeSearchResponseObjectFromJson(json); +} diff --git a/lib/models/metadata/track.dart b/lib/models/metadata/track.dart new file mode 100644 index 00000000..ecf7f0a2 --- /dev/null +++ b/lib/models/metadata/track.dart @@ -0,0 +1,119 @@ +part of 'metadata.dart'; + +@freezed +class SpotubeTrackObject with _$SpotubeTrackObject { + factory SpotubeTrackObject.local({ + required String id, + required String name, + required String externalUri, + @Default([]) List artists, + required SpotubeSimpleAlbumObject album, + required int durationMs, + required String path, + }) = SpotubeLocalTrackObject; + + factory SpotubeTrackObject.full({ + required String id, + required String name, + required String externalUri, + @Default([]) List artists, + required SpotubeSimpleAlbumObject album, + required int durationMs, + required String isrc, + required bool explicit, + }) = SpotubeFullTrackObject; + + factory SpotubeTrackObject.localTrackFromFile( + File file, { + Metadata? metadata, + String? art, + }) { + return SpotubeLocalTrackObject( + id: file.absolute.path, + name: metadata?.title ?? basenameWithoutExtension(file.path), + externalUri: "file://${file.absolute.path}", + artists: metadata?.artist?.split(",").map((a) { + return SpotubeSimpleArtistObject( + id: a.trim(), + name: a.trim(), + externalUri: "file://${file.absolute.path}", + ); + }).toList() ?? + [ + SpotubeSimpleArtistObject( + id: "unknown", + name: "Unknown Artist", + externalUri: "file://${file.absolute.path}", + ), + ], + album: SpotubeSimpleAlbumObject( + albumType: SpotubeAlbumType.album, + id: metadata?.album ?? "unknown", + name: metadata?.album ?? "Unknown Album", + externalUri: "file://${file.absolute.path}", + artists: [ + SpotubeSimpleArtistObject( + id: metadata?.albumArtist ?? "unknown", + name: metadata?.albumArtist ?? "Unknown Artist", + externalUri: "file://${file.absolute.path}", + ), + ], + releaseDate: + metadata?.year != null ? "${metadata!.year}-01-01" : "1970-01-01", + images: [ + if (art != null) + SpotubeImageObject( + url: art, + width: 300, + height: 300, + ), + ], + ), + durationMs: metadata?.durationMs?.toInt() ?? 0, + path: file.path, + ); + } + + factory SpotubeTrackObject.fromJson(Map json) => + _$SpotubeTrackObjectFromJson( + json.containsKey("path") + ? {...json, "runtimeType": "local"} + : {...json, "runtimeType": "full"}, + ); +} + +extension AsMediaListSpotubeTrackObject on Iterable { + List asMediaList() { + return map((track) => SpotubeMedia(track)).toList(); + } +} + +extension ToMetadataSpotubeFullTrackObject on SpotubeFullTrackObject { + Metadata toMetadata({ + required int fileLength, + Uint8List? imageBytes, + String? mimeType, + }) { + return Metadata( + title: name, + artist: artists.map((a) => a.name).join(", "), + album: album.name, + albumArtist: artists.map((a) => a.name).join(", "), + year: album.releaseDate == null + ? 1970 + : DateTime.tryParse(album.releaseDate!)?.year ?? + int.tryParse(album.releaseDate!) ?? + 1970, + durationMs: durationMs.toDouble(), + fileSize: BigInt.from(fileLength), + picture: imageBytes != null + ? Picture( + data: imageBytes, + mimeType: mimeType ?? + lookupMimeType("", headerBytes: imageBytes) ?? + "image/jpeg", + ) + : null, + ); + } +} diff --git a/lib/models/metadata/user.dart b/lib/models/metadata/user.dart new file mode 100644 index 00000000..cd041f9c --- /dev/null +++ b/lib/models/metadata/user.dart @@ -0,0 +1,14 @@ +part of 'metadata.dart'; + +@freezed +class SpotubeUserObject with _$SpotubeUserObject { + factory SpotubeUserObject({ + required final String id, + required final String name, + @Default([]) final List images, + required final String externalUri, + }) = _SpotubeUserObject; + + factory SpotubeUserObject.fromJson(Map json) => + _$SpotubeUserObjectFromJson(json); +} diff --git a/lib/models/playback/track_sources.dart b/lib/models/playback/track_sources.dart new file mode 100644 index 00000000..677b34b8 --- /dev/null +++ b/lib/models/playback/track_sources.dart @@ -0,0 +1,24 @@ +import 'package:freezed_annotation/freezed_annotation.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +part 'track_sources.g.dart'; + +@JsonSerializable() +class BasicSourcedTrack { + final SpotubeFullTrackObject query; + final SpotubeAudioSourceMatchObject info; + final String source; + final List sources; + final List siblings; + BasicSourcedTrack({ + required this.query, + required this.source, + required this.info, + required this.sources, + this.siblings = const [], + }); + + factory BasicSourcedTrack.fromJson(Map json) => + _$BasicSourcedTrackFromJson(json); + Map toJson() => _$BasicSourcedTrackToJson(this); +} diff --git a/lib/models/playback/track_sources.g.dart b/lib/models/playback/track_sources.g.dart new file mode 100644 index 00000000..3088493a --- /dev/null +++ b/lib/models/playback/track_sources.g.dart @@ -0,0 +1,33 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'track_sources.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +BasicSourcedTrack _$BasicSourcedTrackFromJson(Map json) => BasicSourcedTrack( + query: SpotubeFullTrackObject.fromJson( + Map.from(json['query'] as Map)), + source: json['source'] as String, + info: SpotubeAudioSourceMatchObject.fromJson( + Map.from(json['info'] as Map)), + sources: (json['sources'] as List) + .map((e) => SpotubeAudioSourceStreamObject.fromJson( + Map.from(e as Map))) + .toList(), + siblings: (json['siblings'] as List?) + ?.map((e) => SpotubeAudioSourceMatchObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + ); + +Map _$BasicSourcedTrackToJson(BasicSourcedTrack instance) => + { + 'query': instance.query.toJson(), + 'info': instance.info.toJson(), + 'source': instance.source, + 'sources': instance.sources.map((e) => e.toJson()).toList(), + 'siblings': instance.siblings.map((e) => e.toJson()).toList(), + }; diff --git a/lib/models/spotify/home_feed.dart b/lib/models/spotify/home_feed.dart deleted file mode 100644 index e5c2f666..00000000 --- a/lib/models/spotify/home_feed.dart +++ /dev/null @@ -1,247 +0,0 @@ -import 'package:freezed_annotation/freezed_annotation.dart'; -import 'package:spotify/spotify.dart'; - -part 'home_feed.freezed.dart'; -part 'home_feed.g.dart'; - -@freezed -class SpotifySectionPlaylist with _$SpotifySectionPlaylist { - const SpotifySectionPlaylist._(); - - const factory SpotifySectionPlaylist({ - required String description, - required String format, - required List images, - required String name, - required String owner, - required String uri, - }) = _SpotifySectionPlaylist; - - factory SpotifySectionPlaylist.fromJson(Map json) => - _$SpotifySectionPlaylistFromJson(json); - - String get id => uri.split(":").last; - - Playlist get asPlaylist { - return Playlist() - ..id = id - ..name = name - ..description = description - ..collaborative = false - ..images = images.map((e) => e.asImage).toList() - ..owner = (User()..displayName = "Spotify") - ..uri = uri - ..type = "playlist"; - } -} - -@freezed -class SpotifySectionArtist with _$SpotifySectionArtist { - const SpotifySectionArtist._(); - - const factory SpotifySectionArtist({ - required String name, - required String uri, - required List images, - }) = _SpotifySectionArtist; - - factory SpotifySectionArtist.fromJson(Map json) => - _$SpotifySectionArtistFromJson(json); - - String get id => uri.split(":").last; - - Artist get asArtist { - return Artist() - ..id = id - ..name = name - ..images = images.map((e) => e.asImage).toList() - ..type = "artist" - ..uri = uri; - } -} - -@freezed -class SpotifySectionAlbum with _$SpotifySectionAlbum { - const SpotifySectionAlbum._(); - - const factory SpotifySectionAlbum({ - required List artists, - required List images, - required String name, - required String uri, - }) = _SpotifySectionAlbum; - - factory SpotifySectionAlbum.fromJson(Map json) => - _$SpotifySectionAlbumFromJson(json); - - String get id => uri.split(":").last; - - Album get asAlbum { - return Album() - ..id = id - ..name = name - ..artists = artists.map((a) => a.asArtist).toList() - ..albumType = AlbumType.album - ..images = images.map((e) => e.asImage).toList() - ..uri = uri; - } -} - -@freezed -class SpotifySectionAlbumArtist with _$SpotifySectionAlbumArtist { - const SpotifySectionAlbumArtist._(); - - const factory SpotifySectionAlbumArtist({ - required String name, - required String uri, - }) = _SpotifySectionAlbumArtist; - - factory SpotifySectionAlbumArtist.fromJson(Map json) => - _$SpotifySectionAlbumArtistFromJson(json); - - String get id => uri.split(":").last; - - Artist get asArtist { - return Artist() - ..id = id - ..name = name - ..type = "artist" - ..uri = uri; - } -} - -@freezed -class SpotifySectionItemImage with _$SpotifySectionItemImage { - const SpotifySectionItemImage._(); - - const factory SpotifySectionItemImage({ - required num? height, - required String url, - required num? width, - }) = _SpotifySectionItemImage; - - factory SpotifySectionItemImage.fromJson(Map json) => - _$SpotifySectionItemImageFromJson(json); - - Image get asImage { - return Image() - ..height = height?.toInt() - ..width = width?.toInt() - ..url = url; - } -} - -@freezed -class SpotifyHomeFeedSectionItem with _$SpotifyHomeFeedSectionItem { - factory SpotifyHomeFeedSectionItem({ - required String typename, - SpotifySectionPlaylist? playlist, - SpotifySectionArtist? artist, - SpotifySectionAlbum? album, - }) = _SpotifyHomeFeedSectionItem; - - factory SpotifyHomeFeedSectionItem.fromJson(Map json) => - _$SpotifyHomeFeedSectionItemFromJson(json); -} - -@freezed -class SpotifyHomeFeedSection with _$SpotifyHomeFeedSection { - factory SpotifyHomeFeedSection({ - required String typename, - String? title, - required String uri, - required List items, - }) = _SpotifyHomeFeedSection; - - factory SpotifyHomeFeedSection.fromJson(Map json) => - _$SpotifyHomeFeedSectionFromJson(json); -} - -@freezed -class SpotifyHomeFeed with _$SpotifyHomeFeed { - factory SpotifyHomeFeed({ - required String greeting, - required List sections, - }) = _SpotifyHomeFeed; - - factory SpotifyHomeFeed.fromJson(Map json) => - _$SpotifyHomeFeedFromJson(json); -} - -Map transformSectionItemTypeJsonMap( - Map json) { - final data = json["content"]["data"]; - final objType = json["content"]["data"]["__typename"]; - return { - "typename": json["content"]["__typename"], - if (objType == "Playlist") - "playlist": { - "name": data["name"], - "description": data["description"], - "format": data["format"], - "images": (data["images"]["items"] as List) - .expand((j) => j["sources"] as dynamic) - .toList() - .cast>(), - "owner": data["ownerV2"]["data"]["name"], - "uri": data["uri"] - }, - if (objType == "Artist") - "artist": { - "name": data["profile"]["name"], - "uri": data["uri"], - "images": data["visuals"]["avatarImage"]["sources"], - }, - if (objType == "Album") - "album": { - "name": data["name"], - "uri": data["uri"], - "images": data["coverArt"]["sources"], - "artists": data["artists"]["items"] - .map( - (artist) => { - "name": artist["profile"]["name"], - "uri": artist["uri"], - }, - ) - .toList() - }, - }; -} - -Map transformSectionItemJsonMap(Map json) { - return { - "typename": json["data"]["__typename"], - "title": json["data"]?["title"]?["text"], - "uri": json["uri"], - "items": (json["sectionItems"]["items"] as List) - .map( - (data) => - transformSectionItemTypeJsonMap(data as Map) - as dynamic, - ) - .where( - (w) => - w["playlist"] != null || - w["artist"] != null || - w["album"] != null, - ) - .toList() - .cast>() - }; -} - -Map transformHomeFeedJsonMap(Map json) { - return { - "greeting": json["data"]["home"]["greeting"]["text"], - "sections": - (json["data"]["home"]["sectionContainer"]["sections"]["items"] as List) - .map( - (item) => - transformSectionItemJsonMap(item as Map) - as dynamic, - ) - .toList() - .cast>() - }; -} diff --git a/lib/models/spotify/home_feed.freezed.dart b/lib/models/spotify/home_feed.freezed.dart deleted file mode 100644 index 5076da29..00000000 --- a/lib/models/spotify/home_feed.freezed.dart +++ /dev/null @@ -1,1776 +0,0 @@ -// coverage:ignore-file -// GENERATED CODE - DO NOT MODIFY BY HAND -// ignore_for_file: type=lint -// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark - -part of 'home_feed.dart'; - -// ************************************************************************** -// FreezedGenerator -// ************************************************************************** - -T _$identity(T value) => value; - -final _privateConstructorUsedError = UnsupportedError( - 'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); - -SpotifySectionPlaylist _$SpotifySectionPlaylistFromJson( - Map json) { - return _SpotifySectionPlaylist.fromJson(json); -} - -/// @nodoc -mixin _$SpotifySectionPlaylist { - String get description => throw _privateConstructorUsedError; - String get format => throw _privateConstructorUsedError; - List get images => - throw _privateConstructorUsedError; - String get name => throw _privateConstructorUsedError; - String get owner => throw _privateConstructorUsedError; - String get uri => throw _privateConstructorUsedError; - - /// Serializes this SpotifySectionPlaylist to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifySectionPlaylist - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifySectionPlaylistCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifySectionPlaylistCopyWith<$Res> { - factory $SpotifySectionPlaylistCopyWith(SpotifySectionPlaylist value, - $Res Function(SpotifySectionPlaylist) then) = - _$SpotifySectionPlaylistCopyWithImpl<$Res, SpotifySectionPlaylist>; - @useResult - $Res call( - {String description, - String format, - List images, - String name, - String owner, - String uri}); -} - -/// @nodoc -class _$SpotifySectionPlaylistCopyWithImpl<$Res, - $Val extends SpotifySectionPlaylist> - implements $SpotifySectionPlaylistCopyWith<$Res> { - _$SpotifySectionPlaylistCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifySectionPlaylist - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? description = null, - Object? format = null, - Object? images = null, - Object? name = null, - Object? owner = null, - Object? uri = null, - }) { - return _then(_value.copyWith( - description: null == description - ? _value.description - : description // ignore: cast_nullable_to_non_nullable - as String, - format: null == format - ? _value.format - : format // ignore: cast_nullable_to_non_nullable - as String, - images: null == images - ? _value.images - : images // ignore: cast_nullable_to_non_nullable - as List, - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - owner: null == owner - ? _value.owner - : owner // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifySectionPlaylistImplCopyWith<$Res> - implements $SpotifySectionPlaylistCopyWith<$Res> { - factory _$$SpotifySectionPlaylistImplCopyWith( - _$SpotifySectionPlaylistImpl value, - $Res Function(_$SpotifySectionPlaylistImpl) then) = - __$$SpotifySectionPlaylistImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {String description, - String format, - List images, - String name, - String owner, - String uri}); -} - -/// @nodoc -class __$$SpotifySectionPlaylistImplCopyWithImpl<$Res> - extends _$SpotifySectionPlaylistCopyWithImpl<$Res, - _$SpotifySectionPlaylistImpl> - implements _$$SpotifySectionPlaylistImplCopyWith<$Res> { - __$$SpotifySectionPlaylistImplCopyWithImpl( - _$SpotifySectionPlaylistImpl _value, - $Res Function(_$SpotifySectionPlaylistImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifySectionPlaylist - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? description = null, - Object? format = null, - Object? images = null, - Object? name = null, - Object? owner = null, - Object? uri = null, - }) { - return _then(_$SpotifySectionPlaylistImpl( - description: null == description - ? _value.description - : description // ignore: cast_nullable_to_non_nullable - as String, - format: null == format - ? _value.format - : format // ignore: cast_nullable_to_non_nullable - as String, - images: null == images - ? _value._images - : images // ignore: cast_nullable_to_non_nullable - as List, - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - owner: null == owner - ? _value.owner - : owner // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifySectionPlaylistImpl extends _SpotifySectionPlaylist { - const _$SpotifySectionPlaylistImpl( - {required this.description, - required this.format, - required final List images, - required this.name, - required this.owner, - required this.uri}) - : _images = images, - super._(); - - factory _$SpotifySectionPlaylistImpl.fromJson(Map json) => - _$$SpotifySectionPlaylistImplFromJson(json); - - @override - final String description; - @override - final String format; - final List _images; - @override - List get images { - if (_images is EqualUnmodifiableListView) return _images; - // ignore: implicit_dynamic_type - return EqualUnmodifiableListView(_images); - } - - @override - final String name; - @override - final String owner; - @override - final String uri; - - @override - String toString() { - return 'SpotifySectionPlaylist(description: $description, format: $format, images: $images, name: $name, owner: $owner, uri: $uri)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifySectionPlaylistImpl && - (identical(other.description, description) || - other.description == description) && - (identical(other.format, format) || other.format == format) && - const DeepCollectionEquality().equals(other._images, _images) && - (identical(other.name, name) || other.name == name) && - (identical(other.owner, owner) || other.owner == owner) && - (identical(other.uri, uri) || other.uri == uri)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash(runtimeType, description, format, - const DeepCollectionEquality().hash(_images), name, owner, uri); - - /// Create a copy of SpotifySectionPlaylist - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifySectionPlaylistImplCopyWith<_$SpotifySectionPlaylistImpl> - get copyWith => __$$SpotifySectionPlaylistImplCopyWithImpl< - _$SpotifySectionPlaylistImpl>(this, _$identity); - - @override - Map toJson() { - return _$$SpotifySectionPlaylistImplToJson( - this, - ); - } -} - -abstract class _SpotifySectionPlaylist extends SpotifySectionPlaylist { - const factory _SpotifySectionPlaylist( - {required final String description, - required final String format, - required final List images, - required final String name, - required final String owner, - required final String uri}) = _$SpotifySectionPlaylistImpl; - const _SpotifySectionPlaylist._() : super._(); - - factory _SpotifySectionPlaylist.fromJson(Map json) = - _$SpotifySectionPlaylistImpl.fromJson; - - @override - String get description; - @override - String get format; - @override - List get images; - @override - String get name; - @override - String get owner; - @override - String get uri; - - /// Create a copy of SpotifySectionPlaylist - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifySectionPlaylistImplCopyWith<_$SpotifySectionPlaylistImpl> - get copyWith => throw _privateConstructorUsedError; -} - -SpotifySectionArtist _$SpotifySectionArtistFromJson(Map json) { - return _SpotifySectionArtist.fromJson(json); -} - -/// @nodoc -mixin _$SpotifySectionArtist { - String get name => throw _privateConstructorUsedError; - String get uri => throw _privateConstructorUsedError; - List get images => - throw _privateConstructorUsedError; - - /// Serializes this SpotifySectionArtist to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifySectionArtist - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifySectionArtistCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifySectionArtistCopyWith<$Res> { - factory $SpotifySectionArtistCopyWith(SpotifySectionArtist value, - $Res Function(SpotifySectionArtist) then) = - _$SpotifySectionArtistCopyWithImpl<$Res, SpotifySectionArtist>; - @useResult - $Res call({String name, String uri, List images}); -} - -/// @nodoc -class _$SpotifySectionArtistCopyWithImpl<$Res, - $Val extends SpotifySectionArtist> - implements $SpotifySectionArtistCopyWith<$Res> { - _$SpotifySectionArtistCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifySectionArtist - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? name = null, - Object? uri = null, - Object? images = null, - }) { - return _then(_value.copyWith( - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - images: null == images - ? _value.images - : images // ignore: cast_nullable_to_non_nullable - as List, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifySectionArtistImplCopyWith<$Res> - implements $SpotifySectionArtistCopyWith<$Res> { - factory _$$SpotifySectionArtistImplCopyWith(_$SpotifySectionArtistImpl value, - $Res Function(_$SpotifySectionArtistImpl) then) = - __$$SpotifySectionArtistImplCopyWithImpl<$Res>; - @override - @useResult - $Res call({String name, String uri, List images}); -} - -/// @nodoc -class __$$SpotifySectionArtistImplCopyWithImpl<$Res> - extends _$SpotifySectionArtistCopyWithImpl<$Res, _$SpotifySectionArtistImpl> - implements _$$SpotifySectionArtistImplCopyWith<$Res> { - __$$SpotifySectionArtistImplCopyWithImpl(_$SpotifySectionArtistImpl _value, - $Res Function(_$SpotifySectionArtistImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifySectionArtist - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? name = null, - Object? uri = null, - Object? images = null, - }) { - return _then(_$SpotifySectionArtistImpl( - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - images: null == images - ? _value._images - : images // ignore: cast_nullable_to_non_nullable - as List, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifySectionArtistImpl extends _SpotifySectionArtist { - const _$SpotifySectionArtistImpl( - {required this.name, - required this.uri, - required final List images}) - : _images = images, - super._(); - - factory _$SpotifySectionArtistImpl.fromJson(Map json) => - _$$SpotifySectionArtistImplFromJson(json); - - @override - final String name; - @override - final String uri; - final List _images; - @override - List get images { - if (_images is EqualUnmodifiableListView) return _images; - // ignore: implicit_dynamic_type - return EqualUnmodifiableListView(_images); - } - - @override - String toString() { - return 'SpotifySectionArtist(name: $name, uri: $uri, images: $images)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifySectionArtistImpl && - (identical(other.name, name) || other.name == name) && - (identical(other.uri, uri) || other.uri == uri) && - const DeepCollectionEquality().equals(other._images, _images)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash( - runtimeType, name, uri, const DeepCollectionEquality().hash(_images)); - - /// Create a copy of SpotifySectionArtist - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifySectionArtistImplCopyWith<_$SpotifySectionArtistImpl> - get copyWith => - __$$SpotifySectionArtistImplCopyWithImpl<_$SpotifySectionArtistImpl>( - this, _$identity); - - @override - Map toJson() { - return _$$SpotifySectionArtistImplToJson( - this, - ); - } -} - -abstract class _SpotifySectionArtist extends SpotifySectionArtist { - const factory _SpotifySectionArtist( - {required final String name, - required final String uri, - required final List images}) = - _$SpotifySectionArtistImpl; - const _SpotifySectionArtist._() : super._(); - - factory _SpotifySectionArtist.fromJson(Map json) = - _$SpotifySectionArtistImpl.fromJson; - - @override - String get name; - @override - String get uri; - @override - List get images; - - /// Create a copy of SpotifySectionArtist - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifySectionArtistImplCopyWith<_$SpotifySectionArtistImpl> - get copyWith => throw _privateConstructorUsedError; -} - -SpotifySectionAlbum _$SpotifySectionAlbumFromJson(Map json) { - return _SpotifySectionAlbum.fromJson(json); -} - -/// @nodoc -mixin _$SpotifySectionAlbum { - List get artists => - throw _privateConstructorUsedError; - List get images => - throw _privateConstructorUsedError; - String get name => throw _privateConstructorUsedError; - String get uri => throw _privateConstructorUsedError; - - /// Serializes this SpotifySectionAlbum to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifySectionAlbum - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifySectionAlbumCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifySectionAlbumCopyWith<$Res> { - factory $SpotifySectionAlbumCopyWith( - SpotifySectionAlbum value, $Res Function(SpotifySectionAlbum) then) = - _$SpotifySectionAlbumCopyWithImpl<$Res, SpotifySectionAlbum>; - @useResult - $Res call( - {List artists, - List images, - String name, - String uri}); -} - -/// @nodoc -class _$SpotifySectionAlbumCopyWithImpl<$Res, $Val extends SpotifySectionAlbum> - implements $SpotifySectionAlbumCopyWith<$Res> { - _$SpotifySectionAlbumCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifySectionAlbum - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? artists = null, - Object? images = null, - Object? name = null, - Object? uri = null, - }) { - return _then(_value.copyWith( - artists: null == artists - ? _value.artists - : artists // ignore: cast_nullable_to_non_nullable - as List, - images: null == images - ? _value.images - : images // ignore: cast_nullable_to_non_nullable - as List, - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifySectionAlbumImplCopyWith<$Res> - implements $SpotifySectionAlbumCopyWith<$Res> { - factory _$$SpotifySectionAlbumImplCopyWith(_$SpotifySectionAlbumImpl value, - $Res Function(_$SpotifySectionAlbumImpl) then) = - __$$SpotifySectionAlbumImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {List artists, - List images, - String name, - String uri}); -} - -/// @nodoc -class __$$SpotifySectionAlbumImplCopyWithImpl<$Res> - extends _$SpotifySectionAlbumCopyWithImpl<$Res, _$SpotifySectionAlbumImpl> - implements _$$SpotifySectionAlbumImplCopyWith<$Res> { - __$$SpotifySectionAlbumImplCopyWithImpl(_$SpotifySectionAlbumImpl _value, - $Res Function(_$SpotifySectionAlbumImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifySectionAlbum - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? artists = null, - Object? images = null, - Object? name = null, - Object? uri = null, - }) { - return _then(_$SpotifySectionAlbumImpl( - artists: null == artists - ? _value._artists - : artists // ignore: cast_nullable_to_non_nullable - as List, - images: null == images - ? _value._images - : images // ignore: cast_nullable_to_non_nullable - as List, - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifySectionAlbumImpl extends _SpotifySectionAlbum { - const _$SpotifySectionAlbumImpl( - {required final List artists, - required final List images, - required this.name, - required this.uri}) - : _artists = artists, - _images = images, - super._(); - - factory _$SpotifySectionAlbumImpl.fromJson(Map json) => - _$$SpotifySectionAlbumImplFromJson(json); - - final List _artists; - @override - List get artists { - if (_artists is EqualUnmodifiableListView) return _artists; - // ignore: implicit_dynamic_type - return EqualUnmodifiableListView(_artists); - } - - final List _images; - @override - List get images { - if (_images is EqualUnmodifiableListView) return _images; - // ignore: implicit_dynamic_type - return EqualUnmodifiableListView(_images); - } - - @override - final String name; - @override - final String uri; - - @override - String toString() { - return 'SpotifySectionAlbum(artists: $artists, images: $images, name: $name, uri: $uri)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifySectionAlbumImpl && - const DeepCollectionEquality().equals(other._artists, _artists) && - const DeepCollectionEquality().equals(other._images, _images) && - (identical(other.name, name) || other.name == name) && - (identical(other.uri, uri) || other.uri == uri)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash( - runtimeType, - const DeepCollectionEquality().hash(_artists), - const DeepCollectionEquality().hash(_images), - name, - uri); - - /// Create a copy of SpotifySectionAlbum - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifySectionAlbumImplCopyWith<_$SpotifySectionAlbumImpl> get copyWith => - __$$SpotifySectionAlbumImplCopyWithImpl<_$SpotifySectionAlbumImpl>( - this, _$identity); - - @override - Map toJson() { - return _$$SpotifySectionAlbumImplToJson( - this, - ); - } -} - -abstract class _SpotifySectionAlbum extends SpotifySectionAlbum { - const factory _SpotifySectionAlbum( - {required final List artists, - required final List images, - required final String name, - required final String uri}) = _$SpotifySectionAlbumImpl; - const _SpotifySectionAlbum._() : super._(); - - factory _SpotifySectionAlbum.fromJson(Map json) = - _$SpotifySectionAlbumImpl.fromJson; - - @override - List get artists; - @override - List get images; - @override - String get name; - @override - String get uri; - - /// Create a copy of SpotifySectionAlbum - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifySectionAlbumImplCopyWith<_$SpotifySectionAlbumImpl> get copyWith => - throw _privateConstructorUsedError; -} - -SpotifySectionAlbumArtist _$SpotifySectionAlbumArtistFromJson( - Map json) { - return _SpotifySectionAlbumArtist.fromJson(json); -} - -/// @nodoc -mixin _$SpotifySectionAlbumArtist { - String get name => throw _privateConstructorUsedError; - String get uri => throw _privateConstructorUsedError; - - /// Serializes this SpotifySectionAlbumArtist to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifySectionAlbumArtist - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifySectionAlbumArtistCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifySectionAlbumArtistCopyWith<$Res> { - factory $SpotifySectionAlbumArtistCopyWith(SpotifySectionAlbumArtist value, - $Res Function(SpotifySectionAlbumArtist) then) = - _$SpotifySectionAlbumArtistCopyWithImpl<$Res, SpotifySectionAlbumArtist>; - @useResult - $Res call({String name, String uri}); -} - -/// @nodoc -class _$SpotifySectionAlbumArtistCopyWithImpl<$Res, - $Val extends SpotifySectionAlbumArtist> - implements $SpotifySectionAlbumArtistCopyWith<$Res> { - _$SpotifySectionAlbumArtistCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifySectionAlbumArtist - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? name = null, - Object? uri = null, - }) { - return _then(_value.copyWith( - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifySectionAlbumArtistImplCopyWith<$Res> - implements $SpotifySectionAlbumArtistCopyWith<$Res> { - factory _$$SpotifySectionAlbumArtistImplCopyWith( - _$SpotifySectionAlbumArtistImpl value, - $Res Function(_$SpotifySectionAlbumArtistImpl) then) = - __$$SpotifySectionAlbumArtistImplCopyWithImpl<$Res>; - @override - @useResult - $Res call({String name, String uri}); -} - -/// @nodoc -class __$$SpotifySectionAlbumArtistImplCopyWithImpl<$Res> - extends _$SpotifySectionAlbumArtistCopyWithImpl<$Res, - _$SpotifySectionAlbumArtistImpl> - implements _$$SpotifySectionAlbumArtistImplCopyWith<$Res> { - __$$SpotifySectionAlbumArtistImplCopyWithImpl( - _$SpotifySectionAlbumArtistImpl _value, - $Res Function(_$SpotifySectionAlbumArtistImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifySectionAlbumArtist - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? name = null, - Object? uri = null, - }) { - return _then(_$SpotifySectionAlbumArtistImpl( - name: null == name - ? _value.name - : name // ignore: cast_nullable_to_non_nullable - as String, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifySectionAlbumArtistImpl extends _SpotifySectionAlbumArtist { - const _$SpotifySectionAlbumArtistImpl({required this.name, required this.uri}) - : super._(); - - factory _$SpotifySectionAlbumArtistImpl.fromJson(Map json) => - _$$SpotifySectionAlbumArtistImplFromJson(json); - - @override - final String name; - @override - final String uri; - - @override - String toString() { - return 'SpotifySectionAlbumArtist(name: $name, uri: $uri)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifySectionAlbumArtistImpl && - (identical(other.name, name) || other.name == name) && - (identical(other.uri, uri) || other.uri == uri)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash(runtimeType, name, uri); - - /// Create a copy of SpotifySectionAlbumArtist - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifySectionAlbumArtistImplCopyWith<_$SpotifySectionAlbumArtistImpl> - get copyWith => __$$SpotifySectionAlbumArtistImplCopyWithImpl< - _$SpotifySectionAlbumArtistImpl>(this, _$identity); - - @override - Map toJson() { - return _$$SpotifySectionAlbumArtistImplToJson( - this, - ); - } -} - -abstract class _SpotifySectionAlbumArtist extends SpotifySectionAlbumArtist { - const factory _SpotifySectionAlbumArtist( - {required final String name, - required final String uri}) = _$SpotifySectionAlbumArtistImpl; - const _SpotifySectionAlbumArtist._() : super._(); - - factory _SpotifySectionAlbumArtist.fromJson(Map json) = - _$SpotifySectionAlbumArtistImpl.fromJson; - - @override - String get name; - @override - String get uri; - - /// Create a copy of SpotifySectionAlbumArtist - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifySectionAlbumArtistImplCopyWith<_$SpotifySectionAlbumArtistImpl> - get copyWith => throw _privateConstructorUsedError; -} - -SpotifySectionItemImage _$SpotifySectionItemImageFromJson( - Map json) { - return _SpotifySectionItemImage.fromJson(json); -} - -/// @nodoc -mixin _$SpotifySectionItemImage { - num? get height => throw _privateConstructorUsedError; - String get url => throw _privateConstructorUsedError; - num? get width => throw _privateConstructorUsedError; - - /// Serializes this SpotifySectionItemImage to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifySectionItemImage - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifySectionItemImageCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifySectionItemImageCopyWith<$Res> { - factory $SpotifySectionItemImageCopyWith(SpotifySectionItemImage value, - $Res Function(SpotifySectionItemImage) then) = - _$SpotifySectionItemImageCopyWithImpl<$Res, SpotifySectionItemImage>; - @useResult - $Res call({num? height, String url, num? width}); -} - -/// @nodoc -class _$SpotifySectionItemImageCopyWithImpl<$Res, - $Val extends SpotifySectionItemImage> - implements $SpotifySectionItemImageCopyWith<$Res> { - _$SpotifySectionItemImageCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifySectionItemImage - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? height = freezed, - Object? url = null, - Object? width = freezed, - }) { - return _then(_value.copyWith( - height: freezed == height - ? _value.height - : height // ignore: cast_nullable_to_non_nullable - as num?, - url: null == url - ? _value.url - : url // ignore: cast_nullable_to_non_nullable - as String, - width: freezed == width - ? _value.width - : width // ignore: cast_nullable_to_non_nullable - as num?, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifySectionItemImageImplCopyWith<$Res> - implements $SpotifySectionItemImageCopyWith<$Res> { - factory _$$SpotifySectionItemImageImplCopyWith( - _$SpotifySectionItemImageImpl value, - $Res Function(_$SpotifySectionItemImageImpl) then) = - __$$SpotifySectionItemImageImplCopyWithImpl<$Res>; - @override - @useResult - $Res call({num? height, String url, num? width}); -} - -/// @nodoc -class __$$SpotifySectionItemImageImplCopyWithImpl<$Res> - extends _$SpotifySectionItemImageCopyWithImpl<$Res, - _$SpotifySectionItemImageImpl> - implements _$$SpotifySectionItemImageImplCopyWith<$Res> { - __$$SpotifySectionItemImageImplCopyWithImpl( - _$SpotifySectionItemImageImpl _value, - $Res Function(_$SpotifySectionItemImageImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifySectionItemImage - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? height = freezed, - Object? url = null, - Object? width = freezed, - }) { - return _then(_$SpotifySectionItemImageImpl( - height: freezed == height - ? _value.height - : height // ignore: cast_nullable_to_non_nullable - as num?, - url: null == url - ? _value.url - : url // ignore: cast_nullable_to_non_nullable - as String, - width: freezed == width - ? _value.width - : width // ignore: cast_nullable_to_non_nullable - as num?, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifySectionItemImageImpl extends _SpotifySectionItemImage { - const _$SpotifySectionItemImageImpl( - {required this.height, required this.url, required this.width}) - : super._(); - - factory _$SpotifySectionItemImageImpl.fromJson(Map json) => - _$$SpotifySectionItemImageImplFromJson(json); - - @override - final num? height; - @override - final String url; - @override - final num? width; - - @override - String toString() { - return 'SpotifySectionItemImage(height: $height, url: $url, width: $width)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifySectionItemImageImpl && - (identical(other.height, height) || other.height == height) && - (identical(other.url, url) || other.url == url) && - (identical(other.width, width) || other.width == width)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash(runtimeType, height, url, width); - - /// Create a copy of SpotifySectionItemImage - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifySectionItemImageImplCopyWith<_$SpotifySectionItemImageImpl> - get copyWith => __$$SpotifySectionItemImageImplCopyWithImpl< - _$SpotifySectionItemImageImpl>(this, _$identity); - - @override - Map toJson() { - return _$$SpotifySectionItemImageImplToJson( - this, - ); - } -} - -abstract class _SpotifySectionItemImage extends SpotifySectionItemImage { - const factory _SpotifySectionItemImage( - {required final num? height, - required final String url, - required final num? width}) = _$SpotifySectionItemImageImpl; - const _SpotifySectionItemImage._() : super._(); - - factory _SpotifySectionItemImage.fromJson(Map json) = - _$SpotifySectionItemImageImpl.fromJson; - - @override - num? get height; - @override - String get url; - @override - num? get width; - - /// Create a copy of SpotifySectionItemImage - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifySectionItemImageImplCopyWith<_$SpotifySectionItemImageImpl> - get copyWith => throw _privateConstructorUsedError; -} - -SpotifyHomeFeedSectionItem _$SpotifyHomeFeedSectionItemFromJson( - Map json) { - return _SpotifyHomeFeedSectionItem.fromJson(json); -} - -/// @nodoc -mixin _$SpotifyHomeFeedSectionItem { - String get typename => throw _privateConstructorUsedError; - SpotifySectionPlaylist? get playlist => throw _privateConstructorUsedError; - SpotifySectionArtist? get artist => throw _privateConstructorUsedError; - SpotifySectionAlbum? get album => throw _privateConstructorUsedError; - - /// Serializes this SpotifyHomeFeedSectionItem to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifyHomeFeedSectionItemCopyWith - get copyWith => throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifyHomeFeedSectionItemCopyWith<$Res> { - factory $SpotifyHomeFeedSectionItemCopyWith(SpotifyHomeFeedSectionItem value, - $Res Function(SpotifyHomeFeedSectionItem) then) = - _$SpotifyHomeFeedSectionItemCopyWithImpl<$Res, - SpotifyHomeFeedSectionItem>; - @useResult - $Res call( - {String typename, - SpotifySectionPlaylist? playlist, - SpotifySectionArtist? artist, - SpotifySectionAlbum? album}); - - $SpotifySectionPlaylistCopyWith<$Res>? get playlist; - $SpotifySectionArtistCopyWith<$Res>? get artist; - $SpotifySectionAlbumCopyWith<$Res>? get album; -} - -/// @nodoc -class _$SpotifyHomeFeedSectionItemCopyWithImpl<$Res, - $Val extends SpotifyHomeFeedSectionItem> - implements $SpotifyHomeFeedSectionItemCopyWith<$Res> { - _$SpotifyHomeFeedSectionItemCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? typename = null, - Object? playlist = freezed, - Object? artist = freezed, - Object? album = freezed, - }) { - return _then(_value.copyWith( - typename: null == typename - ? _value.typename - : typename // ignore: cast_nullable_to_non_nullable - as String, - playlist: freezed == playlist - ? _value.playlist - : playlist // ignore: cast_nullable_to_non_nullable - as SpotifySectionPlaylist?, - artist: freezed == artist - ? _value.artist - : artist // ignore: cast_nullable_to_non_nullable - as SpotifySectionArtist?, - album: freezed == album - ? _value.album - : album // ignore: cast_nullable_to_non_nullable - as SpotifySectionAlbum?, - ) as $Val); - } - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @override - @pragma('vm:prefer-inline') - $SpotifySectionPlaylistCopyWith<$Res>? get playlist { - if (_value.playlist == null) { - return null; - } - - return $SpotifySectionPlaylistCopyWith<$Res>(_value.playlist!, (value) { - return _then(_value.copyWith(playlist: value) as $Val); - }); - } - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @override - @pragma('vm:prefer-inline') - $SpotifySectionArtistCopyWith<$Res>? get artist { - if (_value.artist == null) { - return null; - } - - return $SpotifySectionArtistCopyWith<$Res>(_value.artist!, (value) { - return _then(_value.copyWith(artist: value) as $Val); - }); - } - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @override - @pragma('vm:prefer-inline') - $SpotifySectionAlbumCopyWith<$Res>? get album { - if (_value.album == null) { - return null; - } - - return $SpotifySectionAlbumCopyWith<$Res>(_value.album!, (value) { - return _then(_value.copyWith(album: value) as $Val); - }); - } -} - -/// @nodoc -abstract class _$$SpotifyHomeFeedSectionItemImplCopyWith<$Res> - implements $SpotifyHomeFeedSectionItemCopyWith<$Res> { - factory _$$SpotifyHomeFeedSectionItemImplCopyWith( - _$SpotifyHomeFeedSectionItemImpl value, - $Res Function(_$SpotifyHomeFeedSectionItemImpl) then) = - __$$SpotifyHomeFeedSectionItemImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {String typename, - SpotifySectionPlaylist? playlist, - SpotifySectionArtist? artist, - SpotifySectionAlbum? album}); - - @override - $SpotifySectionPlaylistCopyWith<$Res>? get playlist; - @override - $SpotifySectionArtistCopyWith<$Res>? get artist; - @override - $SpotifySectionAlbumCopyWith<$Res>? get album; -} - -/// @nodoc -class __$$SpotifyHomeFeedSectionItemImplCopyWithImpl<$Res> - extends _$SpotifyHomeFeedSectionItemCopyWithImpl<$Res, - _$SpotifyHomeFeedSectionItemImpl> - implements _$$SpotifyHomeFeedSectionItemImplCopyWith<$Res> { - __$$SpotifyHomeFeedSectionItemImplCopyWithImpl( - _$SpotifyHomeFeedSectionItemImpl _value, - $Res Function(_$SpotifyHomeFeedSectionItemImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? typename = null, - Object? playlist = freezed, - Object? artist = freezed, - Object? album = freezed, - }) { - return _then(_$SpotifyHomeFeedSectionItemImpl( - typename: null == typename - ? _value.typename - : typename // ignore: cast_nullable_to_non_nullable - as String, - playlist: freezed == playlist - ? _value.playlist - : playlist // ignore: cast_nullable_to_non_nullable - as SpotifySectionPlaylist?, - artist: freezed == artist - ? _value.artist - : artist // ignore: cast_nullable_to_non_nullable - as SpotifySectionArtist?, - album: freezed == album - ? _value.album - : album // ignore: cast_nullable_to_non_nullable - as SpotifySectionAlbum?, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifyHomeFeedSectionItemImpl implements _SpotifyHomeFeedSectionItem { - _$SpotifyHomeFeedSectionItemImpl( - {required this.typename, this.playlist, this.artist, this.album}); - - factory _$SpotifyHomeFeedSectionItemImpl.fromJson( - Map json) => - _$$SpotifyHomeFeedSectionItemImplFromJson(json); - - @override - final String typename; - @override - final SpotifySectionPlaylist? playlist; - @override - final SpotifySectionArtist? artist; - @override - final SpotifySectionAlbum? album; - - @override - String toString() { - return 'SpotifyHomeFeedSectionItem(typename: $typename, playlist: $playlist, artist: $artist, album: $album)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifyHomeFeedSectionItemImpl && - (identical(other.typename, typename) || - other.typename == typename) && - (identical(other.playlist, playlist) || - other.playlist == playlist) && - (identical(other.artist, artist) || other.artist == artist) && - (identical(other.album, album) || other.album == album)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => - Object.hash(runtimeType, typename, playlist, artist, album); - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifyHomeFeedSectionItemImplCopyWith<_$SpotifyHomeFeedSectionItemImpl> - get copyWith => __$$SpotifyHomeFeedSectionItemImplCopyWithImpl< - _$SpotifyHomeFeedSectionItemImpl>(this, _$identity); - - @override - Map toJson() { - return _$$SpotifyHomeFeedSectionItemImplToJson( - this, - ); - } -} - -abstract class _SpotifyHomeFeedSectionItem - implements SpotifyHomeFeedSectionItem { - factory _SpotifyHomeFeedSectionItem( - {required final String typename, - final SpotifySectionPlaylist? playlist, - final SpotifySectionArtist? artist, - final SpotifySectionAlbum? album}) = _$SpotifyHomeFeedSectionItemImpl; - - factory _SpotifyHomeFeedSectionItem.fromJson(Map json) = - _$SpotifyHomeFeedSectionItemImpl.fromJson; - - @override - String get typename; - @override - SpotifySectionPlaylist? get playlist; - @override - SpotifySectionArtist? get artist; - @override - SpotifySectionAlbum? get album; - - /// Create a copy of SpotifyHomeFeedSectionItem - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifyHomeFeedSectionItemImplCopyWith<_$SpotifyHomeFeedSectionItemImpl> - get copyWith => throw _privateConstructorUsedError; -} - -SpotifyHomeFeedSection _$SpotifyHomeFeedSectionFromJson( - Map json) { - return _SpotifyHomeFeedSection.fromJson(json); -} - -/// @nodoc -mixin _$SpotifyHomeFeedSection { - String get typename => throw _privateConstructorUsedError; - String? get title => throw _privateConstructorUsedError; - String get uri => throw _privateConstructorUsedError; - List get items => - throw _privateConstructorUsedError; - - /// Serializes this SpotifyHomeFeedSection to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifyHomeFeedSection - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifyHomeFeedSectionCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifyHomeFeedSectionCopyWith<$Res> { - factory $SpotifyHomeFeedSectionCopyWith(SpotifyHomeFeedSection value, - $Res Function(SpotifyHomeFeedSection) then) = - _$SpotifyHomeFeedSectionCopyWithImpl<$Res, SpotifyHomeFeedSection>; - @useResult - $Res call( - {String typename, - String? title, - String uri, - List items}); -} - -/// @nodoc -class _$SpotifyHomeFeedSectionCopyWithImpl<$Res, - $Val extends SpotifyHomeFeedSection> - implements $SpotifyHomeFeedSectionCopyWith<$Res> { - _$SpotifyHomeFeedSectionCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifyHomeFeedSection - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? typename = null, - Object? title = freezed, - Object? uri = null, - Object? items = null, - }) { - return _then(_value.copyWith( - typename: null == typename - ? _value.typename - : typename // ignore: cast_nullable_to_non_nullable - as String, - title: freezed == title - ? _value.title - : title // ignore: cast_nullable_to_non_nullable - as String?, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - items: null == items - ? _value.items - : items // ignore: cast_nullable_to_non_nullable - as List, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifyHomeFeedSectionImplCopyWith<$Res> - implements $SpotifyHomeFeedSectionCopyWith<$Res> { - factory _$$SpotifyHomeFeedSectionImplCopyWith( - _$SpotifyHomeFeedSectionImpl value, - $Res Function(_$SpotifyHomeFeedSectionImpl) then) = - __$$SpotifyHomeFeedSectionImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {String typename, - String? title, - String uri, - List items}); -} - -/// @nodoc -class __$$SpotifyHomeFeedSectionImplCopyWithImpl<$Res> - extends _$SpotifyHomeFeedSectionCopyWithImpl<$Res, - _$SpotifyHomeFeedSectionImpl> - implements _$$SpotifyHomeFeedSectionImplCopyWith<$Res> { - __$$SpotifyHomeFeedSectionImplCopyWithImpl( - _$SpotifyHomeFeedSectionImpl _value, - $Res Function(_$SpotifyHomeFeedSectionImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifyHomeFeedSection - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? typename = null, - Object? title = freezed, - Object? uri = null, - Object? items = null, - }) { - return _then(_$SpotifyHomeFeedSectionImpl( - typename: null == typename - ? _value.typename - : typename // ignore: cast_nullable_to_non_nullable - as String, - title: freezed == title - ? _value.title - : title // ignore: cast_nullable_to_non_nullable - as String?, - uri: null == uri - ? _value.uri - : uri // ignore: cast_nullable_to_non_nullable - as String, - items: null == items - ? _value._items - : items // ignore: cast_nullable_to_non_nullable - as List, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifyHomeFeedSectionImpl implements _SpotifyHomeFeedSection { - _$SpotifyHomeFeedSectionImpl( - {required this.typename, - this.title, - required this.uri, - required final List items}) - : _items = items; - - factory _$SpotifyHomeFeedSectionImpl.fromJson(Map json) => - _$$SpotifyHomeFeedSectionImplFromJson(json); - - @override - final String typename; - @override - final String? title; - @override - final String uri; - final List _items; - @override - List get items { - if (_items is EqualUnmodifiableListView) return _items; - // ignore: implicit_dynamic_type - return EqualUnmodifiableListView(_items); - } - - @override - String toString() { - return 'SpotifyHomeFeedSection(typename: $typename, title: $title, uri: $uri, items: $items)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifyHomeFeedSectionImpl && - (identical(other.typename, typename) || - other.typename == typename) && - (identical(other.title, title) || other.title == title) && - (identical(other.uri, uri) || other.uri == uri) && - const DeepCollectionEquality().equals(other._items, _items)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash(runtimeType, typename, title, uri, - const DeepCollectionEquality().hash(_items)); - - /// Create a copy of SpotifyHomeFeedSection - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifyHomeFeedSectionImplCopyWith<_$SpotifyHomeFeedSectionImpl> - get copyWith => __$$SpotifyHomeFeedSectionImplCopyWithImpl< - _$SpotifyHomeFeedSectionImpl>(this, _$identity); - - @override - Map toJson() { - return _$$SpotifyHomeFeedSectionImplToJson( - this, - ); - } -} - -abstract class _SpotifyHomeFeedSection implements SpotifyHomeFeedSection { - factory _SpotifyHomeFeedSection( - {required final String typename, - final String? title, - required final String uri, - required final List items}) = - _$SpotifyHomeFeedSectionImpl; - - factory _SpotifyHomeFeedSection.fromJson(Map json) = - _$SpotifyHomeFeedSectionImpl.fromJson; - - @override - String get typename; - @override - String? get title; - @override - String get uri; - @override - List get items; - - /// Create a copy of SpotifyHomeFeedSection - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifyHomeFeedSectionImplCopyWith<_$SpotifyHomeFeedSectionImpl> - get copyWith => throw _privateConstructorUsedError; -} - -SpotifyHomeFeed _$SpotifyHomeFeedFromJson(Map json) { - return _SpotifyHomeFeed.fromJson(json); -} - -/// @nodoc -mixin _$SpotifyHomeFeed { - String get greeting => throw _privateConstructorUsedError; - List get sections => - throw _privateConstructorUsedError; - - /// Serializes this SpotifyHomeFeed to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SpotifyHomeFeed - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SpotifyHomeFeedCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SpotifyHomeFeedCopyWith<$Res> { - factory $SpotifyHomeFeedCopyWith( - SpotifyHomeFeed value, $Res Function(SpotifyHomeFeed) then) = - _$SpotifyHomeFeedCopyWithImpl<$Res, SpotifyHomeFeed>; - @useResult - $Res call({String greeting, List sections}); -} - -/// @nodoc -class _$SpotifyHomeFeedCopyWithImpl<$Res, $Val extends SpotifyHomeFeed> - implements $SpotifyHomeFeedCopyWith<$Res> { - _$SpotifyHomeFeedCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SpotifyHomeFeed - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? greeting = null, - Object? sections = null, - }) { - return _then(_value.copyWith( - greeting: null == greeting - ? _value.greeting - : greeting // ignore: cast_nullable_to_non_nullable - as String, - sections: null == sections - ? _value.sections - : sections // ignore: cast_nullable_to_non_nullable - as List, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SpotifyHomeFeedImplCopyWith<$Res> - implements $SpotifyHomeFeedCopyWith<$Res> { - factory _$$SpotifyHomeFeedImplCopyWith(_$SpotifyHomeFeedImpl value, - $Res Function(_$SpotifyHomeFeedImpl) then) = - __$$SpotifyHomeFeedImplCopyWithImpl<$Res>; - @override - @useResult - $Res call({String greeting, List sections}); -} - -/// @nodoc -class __$$SpotifyHomeFeedImplCopyWithImpl<$Res> - extends _$SpotifyHomeFeedCopyWithImpl<$Res, _$SpotifyHomeFeedImpl> - implements _$$SpotifyHomeFeedImplCopyWith<$Res> { - __$$SpotifyHomeFeedImplCopyWithImpl( - _$SpotifyHomeFeedImpl _value, $Res Function(_$SpotifyHomeFeedImpl) _then) - : super(_value, _then); - - /// Create a copy of SpotifyHomeFeed - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? greeting = null, - Object? sections = null, - }) { - return _then(_$SpotifyHomeFeedImpl( - greeting: null == greeting - ? _value.greeting - : greeting // ignore: cast_nullable_to_non_nullable - as String, - sections: null == sections - ? _value._sections - : sections // ignore: cast_nullable_to_non_nullable - as List, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SpotifyHomeFeedImpl implements _SpotifyHomeFeed { - _$SpotifyHomeFeedImpl( - {required this.greeting, - required final List sections}) - : _sections = sections; - - factory _$SpotifyHomeFeedImpl.fromJson(Map json) => - _$$SpotifyHomeFeedImplFromJson(json); - - @override - final String greeting; - final List _sections; - @override - List get sections { - if (_sections is EqualUnmodifiableListView) return _sections; - // ignore: implicit_dynamic_type - return EqualUnmodifiableListView(_sections); - } - - @override - String toString() { - return 'SpotifyHomeFeed(greeting: $greeting, sections: $sections)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SpotifyHomeFeedImpl && - (identical(other.greeting, greeting) || - other.greeting == greeting) && - const DeepCollectionEquality().equals(other._sections, _sections)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash( - runtimeType, greeting, const DeepCollectionEquality().hash(_sections)); - - /// Create a copy of SpotifyHomeFeed - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SpotifyHomeFeedImplCopyWith<_$SpotifyHomeFeedImpl> get copyWith => - __$$SpotifyHomeFeedImplCopyWithImpl<_$SpotifyHomeFeedImpl>( - this, _$identity); - - @override - Map toJson() { - return _$$SpotifyHomeFeedImplToJson( - this, - ); - } -} - -abstract class _SpotifyHomeFeed implements SpotifyHomeFeed { - factory _SpotifyHomeFeed( - {required final String greeting, - required final List sections}) = - _$SpotifyHomeFeedImpl; - - factory _SpotifyHomeFeed.fromJson(Map json) = - _$SpotifyHomeFeedImpl.fromJson; - - @override - String get greeting; - @override - List get sections; - - /// Create a copy of SpotifyHomeFeed - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SpotifyHomeFeedImplCopyWith<_$SpotifyHomeFeedImpl> get copyWith => - throw _privateConstructorUsedError; -} diff --git a/lib/models/spotify/home_feed.g.dart b/lib/models/spotify/home_feed.g.dart deleted file mode 100644 index fceb3db4..00000000 --- a/lib/models/spotify/home_feed.g.dart +++ /dev/null @@ -1,165 +0,0 @@ -// GENERATED CODE - DO NOT MODIFY BY HAND - -part of 'home_feed.dart'; - -// ************************************************************************** -// JsonSerializableGenerator -// ************************************************************************** - -_$SpotifySectionPlaylistImpl _$$SpotifySectionPlaylistImplFromJson(Map json) => - _$SpotifySectionPlaylistImpl( - description: json['description'] as String, - format: json['format'] as String, - images: (json['images'] as List) - .map((e) => SpotifySectionItemImage.fromJson( - Map.from(e as Map))) - .toList(), - name: json['name'] as String, - owner: json['owner'] as String, - uri: json['uri'] as String, - ); - -Map _$$SpotifySectionPlaylistImplToJson( - _$SpotifySectionPlaylistImpl instance) => - { - 'description': instance.description, - 'format': instance.format, - 'images': instance.images.map((e) => e.toJson()).toList(), - 'name': instance.name, - 'owner': instance.owner, - 'uri': instance.uri, - }; - -_$SpotifySectionArtistImpl _$$SpotifySectionArtistImplFromJson(Map json) => - _$SpotifySectionArtistImpl( - name: json['name'] as String, - uri: json['uri'] as String, - images: (json['images'] as List) - .map((e) => SpotifySectionItemImage.fromJson( - Map.from(e as Map))) - .toList(), - ); - -Map _$$SpotifySectionArtistImplToJson( - _$SpotifySectionArtistImpl instance) => - { - 'name': instance.name, - 'uri': instance.uri, - 'images': instance.images.map((e) => e.toJson()).toList(), - }; - -_$SpotifySectionAlbumImpl _$$SpotifySectionAlbumImplFromJson(Map json) => - _$SpotifySectionAlbumImpl( - artists: (json['artists'] as List) - .map((e) => SpotifySectionAlbumArtist.fromJson( - Map.from(e as Map))) - .toList(), - images: (json['images'] as List) - .map((e) => SpotifySectionItemImage.fromJson( - Map.from(e as Map))) - .toList(), - name: json['name'] as String, - uri: json['uri'] as String, - ); - -Map _$$SpotifySectionAlbumImplToJson( - _$SpotifySectionAlbumImpl instance) => - { - 'artists': instance.artists.map((e) => e.toJson()).toList(), - 'images': instance.images.map((e) => e.toJson()).toList(), - 'name': instance.name, - 'uri': instance.uri, - }; - -_$SpotifySectionAlbumArtistImpl _$$SpotifySectionAlbumArtistImplFromJson( - Map json) => - _$SpotifySectionAlbumArtistImpl( - name: json['name'] as String, - uri: json['uri'] as String, - ); - -Map _$$SpotifySectionAlbumArtistImplToJson( - _$SpotifySectionAlbumArtistImpl instance) => - { - 'name': instance.name, - 'uri': instance.uri, - }; - -_$SpotifySectionItemImageImpl _$$SpotifySectionItemImageImplFromJson( - Map json) => - _$SpotifySectionItemImageImpl( - height: json['height'] as num?, - url: json['url'] as String, - width: json['width'] as num?, - ); - -Map _$$SpotifySectionItemImageImplToJson( - _$SpotifySectionItemImageImpl instance) => - { - 'height': instance.height, - 'url': instance.url, - 'width': instance.width, - }; - -_$SpotifyHomeFeedSectionItemImpl _$$SpotifyHomeFeedSectionItemImplFromJson( - Map json) => - _$SpotifyHomeFeedSectionItemImpl( - typename: json['typename'] as String, - playlist: json['playlist'] == null - ? null - : SpotifySectionPlaylist.fromJson( - Map.from(json['playlist'] as Map)), - artist: json['artist'] == null - ? null - : SpotifySectionArtist.fromJson( - Map.from(json['artist'] as Map)), - album: json['album'] == null - ? null - : SpotifySectionAlbum.fromJson( - Map.from(json['album'] as Map)), - ); - -Map _$$SpotifyHomeFeedSectionItemImplToJson( - _$SpotifyHomeFeedSectionItemImpl instance) => - { - 'typename': instance.typename, - 'playlist': instance.playlist?.toJson(), - 'artist': instance.artist?.toJson(), - 'album': instance.album?.toJson(), - }; - -_$SpotifyHomeFeedSectionImpl _$$SpotifyHomeFeedSectionImplFromJson(Map json) => - _$SpotifyHomeFeedSectionImpl( - typename: json['typename'] as String, - title: json['title'] as String?, - uri: json['uri'] as String, - items: (json['items'] as List) - .map((e) => SpotifyHomeFeedSectionItem.fromJson( - Map.from(e as Map))) - .toList(), - ); - -Map _$$SpotifyHomeFeedSectionImplToJson( - _$SpotifyHomeFeedSectionImpl instance) => - { - 'typename': instance.typename, - 'title': instance.title, - 'uri': instance.uri, - 'items': instance.items.map((e) => e.toJson()).toList(), - }; - -_$SpotifyHomeFeedImpl _$$SpotifyHomeFeedImplFromJson(Map json) => - _$SpotifyHomeFeedImpl( - greeting: json['greeting'] as String, - sections: (json['sections'] as List) - .map((e) => SpotifyHomeFeedSection.fromJson( - Map.from(e as Map))) - .toList(), - ); - -Map _$$SpotifyHomeFeedImplToJson( - _$SpotifyHomeFeedImpl instance) => - { - 'greeting': instance.greeting, - 'sections': instance.sections.map((e) => e.toJson()).toList(), - }; diff --git a/lib/models/spotify/recommendation_seeds.dart b/lib/models/spotify/recommendation_seeds.dart deleted file mode 100644 index 0d874ad6..00000000 --- a/lib/models/spotify/recommendation_seeds.dart +++ /dev/null @@ -1,40 +0,0 @@ -import 'package:freezed_annotation/freezed_annotation.dart'; - -part 'recommendation_seeds.freezed.dart'; -part 'recommendation_seeds.g.dart'; - -@freezed -class GeneratePlaylistProviderInput with _$GeneratePlaylistProviderInput { - factory GeneratePlaylistProviderInput({ - Iterable? seedArtists, - Iterable? seedGenres, - Iterable? seedTracks, - required int limit, - RecommendationSeeds? max, - RecommendationSeeds? min, - RecommendationSeeds? target, - }) = _GeneratePlaylistProviderInput; -} - -@freezed -class RecommendationSeeds with _$RecommendationSeeds { - factory RecommendationSeeds({ - num? acousticness, - num? danceability, - @JsonKey(name: "duration_ms") num? durationMs, - num? energy, - num? instrumentalness, - num? key, - num? liveness, - num? loudness, - num? mode, - num? popularity, - num? speechiness, - num? tempo, - @JsonKey(name: "time_signature") num? timeSignature, - num? valence, - }) = _RecommendationSeeds; - - factory RecommendationSeeds.fromJson(Map json) => - _$RecommendationSeedsFromJson(json); -} diff --git a/lib/models/spotify/recommendation_seeds.freezed.dart b/lib/models/spotify/recommendation_seeds.freezed.dart deleted file mode 100644 index c55a4134..00000000 --- a/lib/models/spotify/recommendation_seeds.freezed.dart +++ /dev/null @@ -1,786 +0,0 @@ -// coverage:ignore-file -// GENERATED CODE - DO NOT MODIFY BY HAND -// ignore_for_file: type=lint -// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark - -part of 'recommendation_seeds.dart'; - -// ************************************************************************** -// FreezedGenerator -// ************************************************************************** - -T _$identity(T value) => value; - -final _privateConstructorUsedError = UnsupportedError( - 'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); - -/// @nodoc -mixin _$GeneratePlaylistProviderInput { - Iterable? get seedArtists => throw _privateConstructorUsedError; - Iterable? get seedGenres => throw _privateConstructorUsedError; - Iterable? get seedTracks => throw _privateConstructorUsedError; - int get limit => throw _privateConstructorUsedError; - RecommendationSeeds? get max => throw _privateConstructorUsedError; - RecommendationSeeds? get min => throw _privateConstructorUsedError; - RecommendationSeeds? get target => throw _privateConstructorUsedError; - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $GeneratePlaylistProviderInputCopyWith - get copyWith => throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $GeneratePlaylistProviderInputCopyWith<$Res> { - factory $GeneratePlaylistProviderInputCopyWith( - GeneratePlaylistProviderInput value, - $Res Function(GeneratePlaylistProviderInput) then) = - _$GeneratePlaylistProviderInputCopyWithImpl<$Res, - GeneratePlaylistProviderInput>; - @useResult - $Res call( - {Iterable? seedArtists, - Iterable? seedGenres, - Iterable? seedTracks, - int limit, - RecommendationSeeds? max, - RecommendationSeeds? min, - RecommendationSeeds? target}); - - $RecommendationSeedsCopyWith<$Res>? get max; - $RecommendationSeedsCopyWith<$Res>? get min; - $RecommendationSeedsCopyWith<$Res>? get target; -} - -/// @nodoc -class _$GeneratePlaylistProviderInputCopyWithImpl<$Res, - $Val extends GeneratePlaylistProviderInput> - implements $GeneratePlaylistProviderInputCopyWith<$Res> { - _$GeneratePlaylistProviderInputCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? seedArtists = freezed, - Object? seedGenres = freezed, - Object? seedTracks = freezed, - Object? limit = null, - Object? max = freezed, - Object? min = freezed, - Object? target = freezed, - }) { - return _then(_value.copyWith( - seedArtists: freezed == seedArtists - ? _value.seedArtists - : seedArtists // ignore: cast_nullable_to_non_nullable - as Iterable?, - seedGenres: freezed == seedGenres - ? _value.seedGenres - : seedGenres // ignore: cast_nullable_to_non_nullable - as Iterable?, - seedTracks: freezed == seedTracks - ? _value.seedTracks - : seedTracks // ignore: cast_nullable_to_non_nullable - as Iterable?, - limit: null == limit - ? _value.limit - : limit // ignore: cast_nullable_to_non_nullable - as int, - max: freezed == max - ? _value.max - : max // ignore: cast_nullable_to_non_nullable - as RecommendationSeeds?, - min: freezed == min - ? _value.min - : min // ignore: cast_nullable_to_non_nullable - as RecommendationSeeds?, - target: freezed == target - ? _value.target - : target // ignore: cast_nullable_to_non_nullable - as RecommendationSeeds?, - ) as $Val); - } - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @override - @pragma('vm:prefer-inline') - $RecommendationSeedsCopyWith<$Res>? get max { - if (_value.max == null) { - return null; - } - - return $RecommendationSeedsCopyWith<$Res>(_value.max!, (value) { - return _then(_value.copyWith(max: value) as $Val); - }); - } - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @override - @pragma('vm:prefer-inline') - $RecommendationSeedsCopyWith<$Res>? get min { - if (_value.min == null) { - return null; - } - - return $RecommendationSeedsCopyWith<$Res>(_value.min!, (value) { - return _then(_value.copyWith(min: value) as $Val); - }); - } - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @override - @pragma('vm:prefer-inline') - $RecommendationSeedsCopyWith<$Res>? get target { - if (_value.target == null) { - return null; - } - - return $RecommendationSeedsCopyWith<$Res>(_value.target!, (value) { - return _then(_value.copyWith(target: value) as $Val); - }); - } -} - -/// @nodoc -abstract class _$$GeneratePlaylistProviderInputImplCopyWith<$Res> - implements $GeneratePlaylistProviderInputCopyWith<$Res> { - factory _$$GeneratePlaylistProviderInputImplCopyWith( - _$GeneratePlaylistProviderInputImpl value, - $Res Function(_$GeneratePlaylistProviderInputImpl) then) = - __$$GeneratePlaylistProviderInputImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {Iterable? seedArtists, - Iterable? seedGenres, - Iterable? seedTracks, - int limit, - RecommendationSeeds? max, - RecommendationSeeds? min, - RecommendationSeeds? target}); - - @override - $RecommendationSeedsCopyWith<$Res>? get max; - @override - $RecommendationSeedsCopyWith<$Res>? get min; - @override - $RecommendationSeedsCopyWith<$Res>? get target; -} - -/// @nodoc -class __$$GeneratePlaylistProviderInputImplCopyWithImpl<$Res> - extends _$GeneratePlaylistProviderInputCopyWithImpl<$Res, - _$GeneratePlaylistProviderInputImpl> - implements _$$GeneratePlaylistProviderInputImplCopyWith<$Res> { - __$$GeneratePlaylistProviderInputImplCopyWithImpl( - _$GeneratePlaylistProviderInputImpl _value, - $Res Function(_$GeneratePlaylistProviderInputImpl) _then) - : super(_value, _then); - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? seedArtists = freezed, - Object? seedGenres = freezed, - Object? seedTracks = freezed, - Object? limit = null, - Object? max = freezed, - Object? min = freezed, - Object? target = freezed, - }) { - return _then(_$GeneratePlaylistProviderInputImpl( - seedArtists: freezed == seedArtists - ? _value.seedArtists - : seedArtists // ignore: cast_nullable_to_non_nullable - as Iterable?, - seedGenres: freezed == seedGenres - ? _value.seedGenres - : seedGenres // ignore: cast_nullable_to_non_nullable - as Iterable?, - seedTracks: freezed == seedTracks - ? _value.seedTracks - : seedTracks // ignore: cast_nullable_to_non_nullable - as Iterable?, - limit: null == limit - ? _value.limit - : limit // ignore: cast_nullable_to_non_nullable - as int, - max: freezed == max - ? _value.max - : max // ignore: cast_nullable_to_non_nullable - as RecommendationSeeds?, - min: freezed == min - ? _value.min - : min // ignore: cast_nullable_to_non_nullable - as RecommendationSeeds?, - target: freezed == target - ? _value.target - : target // ignore: cast_nullable_to_non_nullable - as RecommendationSeeds?, - )); - } -} - -/// @nodoc - -class _$GeneratePlaylistProviderInputImpl - implements _GeneratePlaylistProviderInput { - _$GeneratePlaylistProviderInputImpl( - {this.seedArtists, - this.seedGenres, - this.seedTracks, - required this.limit, - this.max, - this.min, - this.target}); - - @override - final Iterable? seedArtists; - @override - final Iterable? seedGenres; - @override - final Iterable? seedTracks; - @override - final int limit; - @override - final RecommendationSeeds? max; - @override - final RecommendationSeeds? min; - @override - final RecommendationSeeds? target; - - @override - String toString() { - return 'GeneratePlaylistProviderInput(seedArtists: $seedArtists, seedGenres: $seedGenres, seedTracks: $seedTracks, limit: $limit, max: $max, min: $min, target: $target)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$GeneratePlaylistProviderInputImpl && - const DeepCollectionEquality() - .equals(other.seedArtists, seedArtists) && - const DeepCollectionEquality() - .equals(other.seedGenres, seedGenres) && - const DeepCollectionEquality() - .equals(other.seedTracks, seedTracks) && - (identical(other.limit, limit) || other.limit == limit) && - (identical(other.max, max) || other.max == max) && - (identical(other.min, min) || other.min == min) && - (identical(other.target, target) || other.target == target)); - } - - @override - int get hashCode => Object.hash( - runtimeType, - const DeepCollectionEquality().hash(seedArtists), - const DeepCollectionEquality().hash(seedGenres), - const DeepCollectionEquality().hash(seedTracks), - limit, - max, - min, - target); - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$GeneratePlaylistProviderInputImplCopyWith< - _$GeneratePlaylistProviderInputImpl> - get copyWith => __$$GeneratePlaylistProviderInputImplCopyWithImpl< - _$GeneratePlaylistProviderInputImpl>(this, _$identity); -} - -abstract class _GeneratePlaylistProviderInput - implements GeneratePlaylistProviderInput { - factory _GeneratePlaylistProviderInput( - {final Iterable? seedArtists, - final Iterable? seedGenres, - final Iterable? seedTracks, - required final int limit, - final RecommendationSeeds? max, - final RecommendationSeeds? min, - final RecommendationSeeds? target}) = _$GeneratePlaylistProviderInputImpl; - - @override - Iterable? get seedArtists; - @override - Iterable? get seedGenres; - @override - Iterable? get seedTracks; - @override - int get limit; - @override - RecommendationSeeds? get max; - @override - RecommendationSeeds? get min; - @override - RecommendationSeeds? get target; - - /// Create a copy of GeneratePlaylistProviderInput - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$GeneratePlaylistProviderInputImplCopyWith< - _$GeneratePlaylistProviderInputImpl> - get copyWith => throw _privateConstructorUsedError; -} - -RecommendationSeeds _$RecommendationSeedsFromJson(Map json) { - return _RecommendationSeeds.fromJson(json); -} - -/// @nodoc -mixin _$RecommendationSeeds { - num? get acousticness => throw _privateConstructorUsedError; - num? get danceability => throw _privateConstructorUsedError; - @JsonKey(name: "duration_ms") - num? get durationMs => throw _privateConstructorUsedError; - num? get energy => throw _privateConstructorUsedError; - num? get instrumentalness => throw _privateConstructorUsedError; - num? get key => throw _privateConstructorUsedError; - num? get liveness => throw _privateConstructorUsedError; - num? get loudness => throw _privateConstructorUsedError; - num? get mode => throw _privateConstructorUsedError; - num? get popularity => throw _privateConstructorUsedError; - num? get speechiness => throw _privateConstructorUsedError; - num? get tempo => throw _privateConstructorUsedError; - @JsonKey(name: "time_signature") - num? get timeSignature => throw _privateConstructorUsedError; - num? get valence => throw _privateConstructorUsedError; - - /// Serializes this RecommendationSeeds to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of RecommendationSeeds - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $RecommendationSeedsCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $RecommendationSeedsCopyWith<$Res> { - factory $RecommendationSeedsCopyWith( - RecommendationSeeds value, $Res Function(RecommendationSeeds) then) = - _$RecommendationSeedsCopyWithImpl<$Res, RecommendationSeeds>; - @useResult - $Res call( - {num? acousticness, - num? danceability, - @JsonKey(name: "duration_ms") num? durationMs, - num? energy, - num? instrumentalness, - num? key, - num? liveness, - num? loudness, - num? mode, - num? popularity, - num? speechiness, - num? tempo, - @JsonKey(name: "time_signature") num? timeSignature, - num? valence}); -} - -/// @nodoc -class _$RecommendationSeedsCopyWithImpl<$Res, $Val extends RecommendationSeeds> - implements $RecommendationSeedsCopyWith<$Res> { - _$RecommendationSeedsCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of RecommendationSeeds - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? acousticness = freezed, - Object? danceability = freezed, - Object? durationMs = freezed, - Object? energy = freezed, - Object? instrumentalness = freezed, - Object? key = freezed, - Object? liveness = freezed, - Object? loudness = freezed, - Object? mode = freezed, - Object? popularity = freezed, - Object? speechiness = freezed, - Object? tempo = freezed, - Object? timeSignature = freezed, - Object? valence = freezed, - }) { - return _then(_value.copyWith( - acousticness: freezed == acousticness - ? _value.acousticness - : acousticness // ignore: cast_nullable_to_non_nullable - as num?, - danceability: freezed == danceability - ? _value.danceability - : danceability // ignore: cast_nullable_to_non_nullable - as num?, - durationMs: freezed == durationMs - ? _value.durationMs - : durationMs // ignore: cast_nullable_to_non_nullable - as num?, - energy: freezed == energy - ? _value.energy - : energy // ignore: cast_nullable_to_non_nullable - as num?, - instrumentalness: freezed == instrumentalness - ? _value.instrumentalness - : instrumentalness // ignore: cast_nullable_to_non_nullable - as num?, - key: freezed == key - ? _value.key - : key // ignore: cast_nullable_to_non_nullable - as num?, - liveness: freezed == liveness - ? _value.liveness - : liveness // ignore: cast_nullable_to_non_nullable - as num?, - loudness: freezed == loudness - ? _value.loudness - : loudness // ignore: cast_nullable_to_non_nullable - as num?, - mode: freezed == mode - ? _value.mode - : mode // ignore: cast_nullable_to_non_nullable - as num?, - popularity: freezed == popularity - ? _value.popularity - : popularity // ignore: cast_nullable_to_non_nullable - as num?, - speechiness: freezed == speechiness - ? _value.speechiness - : speechiness // ignore: cast_nullable_to_non_nullable - as num?, - tempo: freezed == tempo - ? _value.tempo - : tempo // ignore: cast_nullable_to_non_nullable - as num?, - timeSignature: freezed == timeSignature - ? _value.timeSignature - : timeSignature // ignore: cast_nullable_to_non_nullable - as num?, - valence: freezed == valence - ? _value.valence - : valence // ignore: cast_nullable_to_non_nullable - as num?, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$RecommendationSeedsImplCopyWith<$Res> - implements $RecommendationSeedsCopyWith<$Res> { - factory _$$RecommendationSeedsImplCopyWith(_$RecommendationSeedsImpl value, - $Res Function(_$RecommendationSeedsImpl) then) = - __$$RecommendationSeedsImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {num? acousticness, - num? danceability, - @JsonKey(name: "duration_ms") num? durationMs, - num? energy, - num? instrumentalness, - num? key, - num? liveness, - num? loudness, - num? mode, - num? popularity, - num? speechiness, - num? tempo, - @JsonKey(name: "time_signature") num? timeSignature, - num? valence}); -} - -/// @nodoc -class __$$RecommendationSeedsImplCopyWithImpl<$Res> - extends _$RecommendationSeedsCopyWithImpl<$Res, _$RecommendationSeedsImpl> - implements _$$RecommendationSeedsImplCopyWith<$Res> { - __$$RecommendationSeedsImplCopyWithImpl(_$RecommendationSeedsImpl _value, - $Res Function(_$RecommendationSeedsImpl) _then) - : super(_value, _then); - - /// Create a copy of RecommendationSeeds - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? acousticness = freezed, - Object? danceability = freezed, - Object? durationMs = freezed, - Object? energy = freezed, - Object? instrumentalness = freezed, - Object? key = freezed, - Object? liveness = freezed, - Object? loudness = freezed, - Object? mode = freezed, - Object? popularity = freezed, - Object? speechiness = freezed, - Object? tempo = freezed, - Object? timeSignature = freezed, - Object? valence = freezed, - }) { - return _then(_$RecommendationSeedsImpl( - acousticness: freezed == acousticness - ? _value.acousticness - : acousticness // ignore: cast_nullable_to_non_nullable - as num?, - danceability: freezed == danceability - ? _value.danceability - : danceability // ignore: cast_nullable_to_non_nullable - as num?, - durationMs: freezed == durationMs - ? _value.durationMs - : durationMs // ignore: cast_nullable_to_non_nullable - as num?, - energy: freezed == energy - ? _value.energy - : energy // ignore: cast_nullable_to_non_nullable - as num?, - instrumentalness: freezed == instrumentalness - ? _value.instrumentalness - : instrumentalness // ignore: cast_nullable_to_non_nullable - as num?, - key: freezed == key - ? _value.key - : key // ignore: cast_nullable_to_non_nullable - as num?, - liveness: freezed == liveness - ? _value.liveness - : liveness // ignore: cast_nullable_to_non_nullable - as num?, - loudness: freezed == loudness - ? _value.loudness - : loudness // ignore: cast_nullable_to_non_nullable - as num?, - mode: freezed == mode - ? _value.mode - : mode // ignore: cast_nullable_to_non_nullable - as num?, - popularity: freezed == popularity - ? _value.popularity - : popularity // ignore: cast_nullable_to_non_nullable - as num?, - speechiness: freezed == speechiness - ? _value.speechiness - : speechiness // ignore: cast_nullable_to_non_nullable - as num?, - tempo: freezed == tempo - ? _value.tempo - : tempo // ignore: cast_nullable_to_non_nullable - as num?, - timeSignature: freezed == timeSignature - ? _value.timeSignature - : timeSignature // ignore: cast_nullable_to_non_nullable - as num?, - valence: freezed == valence - ? _value.valence - : valence // ignore: cast_nullable_to_non_nullable - as num?, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$RecommendationSeedsImpl implements _RecommendationSeeds { - _$RecommendationSeedsImpl( - {this.acousticness, - this.danceability, - @JsonKey(name: "duration_ms") this.durationMs, - this.energy, - this.instrumentalness, - this.key, - this.liveness, - this.loudness, - this.mode, - this.popularity, - this.speechiness, - this.tempo, - @JsonKey(name: "time_signature") this.timeSignature, - this.valence}); - - factory _$RecommendationSeedsImpl.fromJson(Map json) => - _$$RecommendationSeedsImplFromJson(json); - - @override - final num? acousticness; - @override - final num? danceability; - @override - @JsonKey(name: "duration_ms") - final num? durationMs; - @override - final num? energy; - @override - final num? instrumentalness; - @override - final num? key; - @override - final num? liveness; - @override - final num? loudness; - @override - final num? mode; - @override - final num? popularity; - @override - final num? speechiness; - @override - final num? tempo; - @override - @JsonKey(name: "time_signature") - final num? timeSignature; - @override - final num? valence; - - @override - String toString() { - return 'RecommendationSeeds(acousticness: $acousticness, danceability: $danceability, durationMs: $durationMs, energy: $energy, instrumentalness: $instrumentalness, key: $key, liveness: $liveness, loudness: $loudness, mode: $mode, popularity: $popularity, speechiness: $speechiness, tempo: $tempo, timeSignature: $timeSignature, valence: $valence)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$RecommendationSeedsImpl && - (identical(other.acousticness, acousticness) || - other.acousticness == acousticness) && - (identical(other.danceability, danceability) || - other.danceability == danceability) && - (identical(other.durationMs, durationMs) || - other.durationMs == durationMs) && - (identical(other.energy, energy) || other.energy == energy) && - (identical(other.instrumentalness, instrumentalness) || - other.instrumentalness == instrumentalness) && - (identical(other.key, key) || other.key == key) && - (identical(other.liveness, liveness) || - other.liveness == liveness) && - (identical(other.loudness, loudness) || - other.loudness == loudness) && - (identical(other.mode, mode) || other.mode == mode) && - (identical(other.popularity, popularity) || - other.popularity == popularity) && - (identical(other.speechiness, speechiness) || - other.speechiness == speechiness) && - (identical(other.tempo, tempo) || other.tempo == tempo) && - (identical(other.timeSignature, timeSignature) || - other.timeSignature == timeSignature) && - (identical(other.valence, valence) || other.valence == valence)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash( - runtimeType, - acousticness, - danceability, - durationMs, - energy, - instrumentalness, - key, - liveness, - loudness, - mode, - popularity, - speechiness, - tempo, - timeSignature, - valence); - - /// Create a copy of RecommendationSeeds - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$RecommendationSeedsImplCopyWith<_$RecommendationSeedsImpl> get copyWith => - __$$RecommendationSeedsImplCopyWithImpl<_$RecommendationSeedsImpl>( - this, _$identity); - - @override - Map toJson() { - return _$$RecommendationSeedsImplToJson( - this, - ); - } -} - -abstract class _RecommendationSeeds implements RecommendationSeeds { - factory _RecommendationSeeds( - {final num? acousticness, - final num? danceability, - @JsonKey(name: "duration_ms") final num? durationMs, - final num? energy, - final num? instrumentalness, - final num? key, - final num? liveness, - final num? loudness, - final num? mode, - final num? popularity, - final num? speechiness, - final num? tempo, - @JsonKey(name: "time_signature") final num? timeSignature, - final num? valence}) = _$RecommendationSeedsImpl; - - factory _RecommendationSeeds.fromJson(Map json) = - _$RecommendationSeedsImpl.fromJson; - - @override - num? get acousticness; - @override - num? get danceability; - @override - @JsonKey(name: "duration_ms") - num? get durationMs; - @override - num? get energy; - @override - num? get instrumentalness; - @override - num? get key; - @override - num? get liveness; - @override - num? get loudness; - @override - num? get mode; - @override - num? get popularity; - @override - num? get speechiness; - @override - num? get tempo; - @override - @JsonKey(name: "time_signature") - num? get timeSignature; - @override - num? get valence; - - /// Create a copy of RecommendationSeeds - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$RecommendationSeedsImplCopyWith<_$RecommendationSeedsImpl> get copyWith => - throw _privateConstructorUsedError; -} diff --git a/lib/models/spotify/recommendation_seeds.g.dart b/lib/models/spotify/recommendation_seeds.g.dart deleted file mode 100644 index accb2ed1..00000000 --- a/lib/models/spotify/recommendation_seeds.g.dart +++ /dev/null @@ -1,44 +0,0 @@ -// GENERATED CODE - DO NOT MODIFY BY HAND - -part of 'recommendation_seeds.dart'; - -// ************************************************************************** -// JsonSerializableGenerator -// ************************************************************************** - -_$RecommendationSeedsImpl _$$RecommendationSeedsImplFromJson(Map json) => - _$RecommendationSeedsImpl( - acousticness: json['acousticness'] as num?, - danceability: json['danceability'] as num?, - durationMs: json['duration_ms'] as num?, - energy: json['energy'] as num?, - instrumentalness: json['instrumentalness'] as num?, - key: json['key'] as num?, - liveness: json['liveness'] as num?, - loudness: json['loudness'] as num?, - mode: json['mode'] as num?, - popularity: json['popularity'] as num?, - speechiness: json['speechiness'] as num?, - tempo: json['tempo'] as num?, - timeSignature: json['time_signature'] as num?, - valence: json['valence'] as num?, - ); - -Map _$$RecommendationSeedsImplToJson( - _$RecommendationSeedsImpl instance) => - { - 'acousticness': instance.acousticness, - 'danceability': instance.danceability, - 'duration_ms': instance.durationMs, - 'energy': instance.energy, - 'instrumentalness': instance.instrumentalness, - 'key': instance.key, - 'liveness': instance.liveness, - 'loudness': instance.loudness, - 'mode': instance.mode, - 'popularity': instance.popularity, - 'speechiness': instance.speechiness, - 'tempo': instance.tempo, - 'time_signature': instance.timeSignature, - 'valence': instance.valence, - }; diff --git a/lib/models/spotify_friends.dart b/lib/models/spotify_friends.dart deleted file mode 100644 index b386fb81..00000000 --- a/lib/models/spotify_friends.dart +++ /dev/null @@ -1,111 +0,0 @@ -import 'package:json_annotation/json_annotation.dart'; - -part 'spotify_friends.g.dart'; - -@JsonSerializable(createToJson: false) -class SpotifyFriend { - final String uri; - final String name; - final String imageUrl; - - const SpotifyFriend({ - required this.uri, - required this.name, - required this.imageUrl, - }); - - factory SpotifyFriend.fromJson(Map json) => - _$SpotifyFriendFromJson(json); - - String get id => uri.split(":").last; -} - -@JsonSerializable(createToJson: false) -class SpotifyActivityArtist { - final String uri; - final String name; - - const SpotifyActivityArtist({required this.uri, required this.name}); - - factory SpotifyActivityArtist.fromJson(Map json) => - _$SpotifyActivityArtistFromJson(json); - - String get id => uri.split(":").last; -} - -@JsonSerializable(createToJson: false) -class SpotifyActivityAlbum { - final String uri; - final String name; - - const SpotifyActivityAlbum({required this.uri, required this.name}); - - factory SpotifyActivityAlbum.fromJson(Map json) => - _$SpotifyActivityAlbumFromJson(json); - - String get id => uri.split(":").last; -} - -@JsonSerializable(createToJson: false) -class SpotifyActivityContext { - final String uri; - final String name; - final num index; - - const SpotifyActivityContext({ - required this.uri, - required this.name, - required this.index, - }); - - factory SpotifyActivityContext.fromJson(Map json) => - _$SpotifyActivityContextFromJson(json); - - String get id => uri.split(":").last; - String get path => uri.split(":").skip(1).join("/"); -} - -@JsonSerializable(createToJson: false) -class SpotifyActivityTrack { - final String uri; - final String name; - final String imageUrl; - final SpotifyActivityArtist artist; - final SpotifyActivityAlbum album; - final SpotifyActivityContext context; - - const SpotifyActivityTrack({ - required this.uri, - required this.name, - required this.imageUrl, - required this.artist, - required this.album, - required this.context, - }); - - factory SpotifyActivityTrack.fromJson(Map json) => - _$SpotifyActivityTrackFromJson(json); - - String get id => uri.split(":").last; -} - -@JsonSerializable(createToJson: false) -class SpotifyFriendActivity { - SpotifyFriend user; - SpotifyActivityTrack track; - - SpotifyFriendActivity({required this.user, required this.track}); - - factory SpotifyFriendActivity.fromJson(Map json) => - _$SpotifyFriendActivityFromJson(json); -} - -@JsonSerializable(createToJson: false) -class SpotifyFriends { - List friends; - - SpotifyFriends({required this.friends}); - - factory SpotifyFriends.fromJson(Map json) => - _$SpotifyFriendsFromJson(json); -} diff --git a/lib/models/spotify_friends.g.dart b/lib/models/spotify_friends.g.dart deleted file mode 100644 index a1248429..00000000 --- a/lib/models/spotify_friends.g.dart +++ /dev/null @@ -1,60 +0,0 @@ -// GENERATED CODE - DO NOT MODIFY BY HAND - -part of 'spotify_friends.dart'; - -// ************************************************************************** -// JsonSerializableGenerator -// ************************************************************************** - -SpotifyFriend _$SpotifyFriendFromJson(Map json) => SpotifyFriend( - uri: json['uri'] as String, - name: json['name'] as String, - imageUrl: json['imageUrl'] as String, - ); - -SpotifyActivityArtist _$SpotifyActivityArtistFromJson(Map json) => - SpotifyActivityArtist( - uri: json['uri'] as String, - name: json['name'] as String, - ); - -SpotifyActivityAlbum _$SpotifyActivityAlbumFromJson(Map json) => - SpotifyActivityAlbum( - uri: json['uri'] as String, - name: json['name'] as String, - ); - -SpotifyActivityContext _$SpotifyActivityContextFromJson(Map json) => - SpotifyActivityContext( - uri: json['uri'] as String, - name: json['name'] as String, - index: json['index'] as num, - ); - -SpotifyActivityTrack _$SpotifyActivityTrackFromJson(Map json) => - SpotifyActivityTrack( - uri: json['uri'] as String, - name: json['name'] as String, - imageUrl: json['imageUrl'] as String, - artist: SpotifyActivityArtist.fromJson( - Map.from(json['artist'] as Map)), - album: SpotifyActivityAlbum.fromJson( - Map.from(json['album'] as Map)), - context: SpotifyActivityContext.fromJson( - Map.from(json['context'] as Map)), - ); - -SpotifyFriendActivity _$SpotifyFriendActivityFromJson(Map json) => - SpotifyFriendActivity( - user: SpotifyFriend.fromJson( - Map.from(json['user'] as Map)), - track: SpotifyActivityTrack.fromJson( - Map.from(json['track'] as Map)), - ); - -SpotifyFriends _$SpotifyFriendsFromJson(Map json) => SpotifyFriends( - friends: (json['friends'] as List) - .map((e) => SpotifyFriendActivity.fromJson( - Map.from(e as Map))) - .toList(), - ); diff --git a/lib/models/spotify_spotube_credentials.dart b/lib/models/spotify_spotube_credentials.dart deleted file mode 100644 index 982ca64a..00000000 --- a/lib/models/spotify_spotube_credentials.dart +++ /dev/null @@ -1,30 +0,0 @@ -class SpotifySpotubeCredentials { - String clientId; - String accessToken; - DateTime expiration; - bool isAnonymous; - - SpotifySpotubeCredentials({ - required this.clientId, - required this.accessToken, - required this.expiration, - required this.isAnonymous, - }); - - SpotifySpotubeCredentials.fromJson(Map json) - : clientId = json['clientId'], - accessToken = json['accessToken'], - expiration = DateTime.fromMillisecondsSinceEpoch( - json['accessTokenExpirationTimestampMs'], - ), - isAnonymous = json['isAnonymous']; - - Map toJson() { - return { - 'clientId': clientId, - 'accessToken': accessToken, - 'accessTokenExpirationTimestampMs': expiration.millisecondsSinceEpoch, - 'isAnonymous': isAnonymous, - }; - } -} diff --git a/lib/modules/album/album_card.dart b/lib/modules/album/album_card.dart index dd914fad..80dfd55b 100644 --- a/lib/modules/album/album_card.dart +++ b/lib/modules/album/album_card.dart @@ -1,33 +1,37 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/dialogs/select_device_dialog.dart'; -import 'package:spotube/components/playbutton_card.dart'; -import 'package:spotube/extensions/artist_simple.dart'; +import 'package:spotube/components/playbutton_view/playbutton_card.dart'; +import 'package:spotube/components/playbutton_view/playbutton_tile.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/extensions/track.dart'; import 'package:spotube/models/connect/connect.dart'; -import 'package:spotube/pages/album/album.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/querying_track_info.dart'; import 'package:spotube/provider/connect/connect.dart'; import 'package:spotube/provider/history/history.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/album.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; -import 'package:spotube/utils/service_utils.dart'; -extension FormattedAlbumType on AlbumType { +extension FormattedAlbumType on SpotubeAlbumType { String get formatted => name.replaceFirst(name[0], name[0].toUpperCase()); } class AlbumCard extends HookConsumerWidget { - final AlbumSimple album; + final SpotubeSimpleAlbumObject album; + final bool _isTile; const AlbumCard( this.album, { super.key, - }); + }) : _isTile = false; + + const AlbumCard.tile( + this.album, { + super.key, + }) : _isTile = true; @override Widget build(BuildContext context, ref) { @@ -38,105 +42,147 @@ class AlbumCard extends HookConsumerWidget { final historyNotifier = ref.read(playbackHistoryActionsProvider); final isFetchingActiveTrack = ref.watch(queryingTrackInfoProvider); - bool isPlaylistPlaying = useMemoized( - () => playlist.containsCollection(album.id!), + final isPlaylistPlaying = useMemoized( + () => playlist.containsCollection(album.id), [playlist, album.id], ); final updating = useState(false); - final scaffoldMessenger = ScaffoldMessenger.maybeOf(context); + final fetchAllTrack = useCallback(() async { + await ref.read(metadataPluginAlbumTracksProvider(album.id).future); + return ref + .read(metadataPluginAlbumTracksProvider(album.id).notifier) + .fetchAll(); + }, [album.id, ref]); - Future> fetchAllTrack() async { - if (album.tracks != null && album.tracks!.isNotEmpty) { - return album.tracks!.map((track) => track.asTrack(album)).toList(); + final imageUrl = useMemoized( + () => album.images.from200PxTo300PxOrSmallestImage( + ImagePlaceholder.collection, + ), + [album.images], + ); + + final isLoading = + (isPlaylistPlaying && isFetchingActiveTrack) || updating.value; + final description = "${album.albumType.name} • ${album.artists.asString()}"; + + final onTap = useCallback(() { + context.navigateTo(AlbumRoute(id: album.id, album: album)); + }, [context, album]); + + final onPlaybuttonPressed = useCallback(() async { + updating.value = true; + try { + if (isPlaylistPlaying) { + return playing ? audioPlayer.pause() : audioPlayer.resume(); + } + + final fetchedTracks = await fetchAllTrack(); + + if (fetchedTracks.isEmpty || !context.mounted) return; + + final isRemoteDevice = await showSelectDeviceDialog(context, ref); + if (isRemoteDevice == null) return; + if (isRemoteDevice) { + final remotePlayback = ref.read(connectProvider.notifier); + await remotePlayback.load( + WebSocketLoadEventData.album( + tracks: fetchedTracks, + collection: album, + ), + ); + } else { + await playlistNotifier.load(fetchedTracks, autoPlay: true); + playlistNotifier.addCollection(album.id); + historyNotifier.addAlbums([album]); + } + } finally { + updating.value = false; } - await ref.read(albumTracksProvider(album).future); - return ref.read(albumTracksProvider(album).notifier).fetchAll(); + }, [ + isPlaylistPlaying, + playing, + audioPlayer, + fetchAllTrack, + context, + ref, + playlistNotifier, + album, + historyNotifier, + updating + ]); + + final onAddToQueuePressed = useCallback(() async { + if (isPlaylistPlaying) { + return; + } + + updating.value = true; + try { + final fetchedTracks = await fetchAllTrack(); + + if (fetchedTracks.isEmpty) return; + playlistNotifier.addTracks(fetchedTracks); + playlistNotifier.addCollection(album.id); + historyNotifier.addAlbums([album]); + if (context.mounted) { + showToast( + context: context, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + content: Text( + context.l10n.added_to_queue(fetchedTracks.length), + ), + trailing: Button.outline( + child: Text(context.l10n.undo), + onPressed: () { + playlistNotifier + .removeTracks(fetchedTracks.map((e) => e.id)); + }, + ), + ), + ); + }, + ); + } + } finally { + updating.value = false; + } + }, [ + isPlaylistPlaying, + updating.value, + fetchAllTrack, + playlistNotifier, + album.id, + historyNotifier, + album, + context + ]); + + if (_isTile) { + return PlaybuttonTile( + imageUrl: imageUrl, + isPlaying: isPlaylistPlaying, + isLoading: isLoading, + title: album.name, + description: description, + onTap: onTap, + onPlaybuttonPressed: onPlaybuttonPressed, + onAddToQueuePressed: onAddToQueuePressed, + ); } return PlaybuttonCard( - imageUrl: album.images.asUrlString( - placeholder: ImagePlaceholder.collection, - ), - margin: const EdgeInsets.symmetric(horizontal: 10), - isPlaying: isPlaylistPlaying, - isLoading: - (isPlaylistPlaying && isFetchingActiveTrack) || updating.value, - title: album.name!, - description: - "${album.albumType?.formatted} • ${album.artists?.asString() ?? ""}", - onTap: () { - ServiceUtils.pushNamed( - context, - AlbumPage.name, - pathParameters: { - "id": album.id!, - }, - extra: album, - ); - }, - onPlaybuttonPressed: () async { - updating.value = true; - try { - if (isPlaylistPlaying) { - return playing ? audioPlayer.pause() : audioPlayer.resume(); - } - - final fetchedTracks = await fetchAllTrack(); - - if (fetchedTracks.isEmpty || !context.mounted) return; - - final isRemoteDevice = await showSelectDeviceDialog(context, ref); - if (isRemoteDevice) { - final remotePlayback = ref.read(connectProvider.notifier); - await remotePlayback.load( - WebSocketLoadEventData.album( - tracks: fetchedTracks, - collection: album, - ), - ); - } else { - await playlistNotifier.load(fetchedTracks, autoPlay: true); - playlistNotifier.addCollection(album.id!); - historyNotifier.addAlbums([album]); - } - } finally { - updating.value = false; - } - }, - onAddToQueuePressed: () async { - if (isPlaylistPlaying) { - return; - } - - updating.value = true; - try { - final fetchedTracks = await fetchAllTrack(); - - if (fetchedTracks.isEmpty) return; - playlistNotifier.addTracks(fetchedTracks); - playlistNotifier.addCollection(album.id!); - historyNotifier.addAlbums([album]); - if (context.mounted) { - final snackbar = SnackBar( - content: Text( - context.l10n.added_to_queue(fetchedTracks.length), - ), - action: SnackBarAction( - label: "Undo", - onPressed: () { - playlistNotifier - .removeTracks(fetchedTracks.map((e) => e.id!)); - }, - ), - ); - - scaffoldMessenger?.showSnackBar(snackbar); - } - } finally { - updating.value = false; - } - }); + imageUrl: imageUrl, + isPlaying: isPlaylistPlaying, + isLoading: isLoading, + title: album.name, + description: description, + onTap: onTap, + onPlaybuttonPressed: onPlaybuttonPressed, + onAddToQueuePressed: onAddToQueuePressed, + ); } } diff --git a/lib/modules/artist/artist_album_list.dart b/lib/modules/artist/artist_album_list.dart index a2dd8006..8d228905 100644 --- a/lib/modules/artist/artist_album_list.dart +++ b/lib/modules/artist/artist_album_list.dart @@ -1,9 +1,10 @@ -import 'package:flutter/material.dart' hide Page; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/artist/albums.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; class ArtistAlbumList extends HookConsumerWidget { final String artistId; @@ -15,22 +16,22 @@ class ArtistAlbumList extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final albumsQuery = ref.watch(artistAlbumsProvider(artistId)); + final albumsQuery = ref.watch(metadataPluginArtistAlbumsProvider(artistId)); final albumsQueryNotifier = - ref.watch(artistAlbumsProvider(artistId).notifier); + ref.watch(metadataPluginArtistAlbumsProvider(artistId).notifier); final albums = albumsQuery.asData?.value.items ?? []; final theme = Theme.of(context); - return HorizontalPlaybuttonCardView( + return HorizontalPlaybuttonCardView( isLoadingNextPage: albumsQuery.isLoadingNextPage, hasNextPage: albumsQuery.asData?.value.hasMore ?? false, items: albums, onFetchMore: albumsQueryNotifier.fetchMore, title: Text( context.l10n.albums, - style: theme.textTheme.headlineSmall, + style: theme.typography.h4, ), ); } diff --git a/lib/modules/artist/artist_card.dart b/lib/modules/artist/artist_card.dart index add2608d..d9e01206 100644 --- a/lib/modules/artist/artist_card.dart +++ b/lib/modules/artist/artist_card.dart @@ -1,20 +1,18 @@ +import 'package:auto_route/auto_route.dart'; import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/hooks/utils/use_breakpoint_value.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; -import 'package:spotube/pages/artist/artist.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + import 'package:spotube/provider/blacklist_provider.dart'; -import 'package:spotube/utils/service_utils.dart'; class ArtistCard extends HookConsumerWidget { - final Artist artist; + final SpotubeFullArtistObject artist; const ArtistCard(this.artist, {super.key}); @override @@ -33,98 +31,44 @@ class ArtistCard extends HookConsumerWidget { ), ); - final radius = BorderRadius.circular(15); - - final double size = useBreakpointValue( - xs: 130, - sm: 130, - md: 150, - others: 170, - ); - - return Container( - width: size, - margin: const EdgeInsets.symmetric(vertical: 5), - child: Material( - shadowColor: theme.colorScheme.surface, - color: Color.lerp( - theme.colorScheme.surfaceContainerHighest, - theme.colorScheme.surface, - useBrightnessValue(.9, .7), - ), - elevation: 3, - shape: RoundedRectangleBorder( - borderRadius: radius, - side: isBlackListed == true - ? const BorderSide( - color: Colors.red, - width: 2, - ) - : BorderSide.none, - ), - child: InkWell( - onTap: () { - ServiceUtils.pushNamed( - context, - ArtistPage.name, - pathParameters: { - "id": artist.id!, - }, - ); - }, - borderRadius: radius, - child: Padding( - padding: const EdgeInsets.all(12), - child: Column( - children: [ - Stack( - children: [ - ConstrainedBox( - constraints: BoxConstraints( - maxHeight: size, - ), - child: CircleAvatar( - backgroundImage: backgroundImage, - radius: size / 2, - ), - ), - Positioned( - right: 0, - child: Container( - padding: const EdgeInsets.symmetric( - horizontal: 10, - vertical: 5, - ), - decoration: BoxDecoration( - color: Colors.blue, - borderRadius: BorderRadius.circular(50)), - child: Skeleton.ignore( - child: Text( - context.l10n.artist, - style: const TextStyle( - color: Colors.white, - fontSize: 12, - fontWeight: FontWeight.bold, - ), - ), - ), - ), - ), - ], - ), - const SizedBox(height: 10), - AutoSizeText( - artist.name!, - maxLines: 1, - textAlign: TextAlign.center, - overflow: TextOverflow.ellipsis, - style: theme.textTheme.bodyMedium?.copyWith( - fontWeight: FontWeight.bold, - ), + return SizedBox( + width: 180, + child: Button.card( + onPressed: () { + context.navigateTo(ArtistRoute(artistId: artist.id)); + }, + child: Column( + children: [ + Avatar( + initials: artist.name.trim()[0].toUpperCase(), + provider: backgroundImage, + size: 130, + ), + const Gap(10), + AutoSizeText( + artist.name, + maxLines: 2, + textAlign: TextAlign.center, + overflow: TextOverflow.ellipsis, + style: theme.typography.bold, + ), + const Spacer(), + Row( + mainAxisAlignment: MainAxisAlignment.end, + children: [ + if (isBlackListed == true) ...[ + DestructiveBadge( + child: Text(context.l10n.blacklisted.toUpperCase()), ), + const Gap(5), ], - ), - )), + SecondaryBadge( + child: Text(context.l10n.artist.toUpperCase()), + ) + ], + ) + ], + ), ), ); } diff --git a/lib/modules/connect/connect_device.dart b/lib/modules/connect/connect_device.dart index f4888534..2c8d612b 100644 --- a/lib/modules/connect/connect_device.dart +++ b/lib/modules/connect/connect_device.dart @@ -1,11 +1,11 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/connect/connect.dart'; import 'package:spotube/provider/connect/clients.dart'; -import 'package:spotube/utils/service_utils.dart'; class ConnectDeviceButton extends HookConsumerWidget { final bool _sidebar; @@ -14,110 +14,66 @@ class ConnectDeviceButton extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:colorScheme) = Theme.of(context); - final pixelRatio = MediaQuery.of(context).devicePixelRatio; final connectClients = ref.watch(connectClientsProvider); + final hasServices = + connectClients.asData?.value.services.isNotEmpty == true; + if (_sidebar) { + final mediaQuery = MediaQuery.sizeOf(context); + + if (mediaQuery.mdAndDown) { + return IconButton.ghost( + icon: const Icon(SpotubeIcons.speaker), + onPressed: () { + context.navigateTo(const ConnectRoute()); + }, + ); + } + return SizedBox( width: double.infinity, - child: TextButton( + child: Button.primary( onPressed: () { - ServiceUtils.pushNamed(context, ConnectPage.name); + context.navigateTo(const ConnectRoute()); }, - style: FilledButton.styleFrom( - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(8), - ), - padding: const EdgeInsets.all(5), - ), - child: Row( - children: [ - Text(context.l10n.devices), - if (connectClients.asData?.value.services.isNotEmpty == true) - Text( - " (${connectClients.asData?.value.services.length})", - ), - const Spacer(), - const Icon(SpotubeIcons.speaker), - const Gap(5), - ], + trailing: const Icon(SpotubeIcons.speaker), + child: Text( + "${context.l10n.devices}" + "${hasServices ? " (${connectClients.asData?.value.services.length})" : ""}", ), ), ); } - return SizedBox( - height: 40 * pixelRatio, - child: Stack( - alignment: Alignment.centerRight, - fit: StackFit.loose, - children: [ - Material( - type: MaterialType.transparency, - child: Center( - child: ClipRect( - clipBehavior: Clip.hardEdge, - child: InkWell( - onTap: () { - ServiceUtils.pushNamed(context, ConnectPage.name); - }, - borderRadius: BorderRadius.circular(50), - child: Ink( - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(50), - color: colorScheme.primaryContainer, - ), - padding: - const EdgeInsets.symmetric(horizontal: 10, vertical: 5), - child: Row( - mainAxisSize: MainAxisSize.min, - children: [ - if (connectClients.asData?.value.resolvedService != - null) ...[ - Container( - width: 7, - height: 7, - decoration: BoxDecoration( - color: Colors.greenAccent, - borderRadius: BorderRadius.circular(50), - ), - ), - const Gap(5), - ], - Text(context.l10n.devices), - if (connectClients.asData?.value.services.isNotEmpty == - true) - Text( - " (${connectClients.asData?.value.services.length})", - style: TextStyle( - color: colorScheme.onPrimaryContainer - .withOpacity(0.5), - ), - ), - const Gap(35), - ], - ), + return Row( + children: [ + SecondaryBadge( + onPressed: () { + context.navigateTo(const ConnectRoute()); + }, + style: const ButtonStyle.secondary(size: ButtonSize(.8)), + leading: connectClients.asData?.value.resolvedService != null + ? const Center( + child: DotItem( + size: 6, + borderRadius: 10, + color: Colors.green, ), - ), - ), - ), + ) + : null, + child: Text( + "${context.l10n.devices}" + "${hasServices ? " (${connectClients.asData?.value.services.length})" : ""}", ), - Positioned( - right: -3, - child: IconButton.filled( - icon: const Icon(SpotubeIcons.speaker), - style: IconButton.styleFrom( - visualDensity: VisualDensity.standard, - foregroundColor: colorScheme.onPrimary, - ), - onPressed: () { - ServiceUtils.pushNamed(context, ConnectPage.name); - }, - ), - ), - ], - ), + ), + IconButton.primary( + icon: const Icon(SpotubeIcons.speaker), + onPressed: () { + context.navigateTo(const ConnectRoute()); + }, + ) + ], ); } } diff --git a/lib/modules/connect/local_devices.dart b/lib/modules/connect/local_devices.dart index dd7db971..dc192e44 100644 --- a/lib/modules/connect/local_devices.dart +++ b/lib/modules/connect/local_devices.dart @@ -1,7 +1,7 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; @@ -10,7 +10,7 @@ class ConnectPageLocalDevices extends HookWidget { @override Widget build(BuildContext context) { - final ThemeData(:textTheme) = Theme.of(context); + final ThemeData(:typography) = Theme.of(context); final devicesFuture = useFuture(audioPlayer.devices); final devicesStream = useStream(audioPlayer.devicesStream); final selectedDeviceFuture = useFuture(audioPlayer.selectedDevice); @@ -32,7 +32,7 @@ class ConnectPageLocalDevices extends HookWidget { sliver: SliverToBoxAdapter( child: Text( context.l10n.this_device, - style: textTheme.titleMedium, + style: typography.bold, ), ), ), @@ -43,17 +43,16 @@ class ConnectPageLocalDevices extends HookWidget { itemBuilder: (context, index) { final device = devices[index]; - return Card( - child: ListTile( - leading: const Icon(SpotubeIcons.speaker), - title: Text(device.description), - subtitle: Text(device.name), - selected: selectedDevice == device, - onTap: () => audioPlayer.setAudioDevice(device), - ), + return ButtonTile( + selected: selectedDevice == device, + onPressed: () => audioPlayer.setAudioDevice(device), + leading: const Icon(SpotubeIcons.speaker), + title: Text(device.description), + subtitle: Text(device.name), ); }, ), + const SliverGap(200) ], ); } diff --git a/lib/modules/getting_started/blur_card.dart b/lib/modules/getting_started/blur_card.dart index db887013..6434c0a3 100644 --- a/lib/modules/getting_started/blur_card.dart +++ b/lib/modules/getting_started/blur_card.dart @@ -1,7 +1,5 @@ -import 'dart:ui'; - -import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; class BlurCard extends HookConsumerWidget { final Widget child; @@ -18,8 +16,7 @@ class BlurCard extends HookConsumerWidget { clipBehavior: Clip.antiAlias, child: SizedBox( width: double.infinity, - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 20, sigmaY: 20), + child: SurfaceCard( child: Padding( padding: const EdgeInsets.all(16.0), child: child, diff --git a/lib/modules/home/sections/featured.dart b/lib/modules/home/sections/featured.dart index 4f30c342..c65ebf89 100644 --- a/lib/modules/home/sections/featured.dart +++ b/lib/modules/home/sections/featured.dart @@ -1,29 +1,47 @@ -import 'package:flutter/material.dart' hide Page; +import 'package:flutter_undraw/flutter_undraw.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +@Deprecated( + "Later a featured playlists API will be added for metadata plugins.") class HomeFeaturedSection extends HookConsumerWidget { const HomeFeaturedSection({super.key}); @override Widget build(BuildContext context, ref) { - final featuredPlaylists = ref.watch(featuredPlaylistsProvider); - final featuredPlaylistsNotifier = - ref.watch(featuredPlaylistsProvider.notifier); + return const SizedBox.shrink(); + // final featuredPlaylists = ref.watch(featuredPlaylistsProvider); + // final featuredPlaylistsNotifier = + // ref.watch(featuredPlaylistsProvider.notifier); - return Skeletonizer( - enabled: featuredPlaylists.isLoading, - child: HorizontalPlaybuttonCardView( - items: featuredPlaylists.asData?.value.items ?? [], - title: Text(context.l10n.featured), - isLoadingNextPage: featuredPlaylists.isLoadingNextPage, - hasNextPage: featuredPlaylists.asData?.value.hasMore ?? false, - onFetchMore: featuredPlaylistsNotifier.fetchMore, - ), - ); + // if (featuredPlaylists.hasError) { + // return Column( + // mainAxisSize: MainAxisSize.min, + // children: [ + // Undraw( + // illustration: UndrawIllustration.fixingBugs, + // height: 200 * context.theme.scaling, + // color: context.theme.colorScheme.primary, + // ), + // Text(context.l10n.something_went_wrong).small().muted(), + // const Gap(8), + // ], + // ); + // } + + // return Skeletonizer( + // enabled: featuredPlaylists.isLoading, + // child: HorizontalPlaybuttonCardView( + // items: featuredPlaylists.asData?.value.items ?? [], + // title: Text(context.l10n.featured), + // isLoadingNextPage: featuredPlaylists.isLoadingNextPage, + // hasNextPage: featuredPlaylists.asData?.value.hasMore ?? false, + // onFetchMore: featuredPlaylistsNotifier.fetchMore, + // ), + // ); } } diff --git a/lib/modules/home/sections/feed.dart b/lib/modules/home/sections/feed.dart deleted file mode 100644 index 8685fe19..00000000 --- a/lib/modules/home/sections/feed.dart +++ /dev/null @@ -1,59 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/home/feed/feed_section.dart'; -import 'package:spotube/provider/spotify/views/home.dart'; -import 'package:spotube/utils/service_utils.dart'; - -class HomePageFeedSection extends HookConsumerWidget { - const HomePageFeedSection({super.key}); - - @override - Widget build(BuildContext context, ref) { - final homeFeed = ref.watch(homeViewProvider); - final nonShortSections = homeFeed.asData?.value?.sections - .where((s) => s.typename == "HomeGenericSectionData") - .toList() ?? - []; - - return SliverList.builder( - itemCount: nonShortSections.length, - itemBuilder: (context, index) { - final section = nonShortSections[index]; - if (section.items.isEmpty) return const SizedBox.shrink(); - - return HorizontalPlaybuttonCardView( - items: [ - for (final item in section.items) - if (item.album != null) - item.album!.asAlbum - else if (item.artist != null) - item.artist!.asArtist - else if (item.playlist != null) - item.playlist!.asPlaylist - ], - title: Text(section.title ?? context.l10n.no_title), - hasNextPage: false, - isLoadingNextPage: false, - onFetchMore: () {}, - titleTrailing: Directionality( - textDirection: TextDirection.rtl, - child: TextButton.icon( - label: Text(context.l10n.browse_more), - icon: const Icon(SpotubeIcons.angleRight), - onPressed: () => ServiceUtils.pushNamed( - context, - HomeFeedSectionPage.name, - pathParameters: { - "feedId": section.uri, - }, - ), - ), - ), - ); - }, - ); - } -} diff --git a/lib/modules/home/sections/friends.dart b/lib/modules/home/sections/friends.dart deleted file mode 100644 index 6f59c209..00000000 --- a/lib/modules/home/sections/friends.dart +++ /dev/null @@ -1,111 +0,0 @@ -import 'dart:ui'; - -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/modules/home/sections/friends/friend_item.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/hooks/utils/use_breakpoint_value.dart'; -import 'package:spotube/models/spotify_friends.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class HomePageFriendsSection extends HookConsumerWidget { - const HomePageFriendsSection({super.key}); - - @override - Widget build(BuildContext context, ref) { - final auth = ref.watch(authenticationProvider); - final friendsQuery = ref.watch(friendsProvider); - final friends = - friendsQuery.asData?.value.friends ?? FakeData.friends.friends; - - final groupCount = useBreakpointValue( - sm: 3, - xs: 2, - md: 4, - lg: 5, - xl: 6, - xxl: 7, - ); - - final friendGroup = useMemoized( - () => friends.fold>>( - [], - (previousValue, element) { - if (previousValue.isEmpty) { - return [ - [element] - ]; - } - - final lastGroup = previousValue.last; - if (lastGroup.length < groupCount) { - return [ - ...previousValue.sublist(0, previousValue.length - 1), - [...lastGroup, element] - ]; - } - - return [ - ...previousValue, - [element] - ]; - }, - ), - [friends, groupCount], - ); - - if (friendsQuery.isLoading || - friendsQuery.asData?.value.friends.isEmpty == true || - auth.asData?.value == null) { - return const SliverToBoxAdapter( - child: SizedBox.shrink(), - ); - } - - return Skeletonizer.sliver( - enabled: friendsQuery.isLoading, - child: SliverMainAxisGroup( - slivers: [ - SliverToBoxAdapter( - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Text( - context.l10n.friends, - style: Theme.of(context).textTheme.titleMedium, - ), - ), - ), - SliverToBoxAdapter( - child: ScrollConfiguration( - behavior: ScrollConfiguration.of(context).copyWith( - dragDevices: PointerDeviceKind.values.toSet(), - ), - child: SingleChildScrollView( - scrollDirection: Axis.horizontal, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - for (final group in friendGroup) - Row( - children: [ - for (final friend in group) - Padding( - padding: const EdgeInsets.all(8.0), - child: FriendItem(friend: friend), - ), - ], - ), - ], - ), - ), - ), - ), - ], - ), - ); - } -} diff --git a/lib/modules/home/sections/friends/friend_item.dart b/lib/modules/home/sections/friends/friend_item.dart deleted file mode 100644 index 773a4a8c..00000000 --- a/lib/modules/home/sections/friends/friend_item.dart +++ /dev/null @@ -1,136 +0,0 @@ -import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/models/spotify_friends.dart'; -import 'package:spotube/pages/album/album.dart'; -import 'package:spotube/pages/artist/artist.dart'; -import 'package:spotube/pages/track/track.dart'; -import 'package:spotube/provider/spotify_provider.dart'; - -class FriendItem extends HookConsumerWidget { - final SpotifyFriendActivity friend; - const FriendItem({ - super.key, - required this.friend, - }); - - @override - Widget build(BuildContext context, ref) { - final ThemeData( - textTheme: textTheme, - colorScheme: colorScheme, - ) = Theme.of(context); - - final spotify = ref.watch(spotifyProvider); - - return Container( - padding: const EdgeInsets.all(8), - decoration: BoxDecoration( - color: colorScheme.surfaceContainerHighest.withOpacity(0.3), - borderRadius: BorderRadius.circular(15), - ), - constraints: const BoxConstraints( - minWidth: 300, - ), - height: 80, - child: Row( - children: [ - CircleAvatar( - backgroundImage: UniversalImage.imageProvider( - friend.user.imageUrl, - ), - ), - const Gap(8), - Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text( - friend.user.name, - style: textTheme.bodyLarge, - ), - RichText( - text: TextSpan( - style: textTheme.bodySmall, - children: [ - TextSpan( - text: friend.track.name, - recognizer: TapGestureRecognizer() - ..onTap = () { - context.pushNamed(TrackPage.name, pathParameters: { - "id": friend.track.id, - }); - }, - ), - const TextSpan(text: " • "), - const WidgetSpan( - child: Icon( - SpotubeIcons.artist, - size: 12, - ), - ), - TextSpan( - text: " ${friend.track.artist.name}", - recognizer: TapGestureRecognizer() - ..onTap = () { - context.pushNamed( - ArtistPage.name, - pathParameters: { - "id": friend.track.artist.id, - }, - extra: friend.track.artist, - ); - }, - ), - const TextSpan(text: "\n"), - TextSpan( - text: friend.track.context.name, - recognizer: TapGestureRecognizer() - ..onTap = () async { - context.push( - "/${friend.track.context.path}", - extra: - !friend.track.context.path.startsWith("album") - ? null - : await spotify.albums - .get(friend.track.context.id), - ); - }, - ), - const TextSpan(text: " • "), - const WidgetSpan( - child: Icon( - SpotubeIcons.album, - size: 12, - ), - ), - TextSpan( - text: " ${friend.track.album.name}", - recognizer: TapGestureRecognizer() - ..onTap = () async { - final album = - await spotify.albums.get(friend.track.album.id); - if (context.mounted) { - context.pushNamed( - AlbumPage.name, - pathParameters: { - "id": friend.track.album.id, - }, - extra: album, - ); - } - }, - ), - ], - ), - ), - ], - ), - ], - ), - ); - } -} diff --git a/lib/modules/home/sections/genres.dart b/lib/modules/home/sections/genres.dart deleted file mode 100644 index 5f2dfa5e..00000000 --- a/lib/modules/home/sections/genres.dart +++ /dev/null @@ -1,160 +0,0 @@ -import 'dart:math'; - -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/collections/gradients.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/home/genres/genre_playlists.dart'; -import 'package:spotube/pages/home/genres/genres.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class HomeGenresSection extends HookConsumerWidget { - const HomeGenresSection({super.key}); - - @override - Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :colorScheme) = Theme.of(context); - final mediaQuery = MediaQuery.of(context); - - final categoriesQuery = ref.watch(categoriesProvider); - final categories = useMemoized( - () => - categoriesQuery.asData?.value - .where((c) => (c.icons?.length ?? 0) > 0) - .take(mediaQuery.mdAndDown ? 6 : 10) - .toList() ?? - [], - [mediaQuery.mdAndDown, categoriesQuery.asData?.value], - ); - - return SliverMainAxisGroup( - slivers: [ - SliverToBoxAdapter( - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 8), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text( - context.l10n.genres, - style: textTheme.headlineSmall, - ), - Directionality( - textDirection: TextDirection.rtl, - child: TextButton.icon( - onPressed: () { - context.pushNamed(GenrePage.name); - }, - icon: const Icon(SpotubeIcons.angleRight), - label: Text( - context.l10n.browse_all, - style: textTheme.bodyMedium?.copyWith( - color: colorScheme.secondary, - ), - ), - ), - ), - ], - ), - ), - ), - const SliverGap(8), - SliverPadding( - padding: const EdgeInsets.symmetric(horizontal: 16), - sliver: Skeletonizer.sliver( - enabled: categoriesQuery.isLoading, - child: SliverGrid.builder( - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: mediaQuery.mdAndDown ? 200 : 250, - mainAxisExtent: 50, - crossAxisSpacing: 16, - mainAxisSpacing: 16, - ), - itemCount: categoriesQuery.isLoading - ? mediaQuery.mdAndDown - ? 6 - : 10 - : categories.length, - itemBuilder: (context, index) { - final category = - categories.elementAtOrNull(index) ?? FakeData.category; - - return HookBuilder(builder: (context) { - final (:gradient, :textColor) = useMemoized( - () { - final gradient = - gradients[Random().nextInt(gradients.length)]; - final text = gradient.colors - .take(2) - .any((c) => c.computeLuminance() > 0.5) - ? Colors.grey[900] - : Colors.white; - return ( - gradient: LinearGradient( - colors: gradient.colors - .map((c) => c.withOpacity(0.8)) - .toList(), - ), - textColor: text - ); - }, - [], - ); - - return InkWell( - onTap: () { - context.pushNamed( - GenrePlaylistsPage.name, - pathParameters: { - "categoryId": category.id!, - }, - extra: category, - ); - }, - borderRadius: BorderRadius.circular(8), - child: Ink( - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(8), - image: DecorationImage( - image: UniversalImage.imageProvider( - category.icons!.first.url!, - ), - fit: BoxFit.cover, - ), - ), - child: Ink( - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(5), - color: colorScheme.surfaceContainerHighest, - gradient: categoriesQuery.isLoading ? null : gradient, - ), - padding: const EdgeInsets.symmetric(horizontal: 16), - child: Align( - alignment: Alignment.centerLeft, - child: Text( - category.name!, - style: textTheme.titleMedium - ?.copyWith(color: textColor), - ), - ), - ), - ), - ); - }); - }, - ), - ), - ), - ], - ); - } -} diff --git a/lib/modules/home/sections/made_for_user.dart b/lib/modules/home/sections/made_for_user.dart deleted file mode 100644 index 1b9854d3..00000000 --- a/lib/modules/home/sections/made_for_user.dart +++ /dev/null @@ -1,35 +0,0 @@ -import 'package:flutter/widgets.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class HomeMadeForUserSection extends HookConsumerWidget { - const HomeMadeForUserSection({super.key}); - - @override - Widget build(BuildContext context, ref) { - final madeForUser = ref.watch(viewProvider("made-for-x-hub")); - - return SliverList.builder( - itemCount: madeForUser.asData?.value["content"]?["items"]?.length ?? 0, - itemBuilder: (context, index) { - final item = madeForUser.asData?.value["content"]?["items"]?[index]; - final playlists = item["content"]?["items"] - ?.where((itemL2) => itemL2["type"] == "playlist") - .map((itemL2) => PlaylistSimple.fromJson(itemL2)) - .toList() - .cast() ?? - []; - if (playlists.isEmpty) return const SizedBox.shrink(); - return HorizontalPlaybuttonCardView( - items: playlists, - title: Text(item["name"] ?? ""), - hasNextPage: false, - isLoadingNextPage: false, - onFetchMore: () {}, - ); - }, - ); - } -} diff --git a/lib/modules/home/sections/new_releases.dart b/lib/modules/home/sections/new_releases.dart index e2b32741..be6d335d 100644 --- a/lib/modules/home/sections/new_releases.dart +++ b/lib/modules/home/sections/new_releases.dart @@ -1,35 +1,55 @@ -import 'package:flutter/material.dart' hide Page; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/album/releases.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; class HomeNewReleasesSection extends HookConsumerWidget { const HomeNewReleasesSection({super.key}); @override Widget build(BuildContext context, ref) { - final auth = ref.watch(authenticationProvider); + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); - final newReleases = ref.watch(albumReleasesProvider); - final newReleasesNotifier = ref.read(albumReleasesProvider.notifier); + final newReleases = ref.watch(metadataPluginAlbumReleasesProvider); + final newReleasesNotifier = + ref.read(metadataPluginAlbumReleasesProvider.notifier); - final albums = ref.watch(userArtistAlbumReleasesProvider); - - if (auth.asData?.value == null || + if (authenticated.asData?.value != true || newReleases.isLoading || newReleases.asData?.value.items.isEmpty == true) { return const SizedBox.shrink(); } - return HorizontalPlaybuttonCardView( - items: albums, + if (newReleases.error + case MetadataPluginException( + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + message: _, + )) { + return const SizedBox.shrink(); + } + + return HorizontalPlaybuttonCardView( + items: newReleases.asData?.value.items ?? [], title: Text(context.l10n.new_releases), isLoadingNextPage: newReleases.isLoadingNextPage, hasNextPage: newReleases.asData?.value.hasMore ?? false, onFetchMore: newReleasesNotifier.fetchMore, + error: newReleases.hasError + ? Center( + child: ErrorBox( + error: newReleases.error!, + onRetry: () { + ref.invalidate(metadataPluginAlbumReleasesProvider); + }, + ), + ) + : null, ); } } diff --git a/lib/modules/home/sections/recent.dart b/lib/modules/home/sections/recent.dart index 43c0459d..5420ad55 100644 --- a/lib/modules/home/sections/recent.dart +++ b/lib/modules/home/sections/recent.dart @@ -1,5 +1,5 @@ -import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/fake.dart'; import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; diff --git a/lib/modules/home/sections/sections.dart b/lib/modules/home/sections/sections.dart new file mode 100644 index 00000000..93055b74 --- /dev/null +++ b/lib/modules/home/sections/sections.dart @@ -0,0 +1,106 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/fallbacks/no_default_metadata_plugin.dart'; +import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/provider/metadata_plugin/browse/sections.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; +import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; + +class HomePageBrowseSection extends HookConsumerWidget { + const HomePageBrowseSection({super.key}); + + @override + Widget build(BuildContext context, ref) { + final browseSections = ref.watch(metadataPluginBrowseSectionsProvider); + final sections = browseSections.asData?.value.items; + final ThemeData(:colorScheme) = Theme.of(context); + + if (browseSections.isLoading) { + return SliverToBoxAdapter( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + spacing: 16, + children: [ + Undraw( + height: 200, + illustration: UndrawIllustration.process, + color: colorScheme.primary, + ), + Row( + mainAxisAlignment: MainAxisAlignment.center, + spacing: 8, + children: [ + const CircularProgressIndicator(), + Text(context.l10n.building_your_timeline).muted, + ], + ), + const Gap(16), + ], + ), + ); + } + + if (browseSections.error + case MetadataPluginException( + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + message: _, + )) { + return const SliverFillRemaining( + child: Center(child: NoDefaultMetadataPlugin()), + ); + } + + if (browseSections.hasError) { + return SliverFillRemaining( + child: Center( + child: ErrorBox( + error: browseSections.error!, + onRetry: () { + ref.invalidate(metadataPluginBrowseSectionsProvider); + }, + ), + ), + ); + } + + return SliverInfiniteList( + hasReachedMax: browseSections.asData?.value.hasMore == false, + isLoading: !browseSections.isLoading && browseSections.isLoadingNextPage, + onFetchData: () { + ref.read(metadataPluginBrowseSectionsProvider.notifier).fetchMore(); + }, + itemCount: sections?.length ?? 0, + itemBuilder: (context, index) { + final section = sections![index]; + if (section.items.isEmpty) return const SizedBox.shrink(); + + return HorizontalPlaybuttonCardView( + items: section.items, + title: Text(section.title), + hasNextPage: false, + isLoadingNextPage: false, + onFetchMore: () {}, + titleTrailing: section.browseMore + ? Button.text( + child: Text(context.l10n.browse_all), + onPressed: () { + context.navigateTo( + HomeBrowseSectionItemsRoute( + sectionId: section.id, + section: section, + ), + ); + }, + ) + : null, + ); + }, + ); + } +} diff --git a/lib/modules/library/local_folder/cache_export_dialog.dart b/lib/modules/library/local_folder/cache_export_dialog.dart index 1d1421be..4c86a8d5 100644 --- a/lib/modules/library/local_folder/cache_export_dialog.dart +++ b/lib/modules/library/local_folder/cache_export_dialog.dart @@ -1,15 +1,13 @@ import 'dart:io'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:path/path.dart'; +import 'package:path/path.dart' as path; import 'package:spotube/extensions/context.dart'; import 'package:spotube/services/logger/logger.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -final codecs = SourceCodecs.values.map((s) => s.name); +const containers = ["m4a", "mp3", "mp4", "ogg", "wav", "flac"]; class LocalFolderCacheExportDialog extends HookConsumerWidget { final Directory exportDir; @@ -22,7 +20,7 @@ class LocalFolderCacheExportDialog extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :colorScheme) = Theme.of(context); + final ThemeData(:typography, :colorScheme) = Theme.of(context); final files = useState>([]); final filesExported = useState(0); @@ -31,7 +29,8 @@ class LocalFolderCacheExportDialog extends HookConsumerWidget { final stream = cacheDir.list().where( (event) => event is File && - codecs.contains(extension(event.path).replaceAll(".", "")), + containers + .contains(path.extension(event.path).replaceAll(".", "")), ); stream.listen( @@ -76,8 +75,8 @@ class LocalFolderCacheExportDialog extends HookConsumerWidget { ), TextSpan( text: "\n${exportDir.path}?", - style: textTheme.labelMedium!.copyWith( - color: colorScheme.secondary, + style: typography.small.copyWith( + color: colorScheme.mutedForeground, ), ), ], @@ -102,7 +101,7 @@ class LocalFolderCacheExportDialog extends HookConsumerWidget { ), ), actions: [ - TextButton( + Button.outline( onPressed: isExportInProgress ? null : () { @@ -110,14 +109,14 @@ class LocalFolderCacheExportDialog extends HookConsumerWidget { }, child: Text(context.l10n.cancel), ), - TextButton( + Button.primary( onPressed: isExportInProgress ? null : () async { for (final file in files.value) { try { final destinationFile = File( - join(exportDir.path, basename(file.path)), + path.join(exportDir.path, path.basename(file.path)), ); if (await destinationFile.exists()) { diff --git a/lib/modules/library/local_folder/local_folder_item.dart b/lib/modules/library/local_folder/local_folder_item.dart index a965a42d..8bed5f7f 100644 --- a/lib/modules/library/local_folder/local_folder_item.dart +++ b/lib/modules/library/local_folder/local_folder_item.dart @@ -1,19 +1,18 @@ import 'dart:math'; -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:path/path.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; import 'package:spotube/extensions/string.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; -import 'package:spotube/pages/library/local_folder.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; @@ -26,8 +25,6 @@ class LocalFolderItem extends HookConsumerWidget { final ThemeData(:colorScheme) = Theme.of(context); final mediaQuery = MediaQuery.of(context); - final lerpValue = useBrightnessValue(.9, .7); - final downloadFolder = ref.watch(userPreferencesProvider.select((s) => s.downloadLocation)); final cacheFolder = useFuture(UserPreferencesNotifier.getMusicCacheDir()); @@ -35,23 +32,6 @@ class LocalFolderItem extends HookConsumerWidget { final isDownloadFolder = folder == downloadFolder; final isCacheFolder = folder == cacheFolder.data; - final Uri(:pathSegments) = Uri.parse( - folder - .replaceFirst(RegExp(r'^/Volumes/[^/]+/Users/'), "") - .replaceFirst(r'C:\Users\', "") - .replaceFirst(r'/home/', ""), - ); - - // if length > 5, we ... all the middle segments after 2 and the last 2 - final segments = pathSegments.length > 5 - ? [ - ...pathSegments.take(2), - "...", - ...pathSegments.skip(pathSegments.length - 3).toList() - ..removeLast(), - ] - : pathSegments.take(max(pathSegments.length - 1, 0)).toList(); - final trackSnapshot = ref.watch( localTracksProvider.select( (s) => s.whenData((tracks) => tracks[folder]?.take(4).toList()), @@ -60,98 +40,91 @@ class LocalFolderItem extends HookConsumerWidget { final tracks = trackSnapshot.value ?? []; - return InkWell( - onTap: () { - context.goNamed( - LocalLibraryPage.name, - queryParameters: { - if (isDownloadFolder) "downloads": "true", - if (isCacheFolder) "cache": "true", - }, - extra: folder, + return Button( + onPressed: () { + context.navigateTo( + LocalLibraryRoute( + location: folder, + isCache: isCacheFolder, + isDownloads: isDownloadFolder, + ), ); }, - borderRadius: BorderRadius.circular(8), - child: Ink( - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(8), - color: Color.lerp( - colorScheme.surfaceContainerHighest, - colorScheme.surface, - lerpValue, - ), - ), - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - if (tracks.isEmpty) - Card( - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Icon( - SpotubeIcons.folder, - size: mediaQuery.smAndDown - ? 95 - : mediaQuery.mdAndDown - ? 100 - : 142, - ), - ), - ) - else - ClipRRect( - borderRadius: BorderRadius.circular(8), - child: GridView.builder( - shrinkWrap: true, - physics: const NeverScrollableScrollPhysics(), - gridDelegate: SliverGridDelegateWithFixedCrossAxisCount( - crossAxisCount: max((tracks.length / 2).ceil(), 2), - ), - itemCount: tracks.length, - itemBuilder: (context, index) { - final track = tracks[index]; - return UniversalImage( - path: (track.album?.images).asUrlString( - placeholder: ImagePlaceholder.albumArt, - ), - fit: BoxFit.cover, - ); - }, - ), + style: ButtonVariance.card.copyWith( + padding: (context, states, value) { + return const EdgeInsets.all(8); + }, + ), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + if (tracks.isEmpty) + Padding( + padding: const EdgeInsets.all(8.0), + child: Icon( + SpotubeIcons.folder, + size: mediaQuery.smAndDown + ? 95 + : mediaQuery.mdAndDown + ? 100 + : 142, + ), + ) + else + ClipRRect( + borderRadius: BorderRadius.circular(8), + child: GridView.builder( + shrinkWrap: true, + physics: const NeverScrollableScrollPhysics(), + gridDelegate: SliverGridDelegateWithFixedCrossAxisCount( + crossAxisCount: max((tracks.length / 2).ceil(), 2), ), - const Gap(8), - Stack( - children: [ - Center( - child: Text( - isDownloadFolder - ? context.l10n.downloads - : isCacheFolder - ? context.l10n.cache_folder.capitalize() - : basename(folder), - style: const TextStyle(fontWeight: FontWeight.bold), - textAlign: TextAlign.center, + itemCount: tracks.length, + itemBuilder: (context, index) { + final track = tracks[index]; + return UniversalImage( + path: track.album.images.asUrlString( + placeholder: ImagePlaceholder.albumArt, ), - ), - if (!isDownloadFolder) - Align( - alignment: Alignment.topRight, - child: PopupMenuButton( - child: const Padding( - padding: EdgeInsets.all(3), - child: Icon(Icons.more_vert), - ), - itemBuilder: (context) { - return [ - PopupMenuItem( - child: ListTile( - leading: const Icon(SpotubeIcons.folderRemove), - iconColor: colorScheme.error, - title: + fit: BoxFit.cover, + ); + }, + ), + ), + const Gap(8), + Stack( + children: [ + Center( + child: Text( + isDownloadFolder + ? context.l10n.downloads + : isCacheFolder + ? context.l10n.cache_folder.capitalize() + : basename(folder), + style: const TextStyle(fontWeight: FontWeight.bold), + textAlign: TextAlign.center, + maxLines: 1, + overflow: TextOverflow.ellipsis, + ), + ), + if (!isDownloadFolder && !isCacheFolder) + Align( + alignment: Alignment.topRight, + child: IconButton.ghost( + icon: const Icon(Icons.more_vert), + size: ButtonSize.small, + onPressed: () { + showDropdown( + context: context, + builder: (context) { + return DropdownMenu( + children: [ + MenuButton( + leading: Icon(SpotubeIcons.folderRemove, + color: colorScheme.destructive), + child: Text(context.l10n.remove_library_location), - onTap: () { + onPressed: (context) { final libraryLocations = ref .read(userPreferencesProvider) .localLibraryLocation; @@ -163,43 +136,18 @@ class LocalFolderItem extends HookConsumerWidget { .toList(), ); }, - ), - ) - ]; + ) + ], + ); }, - ), - ), - ], - ), - const Spacer(), - Wrap( - spacing: 2, - runSpacing: 2, - children: [ - for (final MapEntry(key: index, value: segment) - in segments.asMap().entries) - Text.rich( - TextSpan( - children: [ - if (index != 0) - TextSpan( - text: "/ ", - style: TextStyle(color: colorScheme.primary), - ), - TextSpan(text: segment), - ], - ), - style: TextStyle( - fontSize: 10, - color: colorScheme.tertiary, - ), - ), - ], - ), - const Spacer(), + ); + }, + ), + ), ], ), - ), + const Spacer(), + ], ), ); } diff --git a/lib/modules/library/playlist_generate/multi_select_field.dart b/lib/modules/library/playlist_generate/multi_select_field.dart deleted file mode 100644 index 7118d57d..00000000 --- a/lib/modules/library/playlist_generate/multi_select_field.dart +++ /dev/null @@ -1,272 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; - -class MultiSelectField extends HookWidget { - final List options; - final List selectedOptions; - - final Widget Function(T option, VoidCallback onSelect)? optionBuilder; - final Widget Function(T option)? selectedOptionBuilder; - final ValueChanged> onSelected; - - final Widget? dialogTitle; - - final Object Function(T option) getValueForOption; - - final Widget label; - - final String? helperText; - - final bool enabled; - - const MultiSelectField({ - super.key, - required this.options, - required this.selectedOptions, - required this.getValueForOption, - required this.label, - this.optionBuilder, - this.selectedOptionBuilder, - required this.onSelected, - this.dialogTitle, - this.helperText, - this.enabled = true, - }); - - Widget defaultSelectedOptionBuilder(T option) { - return Chip( - label: Text(option.toString()), - onDeleted: () { - onSelected( - selectedOptions.where((e) => e != getValueForOption(option)).toList(), - ); - }, - ); - } - - @override - Widget build(BuildContext context) { - final theme = Theme.of(context); - - return Column( - crossAxisAlignment: CrossAxisAlignment.stretch, - children: [ - MaterialButton( - elevation: 0, - focusElevation: 0, - hoverElevation: 0, - disabledElevation: 0, - highlightElevation: 0, - padding: const EdgeInsets.symmetric(vertical: 22), - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(15), - side: BorderSide( - color: enabled - ? theme.colorScheme.onSurface - : theme.colorScheme.onSurface.withOpacity(0.1), - ), - ), - mouseCursor: WidgetStateMouseCursor.textable, - onPressed: !enabled - ? null - : () async { - final selected = await showDialog>( - context: context, - builder: (context) { - return _MultiSelectDialog( - dialogTitle: dialogTitle, - options: options, - getValueForOption: getValueForOption, - optionBuilder: optionBuilder, - initialSelection: selectedOptions, - helperText: helperText, - ); - }, - ); - if (selected != null) { - onSelected(selected); - } - }, - child: Container( - alignment: Alignment.centerLeft, - margin: const EdgeInsets.symmetric(horizontal: 10), - child: DefaultTextStyle( - style: theme.textTheme.titleMedium!, - child: label, - ), - ), - ), - if (helperText != null) - Padding( - padding: const EdgeInsets.all(8.0), - child: Text( - helperText!, - style: Theme.of(context).textTheme.bodySmall, - ), - ), - Wrap( - children: [ - ...selectedOptions.map( - (option) => Padding( - padding: const EdgeInsets.all(4.0), - child: (selectedOptionBuilder ?? - defaultSelectedOptionBuilder)(option), - ), - ), - ], - ) - ], - ); - } -} - -class _MultiSelectDialog extends HookWidget { - final Widget? dialogTitle; - final List options; - final Widget Function(T option, VoidCallback onSelect)? optionBuilder; - final Object Function(T option) getValueForOption; - final List initialSelection; - final String? helperText; - - const _MultiSelectDialog({ - super.key, - required this.dialogTitle, - required this.options, - required this.getValueForOption, - this.optionBuilder, - this.initialSelection = const [], - this.helperText, - }); - - @override - Widget build(BuildContext context) { - final mediaQuery = MediaQuery.of(context); - final selected = useState(initialSelection.map(getValueForOption)); - - final searchController = useTextEditingController(); - - // creates render update - useValueListenable(searchController); - - final filteredOptions = useMemoized( - () { - if (searchController.text.isEmpty) { - return options; - } - - return options - .map((e) => ( - weightedRatio( - getValueForOption(e).toString(), searchController.text), - e - )) - .sorted((a, b) => b.$1.compareTo(a.$1)) - .where((e) => e.$1 > 50) - .map((e) => e.$2) - .toList(); - }, - [searchController.text, options, getValueForOption], - ); - - Widget defaultOptionBuilder(T option, VoidCallback onSelect) { - final isSelected = selected.value.contains(getValueForOption(option)); - return ChoiceChip( - label: Text("${!isSelected ? " " : ""}${option.toString()}"), - selected: isSelected, - side: BorderSide.none, - onSelected: (_) { - onSelect(); - }, - ); - } - - return AlertDialog( - scrollable: true, - title: dialogTitle ?? Text(context.l10n.select), - contentPadding: mediaQuery.mdAndUp ? null : const EdgeInsets.all(16), - insetPadding: const EdgeInsets.all(16), - actions: [ - OutlinedButton( - onPressed: () { - Navigator.pop(context); - }, - child: Text(context.l10n.cancel), - ), - ElevatedButton( - onPressed: () { - Navigator.pop( - context, - options - .where( - (option) => - selected.value.contains(getValueForOption(option)), - ) - .toList(), - ); - }, - child: Text(context.l10n.done), - ), - ], - content: SizedBox( - height: mediaQuery.size.height * 0.5, - width: 400, - child: Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: CrossAxisAlignment.stretch, - children: [ - TextField( - autofocus: true, - controller: searchController, - decoration: InputDecoration( - hintText: context.l10n.search, - prefixIcon: const Icon(SpotubeIcons.search), - border: OutlineInputBorder( - borderRadius: BorderRadius.circular(15), - ), - ), - ), - const SizedBox(height: 10), - Expanded( - child: SingleChildScrollView( - child: Wrap( - spacing: 5, - runSpacing: 5, - children: [ - ...filteredOptions.map( - (option) => Padding( - padding: const EdgeInsets.all(4.0), - child: (optionBuilder ?? defaultOptionBuilder)( - option, - () { - final value = getValueForOption(option); - if (selected.value.contains(value)) { - selected.value = selected.value - .where((e) => e != value) - .toList(); - } else { - selected.value = [...selected.value, value]; - } - }, - ), - ), - ), - ], - ), - ), - ), - if (helperText != null) - Text( - helperText!, - style: Theme.of(context).textTheme.labelMedium, - ), - ], - ), - ), - ); - } -} diff --git a/lib/modules/library/playlist_generate/recommendation_attribute_dials.dart b/lib/modules/library/playlist_generate/recommendation_attribute_dials.dart deleted file mode 100644 index d7f51ffb..00000000 --- a/lib/modules/library/playlist_generate/recommendation_attribute_dials.dart +++ /dev/null @@ -1,182 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/library/playlist_generate/playlist_generate.dart'; - -typedef RecommendationAttribute = ({double min, double target, double max}); - -RecommendationAttribute lowValues(double base) => - (min: 1 * base, target: 0.3 * base, max: 0.3 * base); -RecommendationAttribute moderateValues(double base) => - (min: 0.5 * base, target: 1 * base, max: 0.5 * base); -RecommendationAttribute highValues(double base) => - (min: 0.3 * base, target: 0.3 * base, max: 1 * base); - -class RecommendationAttributeDials extends HookWidget { - final Widget title; - final RecommendationAttribute values; - final ValueChanged onChanged; - final double base; - - const RecommendationAttributeDials({ - super.key, - required this.values, - required this.onChanged, - required this.title, - this.base = 1, - }); - - @override - Widget build(BuildContext context) { - final animation = useAnimationController( - duration: const Duration(milliseconds: 300), - ); - final labelStyle = Theme.of(context).textTheme.labelSmall?.copyWith( - fontWeight: FontWeight.w500, - ); - - final minSlider = Row( - children: [ - Text(context.l10n.min, style: labelStyle), - Expanded( - child: Slider( - value: values.min / base, - min: 0, - max: 1, - onChanged: (value) => onChanged(( - min: value * base, - target: values.target, - max: values.max, - )), - ), - ), - ], - ); - - final targetSlider = Row( - children: [ - Text(context.l10n.target, style: labelStyle), - Expanded( - child: Slider( - value: values.target / base, - min: 0, - max: 1, - onChanged: (value) => onChanged(( - min: values.min, - target: value * base, - max: values.max, - )), - ), - ), - ], - ); - - final maxSlider = Row( - children: [ - Text(context.l10n.max, style: labelStyle), - Expanded( - child: Slider( - value: values.max / base, - min: 0, - max: 1, - onChanged: (value) => onChanged(( - min: values.min, - target: values.target, - max: value * base, - )), - ), - ), - ], - ); - - return LayoutBuilder(builder: (context, constrain) { - return Card( - child: ExpansionTile( - title: DefaultTextStyle( - style: Theme.of(context).textTheme.titleSmall!, - child: title, - ), - shape: const Border(), - leading: AnimatedBuilder( - animation: animation, - builder: (context, child) { - return Transform.rotate( - angle: (animation.value * 3.14) / 2, - child: child, - ); - }, - child: const Icon(Icons.chevron_right), - ), - trailing: Padding( - padding: const EdgeInsets.symmetric(vertical: 8.0), - child: ToggleButtons( - borderRadius: BorderRadius.circular(8), - textStyle: labelStyle, - isSelected: [ - values == lowValues(base), - values == moderateValues(base), - values == highValues(base), - ], - onPressed: (index) { - RecommendationAttribute newValues = zeroValues; - switch (index) { - case 0: - newValues = lowValues(base); - break; - case 1: - newValues = moderateValues(base); - break; - case 2: - newValues = highValues(base); - break; - } - - if (newValues == values) { - onChanged(zeroValues); - } else { - onChanged(newValues); - } - }, - children: [ - Text(context.l10n.low), - Text(" ${context.l10n.moderate} "), - Text(context.l10n.high), - ], - ), - ), - onExpansionChanged: (value) { - if (value) { - animation.forward(); - } else { - animation.reverse(); - } - }, - children: [ - if (constrain.mdAndUp) - Row( - mainAxisAlignment: MainAxisAlignment.spaceEvenly, - children: [ - const SizedBox(width: 16), - Expanded(child: minSlider), - Expanded(child: targetSlider), - Expanded(child: maxSlider), - ], - ) - else - Padding( - padding: const EdgeInsets.only(left: 16), - child: Column( - children: [ - minSlider, - targetSlider, - maxSlider, - ], - ), - ), - ], - ), - ); - }); - } -} diff --git a/lib/modules/library/playlist_generate/recommendation_attribute_fields.dart b/lib/modules/library/playlist_generate/recommendation_attribute_fields.dart deleted file mode 100644 index 7feff03a..00000000 --- a/lib/modules/library/playlist_generate/recommendation_attribute_fields.dart +++ /dev/null @@ -1,179 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:spotube/modules/library/playlist_generate/recommendation_attribute_dials.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/library/playlist_generate/playlist_generate.dart'; - -class RecommendationAttributeFields extends HookWidget { - final Widget title; - final RecommendationAttribute values; - final ValueChanged onChanged; - final Map? presets; - - const RecommendationAttributeFields({ - super.key, - required this.values, - required this.onChanged, - required this.title, - this.presets, - }); - - @override - Widget build(BuildContext context) { - final animation = useAnimationController( - duration: const Duration(milliseconds: 300), - ); - final labelStyle = Theme.of(context).textTheme.labelSmall?.copyWith( - fontWeight: FontWeight.w500, - ); - - final minController = useTextEditingController(text: values.min.toString()); - final targetController = - useTextEditingController(text: values.target.toString()); - final maxController = useTextEditingController(text: values.max.toString()); - - useEffect(() { - listener() { - onChanged(( - min: double.tryParse(minController.text) ?? 0, - target: double.tryParse(targetController.text) ?? 0, - max: double.tryParse(maxController.text) ?? 0, - )); - } - - minController.addListener(listener); - targetController.addListener(listener); - maxController.addListener(listener); - - return () { - minController.removeListener(listener); - targetController.removeListener(listener); - maxController.removeListener(listener); - }; - }, [values]); - - final minField = TextField( - controller: minController, - decoration: InputDecoration( - labelText: context.l10n.min, - isDense: true, - ), - keyboardType: const TextInputType.numberWithOptions( - decimal: false, - signed: true, - ), - ); - - final targetField = TextField( - controller: targetController, - decoration: InputDecoration( - labelText: context.l10n.target, - isDense: true, - ), - keyboardType: const TextInputType.numberWithOptions( - decimal: false, - signed: true, - ), - ); - - final maxField = TextField( - controller: maxController, - decoration: InputDecoration( - labelText: context.l10n.max, - isDense: true, - ), - keyboardType: const TextInputType.numberWithOptions( - decimal: false, - signed: true, - ), - ); - - return LayoutBuilder(builder: (context, constrain) { - return Card( - child: ExpansionTile( - title: DefaultTextStyle( - style: Theme.of(context).textTheme.titleSmall!, - child: title, - ), - shape: const Border(), - leading: AnimatedBuilder( - animation: animation, - builder: (context, child) { - return Transform.rotate( - angle: (animation.value * 3.14) / 2, - child: child, - ); - }, - child: const Icon(Icons.chevron_right), - ), - trailing: presets == null - ? const SizedBox.shrink() - : Padding( - padding: const EdgeInsets.symmetric(vertical: 8.0), - child: ToggleButtons( - borderRadius: BorderRadius.circular(8), - textStyle: labelStyle, - isSelected: presets!.values - .map((value) => value == values) - .toList(), - onPressed: (index) { - RecommendationAttribute newValues = - presets!.values.elementAt(index); - if (newValues == values) { - onChanged(zeroValues); - minController.text = zeroValues.min.toString(); - targetController.text = zeroValues.target.toString(); - maxController.text = zeroValues.max.toString(); - } else { - onChanged(newValues); - minController.text = newValues.min.toString(); - targetController.text = newValues.target.toString(); - maxController.text = newValues.max.toString(); - } - }, - children: presets!.keys.map((key) => Text(key)).toList(), - ), - ), - onExpansionChanged: (value) { - if (value) { - animation.forward(); - } else { - animation.reverse(); - } - }, - children: [ - const SizedBox(height: 8), - if (constrain.mdAndUp) - Row( - mainAxisAlignment: MainAxisAlignment.spaceEvenly, - children: [ - const SizedBox(width: 16), - Expanded(child: minField), - const SizedBox(width: 16), - Expanded(child: targetField), - const SizedBox(width: 16), - Expanded(child: maxField), - const SizedBox(width: 16), - ], - ) - else - Padding( - padding: const EdgeInsets.symmetric(horizontal: 16), - child: Column( - children: [ - minField, - const SizedBox(height: 16), - targetField, - const SizedBox(height: 16), - maxField, - ], - ), - ), - const SizedBox(height: 8), - ], - ), - ); - }); - } -} diff --git a/lib/modules/library/playlist_generate/seeds_multi_autocomplete.dart b/lib/modules/library/playlist_generate/seeds_multi_autocomplete.dart deleted file mode 100644 index 73c58deb..00000000 --- a/lib/modules/library/playlist_generate/seeds_multi_autocomplete.dart +++ /dev/null @@ -1,144 +0,0 @@ -import 'dart:async'; -import 'dart:math'; - -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:spotube/extensions/constrains.dart'; - -enum SelectedItemDisplayType { - wrap, - list, -} - -class SeedsMultiAutocomplete extends HookWidget { - final ValueNotifier> seeds; - - final FutureOr> Function(TextEditingValue textEditingValue) - fetchSeeds; - final Widget Function(T option, ValueChanged onSelected) - autocompleteOptionBuilder; - final Widget Function(T option) selectedSeedBuilder; - final String Function(T option) displayStringForOption; - - final InputDecoration? inputDecoration; - final bool enabled; - - final SelectedItemDisplayType selectedItemDisplayType; - - const SeedsMultiAutocomplete({ - super.key, - required this.seeds, - required this.fetchSeeds, - required this.autocompleteOptionBuilder, - required this.displayStringForOption, - required this.selectedSeedBuilder, - this.inputDecoration, - this.enabled = true, - this.selectedItemDisplayType = SelectedItemDisplayType.wrap, - }); - - @override - Widget build(BuildContext context) { - useValueListenable(seeds); - final theme = Theme.of(context); - final mediaQuery = MediaQuery.of(context); - final seedController = useTextEditingController(); - - final containerKey = useRef(GlobalKey()); - - final box = - containerKey.value.currentContext?.findRenderObject() as RenderBox?; - final position = box?.localToGlobal(Offset.zero); //this is global position - final containerYPos = position?.dy ?? 0; //th - final containerHeight = box?.size.height ?? 0; - - final listHeight = mediaQuery.size.height - - (containerYPos + containerHeight) - - // bottom player bar height - (mediaQuery.mdAndUp ? 80 : 0); - - return Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: CrossAxisAlignment.stretch, - children: [ - LayoutBuilder(builder: (context, constrains) { - return Container( - key: containerKey.value, - child: Autocomplete( - optionsBuilder: (textEditingValue) async { - if (textEditingValue.text.isEmpty) return []; - return fetchSeeds(textEditingValue); - }, - onSelected: (value) { - seeds.value = [...seeds.value, value]; - seedController.clear(); - }, - optionsViewBuilder: (context, onSelected, options) { - return Align( - alignment: Alignment.topLeft, - child: Container( - constraints: BoxConstraints( - maxWidth: constrains.maxWidth, - ), - height: max(listHeight, 0), - child: Card( - child: ListView.builder( - shrinkWrap: true, - itemCount: options.length, - itemBuilder: (context, index) { - final option = options.elementAt(index); - return autocompleteOptionBuilder(option, onSelected); - }, - ), - ), - ), - ); - }, - displayStringForOption: displayStringForOption, - fieldViewBuilder: ( - context, - textEditingController, - focusNode, - onFieldSubmitted, - ) { - return TextFormField( - controller: seedController, - onChanged: (value) => textEditingController.text = value, - focusNode: focusNode, - onFieldSubmitted: (_) => onFieldSubmitted(), - enabled: enabled, - decoration: inputDecoration, - ); - }, - ), - ); - }), - const SizedBox(height: 8), - switch (selectedItemDisplayType) { - SelectedItemDisplayType.wrap => Wrap( - spacing: 4, - runSpacing: 4, - children: seeds.value.map(selectedSeedBuilder).toList(), - ), - SelectedItemDisplayType.list => Card( - margin: EdgeInsets.zero, - child: Column( - children: [ - for (final seed in seeds.value) ...[ - selectedSeedBuilder(seed), - if (seeds.value.length > 1 && seed != seeds.value.last) - Divider( - color: theme.colorScheme.primaryContainer, - height: 1, - indent: 12, - endIndent: 12, - ), - ], - ], - ), - ), - }, - ], - ); - } -} diff --git a/lib/modules/library/playlist_generate/simple_track_tile.dart b/lib/modules/library/playlist_generate/simple_track_tile.dart deleted file mode 100644 index e6cc281f..00000000 --- a/lib/modules/library/playlist_generate/simple_track_tile.dart +++ /dev/null @@ -1,45 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/spotube_icons.dart'; - -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/image.dart'; - -class SimpleTrackTile extends HookWidget { - final Track track; - final VoidCallback? onDelete; - const SimpleTrackTile({ - super.key, - required this.track, - this.onDelete, - }); - - @override - Widget build(BuildContext context) { - return ListTile( - leading: ClipRRect( - borderRadius: BorderRadius.circular(8), - child: UniversalImage( - path: (track.album?.images).asUrlString( - placeholder: ImagePlaceholder.artist, - ), - height: 40, - width: 40, - ), - ), - horizontalTitleGap: 10, - contentPadding: const EdgeInsets.symmetric(horizontal: 8), - title: Text(track.name!), - trailing: onDelete == null - ? null - : IconButton( - icon: const Icon(SpotubeIcons.close), - onPressed: onDelete, - ), - subtitle: Text( - track.artists?.map((e) => e.name).join(", ") ?? track.album?.name ?? "", - ), - ); - } -} diff --git a/lib/modules/library/user_albums.dart b/lib/modules/library/user_albums.dart deleted file mode 100644 index 37fca7c0..00000000 --- a/lib/modules/library/user_albums.dart +++ /dev/null @@ -1,118 +0,0 @@ -import 'package:flutter/material.dart' hide Image; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:collection/collection.dart'; -import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:gap/gap.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotube/collections/fake.dart'; - -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/album/album_card.dart'; -import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; -import 'package:spotube/components/waypoint.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class UserAlbums extends HookConsumerWidget { - const UserAlbums({super.key}); - - @override - Widget build(BuildContext context, ref) { - final auth = ref.watch(authenticationProvider); - final albumsQuery = ref.watch(favoriteAlbumsProvider); - final albumsQueryNotifier = ref.watch(favoriteAlbumsProvider.notifier); - - final controller = useScrollController(); - - final searchText = useState(''); - - final albums = useMemoized(() { - if (searchText.value.isEmpty) { - return albumsQuery.asData?.value.items ?? []; - } - return albumsQuery.asData?.value.items - .map((e) => ( - weightedRatio(e.name!, searchText.value), - e, - )) - .sorted((a, b) => b.$1.compareTo(a.$1)) - .where((e) => e.$1 > 50) - .map((e) => e.$2) - .toList() ?? - []; - }, [albumsQuery.asData?.value, searchText.value]); - - if (auth.asData?.value == null) { - return const AnonymousFallback(); - } - - return SafeArea( - child: Scaffold( - body: RefreshIndicator( - onRefresh: () async { - ref.invalidate(favoriteAlbumsProvider); - }, - child: InterScrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - slivers: [ - SliverAppBar( - floating: true, - flexibleSpace: Padding( - padding: const EdgeInsets.symmetric(horizontal: 8.0), - child: SearchBar( - onChanged: (value) => searchText.value = value, - leading: const Icon(SpotubeIcons.filter), - hintText: context.l10n.filter_albums, - ), - ), - ), - const SliverGap(10), - SliverLayoutBuilder(builder: (context, constrains) { - return SliverGrid.builder( - itemCount: albums.isEmpty ? 6 : albums.length + 1, - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: 200, - mainAxisExtent: constrains.smAndDown ? 225 : 250, - crossAxisSpacing: 8, - mainAxisSpacing: 8, - ), - itemBuilder: (context, index) { - if (albums.isNotEmpty && index == albums.length) { - if (albumsQuery.asData?.value.hasMore != true) { - return const SizedBox.shrink(); - } - - return Waypoint( - controller: controller, - isGrid: true, - onTouchEdge: albumsQueryNotifier.fetchMore, - child: Skeletonizer( - enabled: true, - child: AlbumCard(FakeData.albumSimple), - ), - ); - } - - return Skeletonizer( - enabled: albumsQuery.isLoading, - child: AlbumCard( - albums.elementAtOrNull(index) ?? FakeData.albumSimple, - ), - ); - }, - ); - }), - ], - ), - ), - ), - ), - ); - } -} diff --git a/lib/modules/library/user_artists.dart b/lib/modules/library/user_artists.dart deleted file mode 100644 index 7968d91c..00000000 --- a/lib/modules/library/user_artists.dart +++ /dev/null @@ -1,124 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:collection/collection.dart'; -import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:gap/gap.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotube/collections/fake.dart'; - -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; -import 'package:spotube/modules/artist/artist_card.dart'; -import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/components/waypoint.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class UserArtists extends HookConsumerWidget { - const UserArtists({super.key}); - - @override - Widget build(BuildContext context, ref) { - final auth = ref.watch(authenticationProvider); - - final artistQuery = ref.watch(followedArtistsProvider); - final artistQueryNotifier = ref.watch(followedArtistsProvider.notifier); - - final searchText = useState(''); - - final filteredArtists = useMemoized(() { - final artists = artistQuery.asData?.value.items ?? []; - - if (searchText.value.isEmpty) { - return artists.toList(); - } - return artists - .map((e) => ( - weightedRatio(e.name!, searchText.value), - e, - )) - .sorted((a, b) => b.$1.compareTo(a.$1)) - .where((e) => e.$1 > 50) - .map((e) => e.$2) - .toList(); - }, [artistQuery.asData?.value.items, searchText.value]); - - final controller = useScrollController(); - - if (auth.asData?.value == null) { - return const AnonymousFallback(); - } - - return SafeArea( - child: Scaffold( - body: RefreshIndicator( - onRefresh: () async { - ref.invalidate(followedArtistsProvider); - }, - child: InterScrollbar( - controller: controller, - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 8.0), - child: CustomScrollView( - controller: controller, - slivers: [ - SliverAppBar( - floating: true, - flexibleSpace: SearchBar( - onChanged: (value) => searchText.value = value, - leading: const Icon(SpotubeIcons.filter), - hintText: context.l10n.filter_artist, - ), - ), - const SliverGap(10), - SliverLayoutBuilder(builder: (context, constrains) { - return SliverGrid.builder( - itemCount: filteredArtists.isEmpty - ? 6 - : filteredArtists.length + 1, - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: 200, - mainAxisExtent: constrains.smAndDown ? 225 : 250, - crossAxisSpacing: 8, - mainAxisSpacing: 8, - ), - itemBuilder: (context, index) { - if (filteredArtists.isNotEmpty && - index == filteredArtists.length) { - if (artistQuery.asData?.value.hasMore != true) { - return const SizedBox.shrink(); - } - - return Waypoint( - controller: controller, - isGrid: true, - onTouchEdge: artistQueryNotifier.fetchMore, - child: Skeletonizer( - enabled: true, - child: ArtistCard(FakeData.artist), - ), - ); - } - - return Skeletonizer( - enabled: artistQuery.isLoading, - child: ArtistCard( - filteredArtists.elementAtOrNull(index) ?? - FakeData.artist, - ), - ); - }, - ); - }), - ], - ), - ), - ), - ), - ), - ); - } -} diff --git a/lib/modules/library/user_downloads/download_item.dart b/lib/modules/library/user_downloads/download_item.dart index c4bd7bce..b1cd9f62 100644 --- a/lib/modules/library/user_downloads/download_item.dart +++ b/lib/modules/library/user_downloads/download_item.dart @@ -1,52 +1,28 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/links/artist_link.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/track/track.dart'; +import 'package:spotube/components/ui/button_tile.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/download_manager_provider.dart'; -import 'package:spotube/services/download_manager/download_status.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:spotube/utils/service_utils.dart'; class DownloadItem extends HookConsumerWidget { - final Track track; + final DownloadTask task; const DownloadItem({ super.key, - required this.track, + required this.task, }); @override Widget build(BuildContext context, ref) { - final downloadManager = ref.watch(downloadManagerProvider); + final downloadManager = ref.watch(downloadManagerProvider.notifier); - final taskStatus = useState(null); - - useEffect(() { - if (track is! SourcedTrack) return null; - final notifier = downloadManager.getStatusNotifier(track as SourcedTrack); - - taskStatus.value = notifier?.value; - - void listener() { - taskStatus.value = notifier?.value; - } - - notifier?.addListener(listener); - - return () { - notifier?.removeListener(listener); - }; - }, [track]); - - final isQueryingSourceInfo = - taskStatus.value == null || track is! SourcedTrack; - - return ListTile( + return ButtonTile( + style: ButtonVariance.ghost, leading: Padding( padding: const EdgeInsets.symmetric(horizontal: 5), child: ClipRRect( @@ -54,101 +30,72 @@ class DownloadItem extends HookConsumerWidget { child: UniversalImage( height: 40, width: 40, - path: (track.album?.images).asUrlString( + path: task.track.album.images.asUrlString( placeholder: ImagePlaceholder.albumArt, ), ), ), ), - title: Text(track.name ?? ''), + title: Text(task.track.name), subtitle: ArtistLink( - artists: track.artists ?? [], + artists: task.track.artists, mainAxisAlignment: WrapAlignment.start, - onOverflowArtistClick: () => ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": track.id!, - }, - ), + onOverflowArtistClick: () { + context.navigateTo(TrackRoute(trackId: task.track.id)); + }, ), - trailing: isQueryingSourceInfo - ? Text( - context.l10n.querying_info, - style: Theme.of(context).textTheme.labelMedium, - ) - : switch (taskStatus.value!) { - DownloadStatus.downloading => HookBuilder(builder: (context) { - final taskProgress = useListenable(useMemoized( - () => downloadManager - .getProgressNotifier(track as SourcedTrack), - [track], - )); - return SizedBox( - width: 140, - child: Row( - children: [ - CircularProgressIndicator( - value: taskProgress?.value ?? 0, - ), - const SizedBox(width: 10), - IconButton( - icon: const Icon(SpotubeIcons.pause), - onPressed: () { - downloadManager.pause(track as SourcedTrack); - }), - const SizedBox(width: 10), - IconButton( - icon: const Icon(SpotubeIcons.close), - onPressed: () { - downloadManager.cancel(track as SourcedTrack); - }), - ], - ), - ); - }), - DownloadStatus.paused => Row( - mainAxisSize: MainAxisSize.min, - children: [ - IconButton( - icon: const Icon(SpotubeIcons.play), - onPressed: () { - downloadManager.resume(track as SourcedTrack); - }), - const SizedBox(width: 10), - IconButton( - icon: const Icon(SpotubeIcons.close), - onPressed: () { - downloadManager.cancel(track as SourcedTrack); - }) - ], - ), - DownloadStatus.failed || DownloadStatus.canceled => SizedBox( - width: 100, - child: Row( + trailing: switch (task.status) { + DownloadStatus.downloading => HookBuilder(builder: (context) { + return StreamBuilder( + stream: task.downloadedBytesStream, + builder: (context, asyncSnapshot) { + final progress = + task.totalSizeBytes == null || task.totalSizeBytes == 0 + ? 0 + : (asyncSnapshot.data ?? 0) / task.totalSizeBytes!; + + return Row( children: [ - Icon( - SpotubeIcons.error, - color: Colors.red[400], + CircularProgressIndicator( + value: progress.toDouble(), ), const SizedBox(width: 10), - IconButton( - icon: const Icon(SpotubeIcons.refresh), - onPressed: () { - downloadManager.retry(track as SourcedTrack); - }, - ), + const SizedBox(width: 10), + IconButton.ghost( + icon: const Icon(SpotubeIcons.close), + onPressed: () { + downloadManager.cancel(task.track); + }), ], - ), + ); + }); + }), + DownloadStatus.failed || DownloadStatus.canceled => SizedBox( + width: 100, + child: Row( + children: [ + Icon( + SpotubeIcons.error, + color: Colors.red[400], ), - DownloadStatus.completed => - Icon(SpotubeIcons.done, color: Colors.green[400]), - DownloadStatus.queued => IconButton( - icon: const Icon(SpotubeIcons.close), + const SizedBox(width: 10), + IconButton.ghost( + icon: const Icon(SpotubeIcons.refresh), onPressed: () { - downloadManager.removeFromQueue(track as SourcedTrack); - }), - }, + downloadManager.retry(task.track); + }, + ), + ], + ), + ), + DownloadStatus.completed => + Icon(SpotubeIcons.done, color: Colors.green[400]), + DownloadStatus.queued => IconButton.ghost( + icon: const Icon(SpotubeIcons.close), + onPressed: () { + downloadManager.cancel(task.track); + }), + }, ); } } diff --git a/lib/modules/library/user_playlists.dart b/lib/modules/library/user_playlists.dart deleted file mode 100644 index 577f9655..00000000 --- a/lib/modules/library/user_playlists.dart +++ /dev/null @@ -1,163 +0,0 @@ -import 'package:flutter/material.dart' hide Image; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:collection/collection.dart'; -import 'package:gap/gap.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; - -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/playlist/playlist_create_dialog.dart'; -import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; -import 'package:spotube/modules/playlist/playlist_card.dart'; -import 'package:spotube/components/waypoint.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/library/playlist_generate/playlist_generate.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; -import 'package:spotube/utils/platform.dart'; -import 'package:spotube/utils/service_utils.dart'; - -class UserPlaylists extends HookConsumerWidget { - const UserPlaylists({super.key}); - - @override - Widget build(BuildContext context, ref) { - final searchText = useState(''); - - final auth = ref.watch(authenticationProvider); - - final playlistsQuery = ref.watch(favoritePlaylistsProvider); - final playlistsQueryNotifier = - ref.watch(favoritePlaylistsProvider.notifier); - - final likedTracksPlaylist = useMemoized( - () => PlaylistSimple() - ..name = context.l10n.liked_tracks - ..description = context.l10n.liked_tracks_description - ..type = "playlist" - ..collaborative = false - ..public = false - ..id = "user-liked-tracks" - ..images = [ - Image() - ..height = 300 - ..width = 300 - ..url = "assets/liked-tracks.jpg" - ], - [context.l10n], - ); - - final playlists = useMemoized( - () { - if (searchText.value.isEmpty) { - return [ - likedTracksPlaylist, - ...?playlistsQuery.asData?.value.items, - ]; - } - return [ - likedTracksPlaylist, - ...?playlistsQuery.asData?.value.items, - ] - .map((e) => (weightedRatio(e.name!, searchText.value), e)) - .sorted((a, b) => b.$1.compareTo(a.$1)) - .where((e) => e.$1 > 50) - .map((e) => e.$2) - .toList(); - }, - [playlistsQuery, searchText.value], - ); - - final controller = useScrollController(); - - if (auth.asData?.value == null) { - return const AnonymousFallback(); - } - - return RefreshIndicator( - onRefresh: () async { - ref.invalidate(favoritePlaylistsProvider); - }, - child: SafeArea( - child: InterScrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - slivers: [ - SliverAppBar( - floating: true, - flexibleSpace: Padding( - padding: const EdgeInsets.symmetric(horizontal: 8), - child: SearchBar( - onChanged: (value) => searchText.value = value, - hintText: context.l10n.filter_playlists, - leading: const Icon(SpotubeIcons.filter), - ), - ), - bottom: PreferredSize( - preferredSize: - Size.fromHeight(kIsDesktop ? 35 : kToolbarHeight), - child: Row( - children: [ - const Gap(10), - const PlaylistCreateDialogButton(), - const Gap(10), - ElevatedButton.icon( - icon: const Icon(SpotubeIcons.magic), - label: Text(context.l10n.generate_playlist), - onPressed: () { - ServiceUtils.pushNamed( - context, PlaylistGeneratorPage.name); - }, - ), - const Gap(10), - ], - ), - ), - ), - const SliverGap(10), - SliverLayoutBuilder(builder: (context, constrains) { - return SliverGrid.builder( - itemCount: playlists.isEmpty ? 6 : playlists.length + 1, - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: 200, - mainAxisExtent: constrains.smAndDown ? 225 : 250, - crossAxisSpacing: 8, - mainAxisSpacing: 8, - ), - itemBuilder: (context, index) { - if (playlists.isNotEmpty && index == playlists.length) { - if (playlistsQuery.asData?.value.hasMore != true) { - return const SizedBox.shrink(); - } - - return Waypoint( - controller: controller, - isGrid: true, - onTouchEdge: playlistsQueryNotifier.fetchMore, - child: Skeletonizer( - enabled: true, - child: PlaylistCard(FakeData.playlistSimple), - ), - ); - } - - return PlaylistCard( - playlists.elementAtOrNull(index) ?? - FakeData.playlistSimple, - ); - }, - ); - }) - ], - ), - ), - ), - ); - } -} diff --git a/lib/modules/lyrics/zoom_controls.dart b/lib/modules/lyrics/zoom_controls.dart index 73beb4ae..b4eeb9d6 100644 --- a/lib/modules/lyrics/zoom_controls.dart +++ b/lib/modules/lyrics/zoom_controls.dart @@ -1,5 +1,6 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/spotube_icons.dart'; @@ -32,7 +33,7 @@ class ZoomControls extends HookWidget { @override Widget build(BuildContext context) { final actions = [ - IconButton( + IconButton.ghost( icon: decreaseIcon, onPressed: () { if (value == min) return; @@ -40,7 +41,7 @@ class ZoomControls extends HookWidget { }, ), Text("$value$unit"), - IconButton( + IconButton.ghost( icon: increaseIcon, onPressed: () { if (value == max) return; @@ -50,27 +51,28 @@ class ZoomControls extends HookWidget { ]; return Container( - decoration: BoxDecoration( - color: Theme.of(context).cardColor.withOpacity(0.7), - borderRadius: BorderRadius.circular(10), - ), constraints: BoxConstraints( maxHeight: direction == Axis.horizontal ? 50 : 200, maxWidth: direction == Axis.vertical ? 50 : double.infinity, ), margin: const EdgeInsets.all(8), - child: direction == Axis.horizontal - ? Row( - mainAxisAlignment: MainAxisAlignment.end, - mainAxisSize: MainAxisSize.min, - children: actions, - ) - : Column( - mainAxisAlignment: MainAxisAlignment.end, - mainAxisSize: MainAxisSize.min, - verticalDirection: VerticalDirection.up, - children: actions, - ), + child: SurfaceCard( + surfaceBlur: context.theme.surfaceBlur, + surfaceOpacity: context.theme.surfaceOpacity, + padding: EdgeInsets.zero, + child: direction == Axis.horizontal + ? Row( + mainAxisAlignment: MainAxisAlignment.end, + mainAxisSize: MainAxisSize.min, + children: actions, + ) + : Column( + mainAxisAlignment: MainAxisAlignment.end, + mainAxisSize: MainAxisSize.min, + verticalDirection: VerticalDirection.up, + children: actions, + ), + ), ); } } diff --git a/lib/modules/metadata_plugins/installed_plugin.dart b/lib/modules/metadata_plugins/installed_plugin.dart new file mode 100644 index 00000000..7abda5ec --- /dev/null +++ b/lib/modules/metadata_plugins/installed_plugin.dart @@ -0,0 +1,405 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/markdown/markdown.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/metadata_plugins/plugin_update_available_dialog.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/core/support.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/updater/update_checker.dart'; +import 'package:url_launcher/url_launcher.dart'; + +final validAbilities = { + PluginAbilities.metadata: ("Metadata", SpotubeIcons.album), + PluginAbilities.audioSource: ("Audio Source", SpotubeIcons.music), +}; + +class MetadataInstalledPluginItem extends HookConsumerWidget { + final PluginConfiguration plugin; + final bool isDefaultMetadata; + final bool isDefaultAudioSource; + const MetadataInstalledPluginItem({ + super.key, + required this.plugin, + required this.isDefaultMetadata, + required this.isDefaultAudioSource, + }); + + @override + Widget build(BuildContext context, ref) { + final mediaQuery = MediaQuery.sizeOf(context); + + final metadataPlugin = ref.watch(metadataPluginProvider); + final audioSourcePlugin = ref.watch(audioSourcePluginProvider); + final pluginSnapshot = switch ((isDefaultMetadata, isDefaultAudioSource)) { + (true, _) => metadataPlugin, + (false, true) => audioSourcePlugin, + _ => null, + }; + + final pluginsNotifier = ref.watch(metadataPluginsProvider.notifier); + + final requiresAuth = (isDefaultMetadata || isDefaultAudioSource) && + plugin.abilities.contains(PluginAbilities.authentication); + final supportsScrobbling = isDefaultMetadata && + plugin.abilities.contains(PluginAbilities.scrobbling); + + final isMetadataAuthenticatedSnapshot = + ref.watch(metadataPluginAuthenticatedProvider); + final isAudioSourceAuthenticatedSnapshot = + ref.watch(audioSourcePluginAuthenticatedProvider); + final isAuthenticated = (isDefaultMetadata && + isMetadataAuthenticatedSnapshot.asData?.value == true) || + (isDefaultAudioSource && + isAudioSourceAuthenticatedSnapshot.asData?.value == true); + + final metadataUpdateAvailable = + ref.watch(metadataPluginUpdateCheckerProvider); + final audioSourceUpdateAvailable = + ref.watch(audioSourcePluginUpdateCheckerProvider); + final updateAvailable = switch ((isDefaultMetadata, isDefaultAudioSource)) { + (true, _) => metadataUpdateAvailable, + (false, true) => audioSourceUpdateAvailable, + _ => null, + }; + final hasUpdate = updateAvailable?.asData?.value != null; + + return Card( + child: Column( + crossAxisAlignment: CrossAxisAlignment.stretch, + mainAxisSize: MainAxisSize.min, + spacing: 12, + children: [ + FutureBuilder( + future: pluginsNotifier.getLogoPath(plugin), + builder: (context, snapshot) { + final repoUrl = plugin.repository != null + ? Uri.tryParse(plugin.repository!) + : null; + final repoOwner = repoUrl?.pathSegments.firstOrNull; + + final isOfficial = + repoUrl?.host == "github.com" && repoOwner == "KRTirtho"; + + return Basic( + leading: snapshot.hasData + ? ClipRRect( + borderRadius: BorderRadius.circular(8), + child: Image.file( + snapshot.data!, + width: 36, + height: 36, + ), + ) + : Container( + height: 36, + width: 36, + alignment: Alignment.center, + decoration: BoxDecoration( + color: context.theme.colorScheme.secondary, + borderRadius: BorderRadius.circular(8), + ), + child: const Icon(SpotubeIcons.plugin), + ), + title: Text(plugin.name), + subtitle: Column( + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 8, + children: [ + Text(plugin.description), + Wrap( + spacing: 8, + runSpacing: 8, + children: [ + for (final ability in plugin.abilities) + if (validAbilities.keys.contains(ability)) + SecondaryBadge( + leading: Icon(validAbilities[ability]!.$2), + child: Text(validAbilities[ability]!.$1), + ), + ], + ), + if (repoUrl != null) + Wrap( + spacing: 8, + runSpacing: 8, + children: [ + if (isOfficial) + PrimaryBadge( + leading: const Icon(SpotubeIcons.done), + child: Text(context.l10n.official), + ) + else ...[ + Text(context.l10n.author_name(plugin.author)), + Container( + padding: const EdgeInsets.symmetric( + horizontal: 6, + vertical: 2, + ), + decoration: BoxDecoration( + color: Colors.blue, + borderRadius: BorderRadius.circular(6), + ), + child: Row( + mainAxisSize: MainAxisSize.min, + spacing: 4, + children: [ + const Icon(SpotubeIcons.warning, size: 14), + Text( + context.l10n.third_party, + style: const TextStyle(color: Colors.white), + ).xSmall + ], + ), + ), + ], + SecondaryBadge( + leading: const Icon(SpotubeIcons.connect), + child: Text(repoUrl.host), + onPressed: () { + launchUrl(repoUrl); + }, + ), + SecondaryBadge( + child: Padding( + padding: const EdgeInsets.all(1), + child: Text( + "${context.l10n.version}: ${plugin.version}", + ), + ), + ), + ], + ) + ], + ), + trailing: IconButton.ghost( + onPressed: () async { + await pluginsNotifier.removePlugin(plugin); + }, + icon: const Icon( + SpotubeIcons.trash, + color: Colors.red, + ), + ), + ); + }, + ), + if ((requiresAuth && !isAuthenticated) || + hasUpdate || + supportsScrobbling) + Container( + decoration: BoxDecoration( + color: context.theme.colorScheme.secondary, + borderRadius: BorderRadius.circular(8), + ), + padding: const EdgeInsets.all(12), + child: Column( + spacing: 12, + children: [ + if (requiresAuth && !isAuthenticated) + Row( + spacing: 8, + children: [ + const Icon(SpotubeIcons.warning, color: Colors.yellow), + Text(context.l10n.plugin_requires_authentication), + ], + ), + if (hasUpdate) + SizedBox( + width: double.infinity, + child: Basic( + leading: const Icon(SpotubeIcons.update), + title: Text(context.l10n.update_available), + subtitle: Text( + updateAvailable!.asData!.value!.version, + ), + trailing: Button.primary( + onPressed: () { + showDialog( + context: context, + builder: (context) => + MetadataPluginUpdateAvailableDialog( + plugin: plugin, + update: updateAvailable.asData!.value!, + ), + ); + }, + child: Text(context.l10n.update), + ), + ), + ), + if (supportsScrobbling) + SizedBox( + width: double.infinity, + child: Basic( + leading: const Icon(SpotubeIcons.info), + title: Text(context.l10n.supports_scrobbling), + subtitle: Text(context.l10n.plugin_scrobbling_info), + ), + ) + ], + ), + ), + Wrap( + spacing: 8, + runSpacing: 8, + alignment: WrapAlignment.spaceBetween, + children: [ + Wrap( + spacing: 8, + runSpacing: 8, + children: [ + if (plugin.abilities.contains(PluginAbilities.metadata)) + Button.secondary( + enabled: !isDefaultMetadata, + onPressed: () async { + await pluginsNotifier.setDefaultMetadataPlugin(plugin); + }, + child: Text( + isDefaultMetadata + ? context.l10n.default_metadata_source + : context.l10n.set_default_metadata_source, + ), + ), + if (plugin.abilities.contains(PluginAbilities.audioSource)) + Button.secondary( + enabled: !isDefaultAudioSource, + onPressed: () async { + await pluginsNotifier + .setDefaultAudioSourcePlugin(plugin); + }, + child: Text( + isDefaultAudioSource + ? context.l10n.default_audio_source + : context.l10n.set_default_audio_source, + ), + ), + ], + ), + Row( + mainAxisSize: + mediaQuery.smAndUp ? MainAxisSize.min : MainAxisSize.max, + mainAxisAlignment: MainAxisAlignment.end, + spacing: 8, + children: [ + if (isDefaultMetadata || isDefaultAudioSource) + Consumer(builder: (context, ref, _) { + final metadataSupportTextSnapshot = + ref.watch(metadataPluginSupportTextProvider); + final audioSourceSupportTextSnapshot = + ref.watch(audioSourcePluginSupportTextProvider); + + final supportTextSnapshot = + switch ((isDefaultMetadata, isDefaultAudioSource)) { + (true, _) => metadataSupportTextSnapshot, + (false, true) => audioSourceSupportTextSnapshot, + _ => null, + }; + + if ((supportTextSnapshot?.hasValue ?? false) && + supportTextSnapshot?.value == null) { + return const SizedBox.shrink(); + } + + final bgColor = + context.theme.brightness == Brightness.dark + ? const Color.fromARGB(255, 255, 145, 175) + : Colors.pink[600]; + final textColor = + context.theme.brightness == Brightness.dark + ? Colors.pink[700] + : Colors.pink[50]; + + final mediaQuery = MediaQuery.sizeOf(context); + + return Button( + style: ButtonVariance.secondary.copyWith( + decoration: (context, states, value) { + return value.copyWithIfBoxDecoration( + color: bgColor, + ); + }, + textStyle: (context, states, value) { + return value.copyWith( + color: textColor, + ); + }, + iconTheme: (context, states, value) { + return value.copyWith( + color: textColor, + ); + }, + ), + leading: const Icon(SpotubeIcons.heartFilled), + child: Text(context.l10n.support), + onPressed: () { + showDialog( + context: context, + builder: (context) { + return AlertDialog( + title: Text( + context.l10n.support_plugin_development), + content: ConstrainedBox( + constraints: BoxConstraints( + maxHeight: mediaQuery.height * 0.8, + maxWidth: 720, + ), + child: SizedBox( + width: double.infinity, + child: SingleChildScrollView( + child: AppMarkdown( + data: supportTextSnapshot + ?.asData?.value ?? + "", + ), + ), + ), + ), + actions: [ + Button.secondary( + onPressed: () { + Navigator.of(context).pop(); + }, + child: Text(context.l10n.close), + ), + ], + ); + }, + ); + }, + ); + }), + if ((isDefaultMetadata || isDefaultAudioSource) && + requiresAuth && + !isAuthenticated) + Button.primary( + onPressed: () async { + await pluginSnapshot?.asData?.value?.auth + .authenticate(); + }, + leading: const Icon(SpotubeIcons.login), + child: Text(context.l10n.login), + ) + else if ((isDefaultMetadata || isDefaultAudioSource) && + requiresAuth && + isAuthenticated) + Button.destructive( + onPressed: () async { + await pluginSnapshot?.asData?.value?.auth.logout(); + }, + leading: const Icon(SpotubeIcons.logout), + child: Text(context.l10n.logout), + ), + ], + ) + ], + ) + ], + ), + ); + } +} diff --git a/lib/modules/metadata_plugins/plugin_repository.dart b/lib/modules/metadata_plugins/plugin_repository.dart new file mode 100644 index 00000000..9bd71f0a --- /dev/null +++ b/lib/modules/metadata_plugins/plugin_repository.dart @@ -0,0 +1,237 @@ +import 'package:flutter/gestures.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/markdown/markdown.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:url_launcher/url_launcher_string.dart'; +import 'package:change_case/change_case.dart'; + +final validTopics = { + "spotube-metadata-plugin": ("Metadata", SpotubeIcons.album), + "spotube-audio-source-plugin": ("Audio Source", SpotubeIcons.music), +}; + +class MetadataPluginRepositoryItem extends HookConsumerWidget { + final MetadataPluginRepository pluginRepo; + const MetadataPluginRepositoryItem({ + super.key, + required this.pluginRepo, + }); + + @override + Widget build(BuildContext context, ref) { + final pluginsNotifier = ref.watch(metadataPluginsProvider.notifier); + final host = useMemoized( + () => Uri.parse(pluginRepo.repoUrl).host, + [pluginRepo.repoUrl], + ); + final isInstalling = useState(false); + + return Card( + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.stretch, + spacing: 8, + children: [ + Basic( + title: Text( + pluginRepo.name.startsWith("spotube-plugin") + ? pluginRepo.name + .replaceFirst("spotube-plugin-", "") + .trim() + .toCapitalCase() + : pluginRepo.name.toCapitalCase(), + ), + subtitle: Text(pluginRepo.description), + trailing: Button.primary( + enabled: !isInstalling.value, + onPressed: () async { + try { + isInstalling.value = true; + final pluginConfig = await pluginsNotifier + .downloadAndCachePlugin(pluginRepo.repoUrl); + + if (!context.mounted) return; + final isOfficialPlugin = pluginRepo.owner == "KRTirtho"; + + final isAllowed = isOfficialPlugin + ? true + : await showDialog( + context: context, + builder: (context) { + final pluginAbilities = pluginConfig.apis + .map((e) => + context.l10n.can_access_name_api(e.name)) + .join("\n\n"); + + return AlertDialog( + title: Text( + context.l10n.do_you_want_to_install_this_plugin, + ), + content: Column( + mainAxisSize: MainAxisSize.min, + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text(context.l10n.third_party_plugin_warning), + const Gap(8), + FutureBuilder( + future: pluginsNotifier + .getLogoPath(pluginConfig), + builder: (context, snapshot) { + return Basic( + leading: snapshot.hasData + ? Image.file( + snapshot.data!, + width: 36, + height: 36, + ) + : Container( + height: 36, + width: 36, + alignment: Alignment.center, + decoration: BoxDecoration( + color: context.theme + .colorScheme.secondary, + borderRadius: + BorderRadius.circular(8), + ), + child: const Icon( + SpotubeIcons.plugin), + ), + title: Text(pluginConfig.name), + subtitle: + Text(pluginConfig.description), + ); + }, + ), + const Gap(8), + AppMarkdown( + data: + "**${context.l10n.author}**: ${pluginConfig.author}\n\n" + "**${context.l10n.repository}**: [${pluginConfig.repository ?? 'N/A'}](${pluginConfig.repository})\n\n\n\n" + "${context.l10n.this_plugin_can_do_following}:\n\n" + "$pluginAbilities", + ), + ], + ), + actions: [ + Button.secondary( + onPressed: () { + Navigator.of(context).pop(false); + }, + child: Text(context.l10n.decline), + ), + Button.primary( + onPressed: () { + Navigator.of(context).pop(true); + }, + child: Text(context.l10n.accept), + ), + ], + ); + }, + ); + + if (isAllowed != true) return; + await pluginsNotifier.addPlugin(pluginConfig); + } finally { + if (context.mounted) { + isInstalling.value = false; + } + } + }, + leading: isInstalling.value + ? SizedBox.square( + dimension: 20, + child: CircularProgressIndicator( + color: context.theme.colorScheme.primaryForeground, + ), + ) + : const Icon(SpotubeIcons.add), + child: Text(context.l10n.install), + ), + ), + if (pluginRepo.owner != "KRTirtho") + Text.rich( + TextSpan( + children: [ + TextSpan(text: context.l10n.source), + TextSpan( + text: pluginRepo.repoUrl.replaceAll("https://", ""), + style: const TextStyle( + color: Colors.blue, + decoration: TextDecoration.underline, + ), + recognizer: TapGestureRecognizer() + ..onTap = () async { + launchUrlString(pluginRepo.repoUrl); + }, + ), + ], + ), + style: context.theme.typography.xSmall, + ), + Wrap( + spacing: 8, + runSpacing: 8, + children: [ + if (pluginRepo.owner == "KRTirtho") + PrimaryBadge( + leading: const Icon(SpotubeIcons.done), + child: Text(context.l10n.official), + ) + else ...[ + Text( + context.l10n.author_name(pluginRepo.owner), + style: context.theme.typography.xSmall, + ), + Container( + padding: const EdgeInsets.symmetric( + horizontal: 6, + vertical: 2, + ), + decoration: BoxDecoration( + color: Colors.blue, + borderRadius: BorderRadius.circular(6), + ), + child: Row( + mainAxisSize: MainAxisSize.min, + spacing: 4, + children: [ + const Icon(SpotubeIcons.warning, size: 14), + Text( + context.l10n.third_party, + style: const TextStyle(color: Colors.white), + ).xSmall + ], + ), + ), + ], + for (final topic in pluginRepo.topics) + if (validTopics.keys.contains(topic)) + SecondaryBadge( + leading: Icon(validTopics[topic]!.$2), + child: Text(validTopics[topic]!.$1), + ), + SecondaryBadge( + leading: host == "github.com" + ? const Icon(SpotubeIcons.github) + : null, + child: Text(host), + onPressed: () { + launchUrlString(pluginRepo.repoUrl); + }, + ), + ], + ), + ], + ), + ); + } +} diff --git a/lib/modules/metadata_plugins/plugin_update_available_dialog.dart b/lib/modules/metadata_plugins/plugin_update_available_dialog.dart new file mode 100644 index 00000000..d16a0a35 --- /dev/null +++ b/lib/modules/metadata_plugins/plugin_update_available_dialog.dart @@ -0,0 +1,93 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/markdown/markdown.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; + +class MetadataPluginUpdateAvailableDialog extends HookConsumerWidget { + final PluginConfiguration plugin; + final PluginUpdateAvailable update; + const MetadataPluginUpdateAvailableDialog({ + super.key, + required this.plugin, + required this.update, + }); + + @override + Widget build(BuildContext context, ref) { + final isUpdating = useState(false); + + final showErrorSnackbar = useCallback( + (BuildContext context, String message) { + showToast( + context: context, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + leading: const Icon(SpotubeIcons.error, color: Colors.red), + title: Text(message), + leadingAlignment: Alignment.center, + trailing: IconButton.ghost( + size: ButtonSize.small, + icon: const Icon(SpotubeIcons.close), + onPressed: () { + overlay.close(); + }, + ), + ), + ); + }); + }, + [], + ); + + return AlertDialog( + title: const Text('Plugin update available'), + content: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 8, + children: [ + Text('${plugin.name} (${update.version}) available.'), + if (update.changelog != null && update.changelog!.isNotEmpty) + AppMarkdown( + data: '### Changelog: \n\n${update.changelog}', + ), + ], + ), + actions: [ + SecondaryButton( + onPressed: () { + Navigator.of(context).pop(); + }, + child: const Text('Dismiss'), + ), + PrimaryButton( + enabled: !isUpdating.value, + onPressed: () async { + isUpdating.value = true; + try { + await ref + .read(metadataPluginsProvider.notifier) + .updatePlugin(plugin, update); + if (context.mounted) { + Navigator.of(context).pop(); + } + } catch (e) { + if (context.mounted) { + showErrorSnackbar(context, e.toString()); + } + } finally { + if (context.mounted) { + isUpdating.value = false; + } + } + }, + child: const Text('Update'), + ), + ], + ); + } +} diff --git a/lib/modules/player/player.dart b/lib/modules/player/player.dart index 925afadc..5ea690e0 100644 --- a/lib/modules/player/player.dart +++ b/lib/modules/player/player.dart @@ -1,39 +1,29 @@ +import 'package:auto_route/auto_route.dart'; import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:sliding_up_panel/sliding_up_panel.dart'; import 'package:spotube/collections/assets.gen.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/framework/app_pop_scope.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/player/player_actions.dart'; import 'package:spotube/modules/player/player_controls.dart'; -import 'package:spotube/modules/player/player_queue.dart'; import 'package:spotube/modules/player/volume_slider.dart'; -import 'package:spotube/components/animated_gradient.dart'; import 'package:spotube/components/dialogs/track_details_dialog.dart'; import 'package:spotube/components/links/artist_link.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/panels/sliding_up_panel.dart'; -import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/hooks/utils/use_custom_status_bar_color.dart'; -import 'package:spotube/hooks/utils/use_palette_color.dart'; -import 'package:spotube/models/local_track.dart'; -import 'package:spotube/pages/lyrics/lyrics.dart'; -import 'package:spotube/pages/track/track.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; +import 'package:spotube/modules/root/spotube_navigation_bar.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/server/active_sourced_track.dart'; +import 'package:spotube/provider/metadata_plugin/audio_source/quality_label.dart'; +import 'package:spotube/provider/server/active_track_sources.dart'; import 'package:spotube/provider/volume_provider.dart'; -import 'package:spotube/services/sourced_track/sources/youtube.dart'; -import 'package:spotube/utils/service_utils.dart'; - -import 'package:url_launcher/url_launcher_string.dart'; class PlayerView extends HookConsumerWidget { final PanelController panelController; @@ -47,13 +37,23 @@ class PlayerView extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { final theme = Theme.of(context); - final auth = ref.watch(authenticationProvider); - final sourcedCurrentTrack = ref.watch(activeSourcedTrackProvider); + final sourcedCurrentTrack = ref.watch(activeTrackSourcesProvider); final currentActiveTrack = ref.watch(audioPlayerProvider.select((s) => s.activeTrack)); - final currentTrack = sourcedCurrentTrack ?? currentActiveTrack; - final isLocalTrack = currentTrack is LocalTrack; - final mediaQuery = MediaQuery.of(context); + final currentActiveTrackSource = sourcedCurrentTrack.asData?.value?.source; + final isLocalTrack = currentActiveTrack is SpotubeLocalTrackObject; + final mediaQuery = MediaQuery.sizeOf(context); + final qualityLabel = ref.watch(audioSourceQualityLabelProvider); + + final shouldHide = useState(true); + + ref.listen(navigationPanelHeight, (_, height) { + shouldHide.value = height.ceil() == 50; + }); + + if (shouldHide.value) { + return const SizedBox(); + } useEffect(() { if (mediaQuery.lgAndUp) { @@ -65,21 +65,12 @@ class PlayerView extends HookConsumerWidget { }, [mediaQuery.lgAndUp]); String albumArt = useMemoized( - () => (currentTrack?.album?.images).asUrlString( + () => (currentActiveTrack?.album.images).asUrlString( placeholder: ImagePlaceholder.albumArt, ), - [currentTrack?.album?.images], + [currentActiveTrack?.album.images], ); - final palette = usePaletteGenerator(albumArt); - final titleTextColor = palette.dominantColor?.titleTextColor; - final bodyTextColor = palette.dominantColor?.bodyTextColor; - - final bgColor = palette.dominantColor?.color ?? theme.colorScheme.primary; - - final GlobalKey scaffoldKey = - useMemoized(() => GlobalKey(), []); - useEffect(() { for (final renderView in WidgetsBinding.instance.renderViews) { renderView.automaticSystemUiAdjustment = false; @@ -90,329 +81,190 @@ class PlayerView extends HookConsumerWidget { renderView.automaticSystemUiAdjustment = true; } }; - }, [panelController.isPanelOpen]); - - useCustomStatusBarColor( - bgColor, - panelController.isPanelOpen, - noSetBGColor: true, - automaticSystemUiAdjustment: false, - ); - - final topPadding = MediaQueryData.fromView(View.of(context)).padding.top; + }, [panelController.isAttached && panelController.isPanelOpen]); return AppPopScope( - canPop: context.canPop(), + canPop: false, onPopInvoked: (didPop) async { await panelController.close(); }, - child: IconTheme( - data: theme.iconTheme.copyWith(color: bodyTextColor), - child: AnimateGradient( - animateAlignments: true, - primaryBegin: Alignment.topLeft, - primaryEnd: Alignment.bottomLeft, - secondaryBegin: Alignment.bottomRight, - secondaryEnd: Alignment.topRight, - duration: const Duration(seconds: 15), - primaryColors: [ - palette.dominantColor?.color ?? theme.colorScheme.primary, - palette.mutedColor?.color ?? theme.colorScheme.secondary, - ], - secondaryColors: [ - (palette.darkVibrantColor ?? palette.lightVibrantColor)?.color ?? - theme.colorScheme.primaryContainer, - (palette.darkMutedColor ?? palette.lightMutedColor)?.color ?? - theme.colorScheme.secondaryContainer, - ], - child: Scaffold( - key: scaffoldKey, - backgroundColor: Colors.transparent, - appBar: PreferredSize( - preferredSize: Size.fromHeight( - kToolbarHeight + topPadding, - ), - child: ForceDraggableWidget( - child: Padding( - padding: EdgeInsets.only(top: topPadding), - child: PageWindowTitleBar( - backgroundColor: Colors.transparent, - foregroundColor: titleTextColor, - toolbarOpacity: 1, - leading: IconButton( - icon: const Icon(SpotubeIcons.angleDown, size: 18), - onPressed: panelController.close, - ), - actions: [ - if (currentTrack is YoutubeSourcedTrack) - TextButton.icon( - icon: Assets.logos.songlinkTransparent.image( - width: 20, - height: 20, - color: bodyTextColor, - ), - label: Text(context.l10n.song_link), - style: TextButton.styleFrom( - foregroundColor: bodyTextColor, - padding: const EdgeInsets.symmetric(horizontal: 10), - ), - onPressed: () { - final url = - "https://song.link/s/${currentTrack.id}"; - - launchUrlString(url); - }, - ), - IconButton( - icon: const Icon(SpotubeIcons.info, size: 18), - tooltip: context.l10n.details, - style: IconButton.styleFrom( - foregroundColor: bodyTextColor, - ), - onPressed: currentTrack == null + child: SurfaceCard( + borderWidth: 0, + surfaceOpacity: 0.9, + padding: EdgeInsets.zero, + child: Scaffold( + backgroundColor: Colors.transparent, + headers: [ + SafeArea( + bottom: false, + child: TitleBar( + surfaceOpacity: 0, + surfaceBlur: 0, + leading: [ + IconButton.ghost( + size: const ButtonSize(1.2), + icon: const Icon(SpotubeIcons.angleDown), + onPressed: panelController.close, + ) + ], + trailing: [ + if (!isLocalTrack) + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.details), + ).call, + child: IconButton.ghost( + size: const ButtonSize(1.2), + icon: const Icon(SpotubeIcons.info), + onPressed: currentActiveTrackSource == null ? null : () { showDialog( context: context, builder: (context) { return TrackDetailsDialog( - track: currentTrack, + track: currentActiveTrack + as SpotubeFullTrackObject, ); }); }, - ) - ], - ), - ), + ), + ) + ], ), ), - extendBodyBehindAppBar: true, - body: SingleChildScrollView( - controller: scrollController, - child: Container( - alignment: Alignment.center, - width: double.infinity, - child: ConstrainedBox( - constraints: const BoxConstraints(maxWidth: 580), - child: SafeArea( - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Column( - children: [ - ForceDraggableWidget( - child: Container( - margin: const EdgeInsets.all(8), - constraints: const BoxConstraints( - maxHeight: 300, maxWidth: 300), - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(20), - boxShadow: const [ - BoxShadow( - color: Colors.black26, - spreadRadius: 2, - blurRadius: 10, - offset: Offset(0, 0), - ), - ], - ), - child: ClipRRect( - borderRadius: BorderRadius.circular(20), - child: UniversalImage( - path: albumArt, - placeholder: Assets.albumPlaceholder.path, - fit: BoxFit.cover, - ), - ), - ), - ), - const SizedBox(height: 60), - Container( - padding: const EdgeInsets.symmetric(horizontal: 16), - alignment: Alignment.centerLeft, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - AutoSizeText( - currentTrack?.name ?? - context.l10n.not_playing, - style: TextStyle( - color: titleTextColor, - fontSize: 22, - ), - maxFontSize: 22, - maxLines: 1, - textAlign: TextAlign.start, - ), - if (isLocalTrack) - Text( - currentTrack.artists?.asString() ?? "", - style: theme.textTheme.bodyMedium!.copyWith( - fontWeight: FontWeight.bold, - color: bodyTextColor, - ), - ) - else - ArtistLink( - artists: currentTrack?.artists ?? [], - textStyle: - theme.textTheme.bodyMedium!.copyWith( - fontWeight: FontWeight.bold, - color: bodyTextColor, - ), - onRouteChange: (route) { - panelController.close(); - GoRouter.of(context).push(route); - }, - onOverflowArtistClick: () => - ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": currentTrack!.id!, - }, - ), - ), - ], - ), - ), - const SizedBox(height: 10), - PlayerControls(palette: palette), - const SizedBox(height: 25), - const PlayerActions( - mainAxisAlignment: MainAxisAlignment.spaceEvenly, - showQueue: false, - ), - const SizedBox(height: 10), - Row( - mainAxisAlignment: MainAxisAlignment.spaceEvenly, - children: [ - const SizedBox(width: 10), - Expanded( - child: OutlinedButton.icon( - icon: const Icon(SpotubeIcons.queue), - label: Text(context.l10n.queue), - style: OutlinedButton.styleFrom( - foregroundColor: bodyTextColor, - side: BorderSide( - color: bodyTextColor ?? Colors.white, - ), - ), - onPressed: currentTrack != null - ? () { - showModalBottomSheet( - context: context, - isDismissible: true, - enableDrag: true, - isScrollControlled: true, - backgroundColor: Colors.black12, - barrierColor: Colors.black12, - shape: RoundedRectangleBorder( - borderRadius: - BorderRadius.circular(10), - ), - constraints: BoxConstraints( - maxHeight: - MediaQuery.of(context) - .size - .height * - .7, - ), - builder: (context) => Consumer( - builder: (context, ref, _) { - final playlist = ref.watch( - audioPlayerProvider, - ); - final playlistNotifier = ref - .read(audioPlayerProvider - .notifier); - return PlayerQueue - .fromAudioPlayerNotifier( - floating: false, - playlist: playlist, - notifier: playlistNotifier, - ); - }, - ), - ); - } - : null), - ), - if (auth.asData?.value != null) - const SizedBox(width: 10), - if (auth.asData?.value != null) - Expanded( - child: OutlinedButton.icon( - label: Text(context.l10n.lyrics), - icon: const Icon(SpotubeIcons.music), - style: OutlinedButton.styleFrom( - foregroundColor: bodyTextColor, - side: BorderSide( - color: bodyTextColor ?? Colors.white, - ), - ), - onPressed: () { - showModalBottomSheet( - context: context, - isDismissible: true, - enableDrag: true, - isScrollControlled: true, - backgroundColor: Colors.black38, - barrierColor: Colors.black12, - shape: const RoundedRectangleBorder( - borderRadius: BorderRadius.only( - topLeft: Radius.circular(20), - topRight: Radius.circular(20), - ), - ), - constraints: BoxConstraints( - maxHeight: MediaQuery.of(context) - .size - .height * - 0.8, - ), - builder: (context) => - const LyricsPage(isModal: true), - ); - }, - ), - ), - const SizedBox(width: 10), - ], - ), - const SizedBox(height: 25), - SliderTheme( - data: theme.sliderTheme.copyWith( - activeTrackColor: titleTextColor, - inactiveTrackColor: bodyTextColor, - thumbColor: titleTextColor, - overlayColor: titleTextColor?.withOpacity(0.2), - trackHeight: 2, - thumbShape: const RoundSliderThumbShape( - enabledThumbRadius: 8, - ), - ), - child: Padding( - padding: - const EdgeInsets.symmetric(horizontal: 16), - child: Consumer(builder: (context, ref, _) { - final volume = ref.watch(volumeProvider); - return VolumeSlider( - fullWidth: true, - value: volume, - onChanged: (value) { - ref - .read(volumeProvider.notifier) - .setVolume(value); - }, - ); - }), - ), - ), - ], + ], + child: SingleChildScrollView( + controller: scrollController, + child: Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + children: [ + Container( + margin: const EdgeInsets.all(8), + constraints: + const BoxConstraints(maxHeight: 300, maxWidth: 300), + decoration: BoxDecoration( + borderRadius: BorderRadius.circular(20), + boxShadow: [ + BoxShadow( + color: Colors.black.withAlpha(100), + spreadRadius: 2, + blurRadius: 10, + offset: Offset.zero, + ), + ], + ), + child: ClipRRect( + borderRadius: BorderRadius.circular(20), + child: UniversalImage( + path: albumArt, + placeholder: Assets.images.albumPlaceholder.path, + fit: BoxFit.cover, ), ), ), - ), + const SizedBox(height: 60), + Container( + padding: const EdgeInsets.symmetric(horizontal: 16), + alignment: Alignment.centerLeft, + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + AutoSizeText( + currentActiveTrack?.name ?? context.l10n.not_playing, + style: const TextStyle(fontSize: 22), + maxFontSize: 22, + maxLines: 1, + textAlign: TextAlign.start, + ), + if (isLocalTrack) + Text( + currentActiveTrack.artists.asString(), + style: theme.typography.normal + .copyWith(fontWeight: FontWeight.bold), + ) + else + ArtistLink( + artists: currentActiveTrack?.artists ?? [], + textStyle: theme.typography.normal + .copyWith(fontWeight: FontWeight.bold), + onRouteChange: (route) { + panelController.close(); + context.router.navigateNamed(route); + }, + onOverflowArtistClick: () => context.navigateTo( + TrackRoute( + trackId: currentActiveTrack!.id, + ), + ), + ), + ], + ), + ), + const SizedBox(height: 10), + const PlayerControls(), + const SizedBox(height: 25), + const PlayerActions( + mainAxisAlignment: MainAxisAlignment.spaceEvenly, + showQueue: false, + ), + const SizedBox(height: 10), + Row( + mainAxisAlignment: MainAxisAlignment.spaceEvenly, + children: [ + const SizedBox(width: 10), + Expanded( + child: OutlineButton( + leading: const Icon(SpotubeIcons.queue), + child: Text(context.l10n.queue), + onPressed: () { + context.pushRoute(const PlayerQueueRoute()); + }, + ), + ), + const SizedBox(width: 10), + Expanded( + child: OutlineButton( + leading: const Icon(SpotubeIcons.music), + child: Text(context.l10n.lyrics), + onPressed: () { + context.pushRoute(const PlayerLyricsRoute()); + }, + ), + ), + const SizedBox(width: 10), + ], + ), + const SizedBox(height: 25), + Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: Consumer(builder: (context, ref, _) { + final volume = ref.watch(volumeProvider); + return VolumeSlider( + fullWidth: true, + value: volume, + onChanged: (value) { + ref.read(volumeProvider.notifier).setVolume(value); + }, + ); + }), + ), + const Gap(25), + OutlineBadge( + style: const ButtonStyle.outline( + size: ButtonSize.normal, + density: ButtonDensity.dense, + shape: ButtonShape.rectangle, + ).copyWith( + textStyle: (context, states, value) { + return value.copyWith(fontWeight: FontWeight.w500); + }, + ), + leading: const Icon(SpotubeIcons.lightningOutlined), + child: Text(qualityLabel), + ) + ], ), ), ), diff --git a/lib/modules/player/player_actions.dart b/lib/modules/player/player_actions.dart index a47c992d..9f8639ec 100644 --- a/lib/modules/player/player_actions.dart +++ b/lib/modules/player/player_actions.dart @@ -1,19 +1,24 @@ +import 'package:auto_route/auto_route.dart'; import 'package:flutter/foundation.dart'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/player/player_queue.dart'; import 'package:spotube/modules/player/sibling_tracks_sheet.dart'; import 'package:spotube/components/adaptive/adaptive_pop_sheet_list.dart'; import 'package:spotube/components/heart_button/heart_button.dart'; -import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/extensions/duration.dart'; -import 'package:spotube/models/local_track.dart'; import 'package:spotube/provider/download_manager_provider.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; import 'package:spotube/provider/sleep_timer_provider.dart'; class PlayerActions extends HookConsumerWidget { @@ -33,30 +38,35 @@ class PlayerActions extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { final playlist = ref.watch(audioPlayerProvider); - final isLocalTrack = playlist.activeTrack is LocalTrack; + final isLocalTrack = playlist.activeTrack is SpotubeLocalTrackObject; ref.watch(downloadManagerProvider); final downloader = ref.watch(downloadManagerProvider.notifier); final isInQueue = useMemoized(() { - if (playlist.activeTrack == null) return false; - return downloader.isActive(playlist.activeTrack!); + if (playlist.activeTrack is! SpotubeFullTrackObject) return false; + final downloadTask = + downloader.getTaskByTrackId(playlist.activeTrack!.id); + return const [ + DownloadStatus.queued, + DownloadStatus.downloading, + ].contains(downloadTask?.status); }, [ playlist.activeTrack, downloader, ]); - final localTracks = [] /* ref.watch(localTracksProvider).value */; - final auth = ref.watch(authenticationProvider); + final localTracks = ref.watch(localTracksProvider).value; + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); final sleepTimer = ref.watch(sleepTimerProvider); final sleepTimerNotifier = ref.watch(sleepTimerProvider.notifier); final isDownloaded = useMemoized(() { - return localTracks.any( - (element) => - element.name == playlist.activeTrack?.name && - element.album?.name == playlist.activeTrack?.album?.name && - element.artists?.asString() == - playlist.activeTrack?.artists?.asString(), - ) == + return localTracks?.values.expand((e) => e).any( + (element) => + element.name == playlist.activeTrack?.name && + element.album.name == playlist.activeTrack?.album.name && + element.artists.asString() == + playlist.activeTrack?.artists.asString(), + ) == true; }, [localTracks, playlist.activeTrack]); @@ -76,38 +86,73 @@ class PlayerActions extends HookConsumerWidget { mainAxisAlignment: mainAxisAlignment, children: [ if (showQueue) - IconButton( - icon: const Icon(SpotubeIcons.queue), - tooltip: context.l10n.queue, - onPressed: playlist.activeTrack != null - ? () { - Scaffold.of(context).openEndDrawer(); - } - : null, + Tooltip( + tooltip: TooltipContainer(child: Text(context.l10n.queue)).call, + child: IconButton.ghost( + icon: const Icon(SpotubeIcons.queue), + enabled: playlist.activeTrack != null, + onPressed: () { + openDrawer( + context: context, + position: OverlayPosition.right, + transformBackdrop: false, + draggable: false, + surfaceBlur: context.theme.surfaceBlur, + surfaceOpacity: 0.7, + builder: (context) { + return Container( + constraints: const BoxConstraints(maxWidth: 800), + child: Consumer( + builder: (context, ref, _) { + final playlist = ref.watch(audioPlayerProvider); + final playlistNotifier = + ref.read(audioPlayerProvider.notifier); + + return PlayerQueue.fromAudioPlayerNotifier( + floating: true, + playlist: playlist, + notifier: playlistNotifier, + ); + }, + ), + ); + }, + ); + }, + ), ), if (!isLocalTrack) - IconButton( - icon: const Icon(SpotubeIcons.alternativeRoute), - tooltip: context.l10n.alternative_track_sources, - onPressed: playlist.activeTrack != null - ? () { - showModalBottomSheet( - context: context, - isDismissible: true, - enableDrag: true, - isScrollControlled: true, - backgroundColor: Colors.black12, - barrierColor: Colors.black12, - elevation: 0, - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(10), - ), - builder: (context) { - return SiblingTracksSheet(floating: floatingQueue); - }, - ); - } - : null, + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.alternative_track_sources), + ).call, + child: IconButton.ghost( + enabled: playlist.activeTrack != null, + icon: const Icon(SpotubeIcons.alternativeRoute), + onPressed: () { + final screenSize = MediaQuery.sizeOf(context); + if (screenSize.mdAndUp) { + showPopover( + alignment: Alignment.bottomCenter, + context: context, + builder: (context) { + return SurfaceCard( + padding: EdgeInsets.zero, + child: ConstrainedBox( + constraints: const BoxConstraints( + maxHeight: 600, + maxWidth: 500, + ), + child: SiblingTracksSheet(floating: floatingQueue), + ), + ); + }, + ); + } else { + context.pushRoute(const PlayerTrackSourcesRoute()); + } + }, + ), ), if (!kIsWeb && !isLocalTrack) if (isInQueue) @@ -115,24 +160,30 @@ class PlayerActions extends HookConsumerWidget { height: 20, width: 20, child: CircularProgressIndicator( - strokeWidth: 2, + size: 2, ), ) else - IconButton( - tooltip: context.l10n.download_track, - icon: Icon( - isDownloaded ? SpotubeIcons.done : SpotubeIcons.download, + Tooltip( + tooltip: + TooltipContainer(child: Text(context.l10n.download_track)) + .call, + child: IconButton.ghost( + icon: Icon( + isDownloaded ? SpotubeIcons.done : SpotubeIcons.download, + ), + onPressed: playlist.activeTrack != null + ? () => downloader.addToQueue( + playlist.activeTrack! as SpotubeFullTrackObject) + : null, ), - onPressed: playlist.activeTrack != null - ? () => downloader.addToQueue(playlist.activeTrack!) - : null, ), if (playlist.activeTrack != null && !isLocalTrack && - auth.asData?.value != null) + authenticated.asData?.value == true) TrackHeartButton(track: playlist.activeTrack!), - AdaptivePopSheetList( + AdaptivePopSheetList( + tooltip: context.l10n.sleep_timer, offset: Offset(0, -50 * (sleepTimerEntries.values.length + 2)), headings: [ Text(context.l10n.sleep_timer), @@ -148,26 +199,49 @@ class PlayerActions extends HookConsumerWidget { sleepTimerNotifier.setSleepTimer(value); } }, - children: [ + items: (context) => [ for (final entry in sleepTimerEntries.entries) - PopSheetEntry( + AdaptiveMenuButton( value: entry.value, enabled: sleepTimer != entry.value, - title: Text(entry.key), + child: Text(entry.key), ), - PopSheetEntry( - title: Text( - customHoursEnabled - ? context.l10n.custom_hours - : sleepTimer.format(abbreviated: true), - ), - // only enabled when there's no preset timers selected + AdaptiveMenuButton( enabled: customHoursEnabled, - onTap: () async { + onPressed: (context) async { final currentTime = TimeOfDay.now(); - final time = await showTimePicker( + final time = await showDialog( context: context, - initialTime: currentTime, + builder: (context) => HookBuilder(builder: (context) { + final timeRef = useRef(null); + return AlertDialog( + trailing: IconButton.ghost( + size: ButtonSize.xSmall, + icon: const Icon(SpotubeIcons.close), + onPressed: () { + Navigator.of(context).pop(); + }, + ), + title: Text( + ShadcnLocalizations.of(context).placeholderTimePicker, + ), + content: TimePickerDialog( + use24HourFormat: false, + initialValue: TimeOfDay.fromDateTime( + DateTime.now().add(sleepTimer ?? Duration.zero), + ), + onChanged: (value) => timeRef.value = value, + ), + actions: [ + Button.primary( + onPressed: () { + Navigator.of(context).pop(timeRef.value); + }, + child: Text(context.l10n.save), + ), + ], + ); + }), ); if (time != null) { @@ -179,12 +253,19 @@ class PlayerActions extends HookConsumerWidget { ); } }, + child: Text( + customHoursEnabled + ? context.l10n.custom_hours + : sleepTimer.format(abbreviated: true), + ), ), - PopSheetEntry( + AdaptiveMenuButton( value: Duration.zero, enabled: sleepTimer != Duration.zero && sleepTimer != null, - textColor: Colors.green, - title: Text(context.l10n.cancel), + child: Text( + context.l10n.cancel, + style: const TextStyle(color: Colors.green), + ), ), ], ), diff --git a/lib/modules/player/player_controls.dart b/lib/modules/player/player_controls.dart index 12288a3d..3da36bf8 100644 --- a/lib/modules/player/player_controls.dart +++ b/lib/modules/player/player_controls.dart @@ -1,18 +1,20 @@ -import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:media_kit/media_kit.dart'; import 'package:palette_generator/palette_generator.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/collections/intents.dart'; +import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/extensions/duration.dart'; import 'package:spotube/modules/player/use_progress.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/audio_player/querying_track_info.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/utils/platform.dart'; class PlayerControls extends HookConsumerWidget { final PaletteGenerator? palette; @@ -47,43 +49,8 @@ class PlayerControls extends HookConsumerWidget { useStream(audioPlayer.playingStream).data ?? audioPlayer.isPlaying; final theme = Theme.of(context); - final isDominantColorDark = ThemeData.estimateBrightnessForColor( - palette?.dominantColor?.color ?? theme.colorScheme.primary, - ) == - Brightness.dark; - - final dominantColor = isDominantColorDark - ? palette?.mutedColor ?? palette?.dominantColor - : palette?.dominantColor; - - final sliderColor = - palette?.dominantColor?.titleTextColor ?? theme.colorScheme.primary; - - final buttonStyle = IconButton.styleFrom( - backgroundColor: dominantColor?.color.withOpacity(0.2) ?? - theme.colorScheme.surface.withOpacity(0.4), - minimumSize: const Size(28, 28), - ); - - final activeButtonStyle = IconButton.styleFrom( - backgroundColor: - dominantColor?.titleTextColor ?? theme.colorScheme.primaryContainer, - foregroundColor: - dominantColor?.color ?? theme.colorScheme.onPrimaryContainer, - minimumSize: const Size(28, 28), - ); - - final accentColor = palette?.lightVibrantColor ?? - palette?.darkVibrantColor ?? - dominantColor; - - final resumePauseStyle = IconButton.styleFrom( - backgroundColor: accentColor?.color ?? theme.colorScheme.primary, - foregroundColor: - accentColor?.titleTextColor ?? theme.colorScheme.onPrimary, - padding: EdgeInsets.all(compact ? 10 : 12), - iconSize: compact ? 18 : 24, - ); + final buttonSize = + kIsMobile ? const ButtonSize(1.5) : const ButtonSize(1.2); return GestureDetector( behavior: HitTestBehavior.translucent, @@ -103,6 +70,8 @@ class PlayerControls extends HookConsumerWidget { if (!compact) HookBuilder( builder: (context) { + final mediaQuery = MediaQuery.sizeOf(context); + final ( :bufferProgress, :duration, @@ -122,45 +91,47 @@ class PlayerControls extends HookConsumerWidget { return Column( children: [ Tooltip( - message: context.l10n.slide_to_seek, - child: Slider( - // cannot divide by zero - // there's an edge case for value being bigger - // than total duration. Keeping it resolved - value: progress.value.toDouble(), - secondaryTrackValue: bufferProgress, - onChanged: isFetchingActiveTrack - ? null - : (v) { - progress.value = v; - }, - onChangeEnd: (value) async { - await audioPlayer.seek( - Duration( - seconds: (value * duration.inSeconds).toInt(), - ), - ); - }, - activeColor: sliderColor, - secondaryActiveColor: sliderColor.withOpacity(0.2), - inactiveColor: sliderColor.withOpacity(0.15), + tooltip: TooltipContainer( + child: Text(context.l10n.slide_to_seek), + ).call, + child: SizedBox( + width: mediaQuery.xlAndUp ? 600 : 500, + child: Slider( + hintValue: SliderValue.single(bufferProgress), + value: + SliderValue.single(progress.value.toDouble()), + onChanged: isFetchingActiveTrack + ? null + : (v) { + progress.value = v.value; + }, + onChangeEnd: (value) async { + await audioPlayer.seek( + Duration( + seconds: (value.value * duration.inSeconds) + .toInt(), + ), + ); + }, + ), ), ), Padding( padding: const EdgeInsets.symmetric( horizontal: 8.0, ), - child: DefaultTextStyle( - style: theme.textTheme.bodySmall!.copyWith( - color: palette?.dominantColor?.bodyTextColor, - ), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text(position.toHumanReadableString()), - Text(duration.toHumanReadableString()), - ], - ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + position.toHumanReadableString(), + style: theme.typography.xSmall, + ), + Text( + duration.toHumanReadableString(), + style: theme.typography.xSmall, + ), + ], ), ), ], @@ -173,92 +144,126 @@ class PlayerControls extends HookConsumerWidget { Consumer(builder: (context, ref, _) { final shuffled = ref .watch(audioPlayerProvider.select((s) => s.shuffled)); - return IconButton( - tooltip: shuffled - ? context.l10n.unshuffle_playlist - : context.l10n.shuffle_playlist, - icon: const Icon(SpotubeIcons.shuffle), - style: shuffled ? activeButtonStyle : buttonStyle, - onPressed: isFetchingActiveTrack - ? null - : () { - if (shuffled) { - audioPlayer.setShuffle(false); - } else { - audioPlayer.setShuffle(true); - } - }, + return Tooltip( + tooltip: TooltipContainer( + child: Text( + shuffled + ? context.l10n.unshuffle_playlist + : context.l10n.shuffle_playlist, + ), + ).call, + child: IconButton( + size: buttonSize, + icon: Icon( + SpotubeIcons.shuffle, + color: shuffled ? theme.colorScheme.primary : null, + size: 22, + ), + variance: shuffled + ? ButtonVariance.secondary + : ButtonVariance.ghost, + onPressed: isFetchingActiveTrack + ? null + : () { + if (shuffled) { + audioPlayer.setShuffle(false); + } else { + audioPlayer.setShuffle(true); + } + }, + ), ); }), - IconButton( - tooltip: context.l10n.previous_track, - icon: const Icon(SpotubeIcons.skipBack), - style: buttonStyle, - onPressed: isFetchingActiveTrack - ? null - : audioPlayer.skipToPrevious, + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.previous_track), + ).call, + child: IconButton.ghost( + size: buttonSize, + enabled: !isFetchingActiveTrack, + icon: const Icon(SpotubeIcons.skipBack), + onPressed: audioPlayer.skipToPrevious, + ), ), - IconButton( - tooltip: playing - ? context.l10n.pause_playback - : context.l10n.resume_playback, - icon: isFetchingActiveTrack - ? SizedBox( - height: 20, - width: 20, - child: CircularProgressIndicator( - color: accentColor?.titleTextColor ?? - theme.colorScheme.onPrimary, + Tooltip( + tooltip: TooltipContainer( + child: Text( + playing + ? context.l10n.pause_playback + : context.l10n.resume_playback, + ), + ).call, + child: IconButton.primary( + size: buttonSize, + shape: ButtonShape.circle, + icon: isFetchingActiveTrack + ? const SizedBox( + height: 20, + width: 20, + child: CircularProgressIndicator(), + ) + : Icon( + playing ? SpotubeIcons.pause : SpotubeIcons.play, ), - ) - : Icon( - playing ? SpotubeIcons.pause : SpotubeIcons.play, - ), - style: resumePauseStyle, - onPressed: isFetchingActiveTrack - ? null - : Actions.handler( - context, - PlayPauseIntent(ref), - ), + onPressed: isFetchingActiveTrack + ? null + : Actions.handler( + context, + PlayPauseIntent(ref), + ), + ), ), - IconButton( - tooltip: context.l10n.next_track, - icon: const Icon(SpotubeIcons.skipForward), - style: buttonStyle, - onPressed: - isFetchingActiveTrack ? null : audioPlayer.skipToNext, + Tooltip( + tooltip: + TooltipContainer(child: Text(context.l10n.next_track)) + .call, + child: IconButton.ghost( + size: buttonSize, + icon: const Icon(SpotubeIcons.skipForward), + onPressed: + isFetchingActiveTrack ? null : audioPlayer.skipToNext, + ), ), Consumer(builder: (context, ref, _) { final loopMode = ref .watch(audioPlayerProvider.select((s) => s.loopMode)); - return IconButton( - tooltip: loopMode == PlaylistMode.single - ? context.l10n.loop_track - : loopMode == PlaylistMode.loop - ? context.l10n.repeat_playlist + return Tooltip( + tooltip: TooltipContainer( + child: Text( + loopMode == PlaylistMode.single + ? context.l10n.loop_track + : loopMode == PlaylistMode.loop + ? context.l10n.repeat_playlist + : "", + ), + ).call, + child: IconButton( + size: buttonSize, + icon: Icon( + loopMode == PlaylistMode.single + ? SpotubeIcons.repeatOne + : SpotubeIcons.repeat, + color: loopMode != PlaylistMode.none + ? theme.colorScheme.primary : null, - icon: Icon( - loopMode == PlaylistMode.single - ? SpotubeIcons.repeatOne - : SpotubeIcons.repeat, + ), + variance: loopMode == PlaylistMode.single || + loopMode == PlaylistMode.loop + ? ButtonVariance.secondary + : ButtonVariance.ghost, + onPressed: isFetchingActiveTrack + ? null + : () async { + await audioPlayer.setLoopMode( + switch (loopMode) { + PlaylistMode.loop => PlaylistMode.single, + PlaylistMode.single => PlaylistMode.none, + PlaylistMode.none => PlaylistMode.loop, + }, + ); + }, ), - style: loopMode == PlaylistMode.single || - loopMode == PlaylistMode.loop - ? activeButtonStyle - : buttonStyle, - onPressed: isFetchingActiveTrack - ? null - : () async { - await audioPlayer.setLoopMode( - switch (loopMode) { - PlaylistMode.loop => PlaylistMode.single, - PlaylistMode.single => PlaylistMode.none, - PlaylistMode.none => PlaylistMode.loop, - }, - ); - }, ); }), ], diff --git a/lib/modules/player/player_overlay.dart b/lib/modules/player/player_overlay.dart index 2322bcba..3c3ff373 100644 --- a/lib/modules/player/player_overlay.dart +++ b/lib/modules/player/player_overlay.dart @@ -1,19 +1,15 @@ -import 'dart:ui'; - -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:sliding_up_panel/sliding_up_panel.dart'; +import 'package:spotube/modules/player/player_overlay_collapsed.dart'; -import 'package:spotube/modules/player/player_track_details.dart'; import 'package:spotube/modules/root/spotube_navigation_bar.dart'; -import 'package:spotube/components/panels/sliding_up_panel.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/collections/intents.dart'; -import 'package:spotube/modules/player/use_progress.dart'; import 'package:spotube/modules/player/player.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/audio_player/querying_track_info.dart'; -import 'package:spotube/services/audio_player/audio_player.dart'; + +final playerOverlayControllerProvider = StateProvider((ref) { + return PanelController(); +}); class PlayerOverlay extends HookConsumerWidget { final String albumArt; @@ -25,180 +21,34 @@ class PlayerOverlay extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final isFetchingActiveTrack = ref.watch(queryingTrackInfoProvider); final playlist = ref.watch(audioPlayerProvider); final canShow = playlist.activeTrack != null; - final playing = - useStream(audioPlayer.playingStream).data ?? audioPlayer.isPlaying; + final screenSize = MediaQuery.sizeOf(context); - final theme = Theme.of(context); - final textColor = theme.colorScheme.primary; - - const radius = BorderRadius.only( - topLeft: Radius.circular(10), - topRight: Radius.circular(10), - ); - - final mediaQuery = MediaQuery.of(context); - - final panelController = useMemoized(() => PanelController(), []); - final scrollController = useScrollController(); - - useEffect(() { - return () { - panelController.dispose(); - }; - }, []); + final panelController = ref.watch(playerOverlayControllerProvider); return SlidingUpPanel( - maxHeight: mediaQuery.size.height, + maxHeight: screenSize.height, backdropEnabled: false, - minHeight: canShow ? 53 : 0, + minHeight: canShow ? 63 : 0, onPanelSlide: (position) { final invertedPosition = 1 - position; ref.read(navigationPanelHeight.notifier).state = 50 * invertedPosition; }, controller: panelController, - collapsed: ClipRRect( - borderRadius: radius, - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 15, sigmaY: 15), - child: AnimatedContainer( - duration: const Duration(milliseconds: 250), - width: mediaQuery.size.width, - decoration: BoxDecoration( - color: theme.colorScheme.secondaryContainer.withOpacity(.8), - borderRadius: radius, - ), - child: AnimatedOpacity( - duration: const Duration(milliseconds: 250), - opacity: canShow ? 1 : 0, - child: Material( - type: MaterialType.transparency, - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - HookBuilder( - builder: (context) { - final progress = useProgress(ref); - // animated - return TweenAnimationBuilder( - duration: const Duration(milliseconds: 250), - tween: Tween( - begin: 0, - end: progress.progressStatic, - ), - builder: (context, value, child) { - return LinearProgressIndicator( - value: value, - minHeight: 2, - backgroundColor: Colors.transparent, - valueColor: AlwaysStoppedAnimation( - theme.colorScheme.primary, - ), - ); - }, - ); - }, - ), - Expanded( - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Expanded( - child: GestureDetector( - onTap: () { - panelController.open(); - }, - child: Container( - width: double.infinity, - color: Colors.transparent, - child: PlayerTrackDetails( - track: playlist.activeTrack, - color: textColor, - ), - ), - ), - ), - Row( - children: [ - IconButton( - icon: Icon( - SpotubeIcons.skipBack, - color: textColor, - ), - onPressed: isFetchingActiveTrack - ? null - : audioPlayer.skipToPrevious, - ), - Consumer( - builder: (context, ref, _) { - return IconButton( - icon: isFetchingActiveTrack - ? const SizedBox( - height: 20, - width: 20, - child: CircularProgressIndicator(), - ) - : Icon( - playing - ? SpotubeIcons.pause - : SpotubeIcons.play, - color: textColor, - ), - onPressed: Actions.handler( - context, - PlayPauseIntent(ref), - ), - ); - }, - ), - IconButton( - icon: Icon( - SpotubeIcons.skipForward, - color: textColor, - ), - onPressed: isFetchingActiveTrack - ? null - : audioPlayer.skipToNext, - ), - ], - ), - ], - ), - ), - ], - ), - ), - ), - ), - ), + color: Colors.transparent, + parallaxEnabled: true, + renderPanelSheet: false, + header: SizedBox( + height: 63, + width: screenSize.width, + child: PlayerOverlayCollapsedSection(panelController: panelController), + ), + panelBuilder: (scrollController) => PlayerView( + panelController: panelController, + scrollController: scrollController, ), - scrollController: scrollController, - panelBuilder: (position) { - // this is the reason we're getting an update - final navigationHeight = ref.watch(navigationPanelHeight); - - if (navigationHeight == 50) return const SizedBox(); - - return IgnorePointer( - ignoring: !panelController.isPanelOpen, - child: AnimatedContainer( - clipBehavior: Clip.antiAlias, - duration: const Duration(milliseconds: 250), - decoration: navigationHeight == 0 - ? const BoxDecoration(borderRadius: BorderRadius.zero) - : const BoxDecoration(borderRadius: radius), - child: IgnoreDraggableWidget( - child: PlayerView( - panelController: panelController, - scrollController: scrollController, - ), - ), - ), - ); - }, ); } } diff --git a/lib/modules/player/player_overlay_collapsed.dart b/lib/modules/player/player_overlay_collapsed.dart new file mode 100644 index 00000000..d0961ade --- /dev/null +++ b/lib/modules/player/player_overlay_collapsed.dart @@ -0,0 +1,117 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:sliding_up_panel/sliding_up_panel.dart'; +import 'package:spotube/collections/intents.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/modules/player/player_track_details.dart'; +import 'package:spotube/modules/root/spotube_navigation_bar.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/audio_player/querying_track_info.dart'; +import 'package:spotube/services/audio_player/audio_player.dart'; + +class PlayerOverlayCollapsedSection extends HookConsumerWidget { + final PanelController panelController; + const PlayerOverlayCollapsedSection({ + super.key, + required this.panelController, + }); + + @override + Widget build(BuildContext context, ref) { + final playlist = ref.watch(audioPlayerProvider); + final canShow = playlist.activeTrack != null; + + final isFetchingActiveTrack = ref.watch(queryingTrackInfoProvider); + final playing = + useStream(audioPlayer.playingStream).data ?? audioPlayer.isPlaying; + + final theme = Theme.of(context); + + final shouldShow = useState(true); + + ref.listen(navigationPanelHeight, (_, height) { + shouldShow.value = height.ceil() == 50; + }); + + return AnimatedSwitcher( + duration: const Duration(milliseconds: 250), + child: canShow && shouldShow.value + ? Padding( + padding: const EdgeInsets.all(5), + child: SurfaceCard( + surfaceBlur: theme.surfaceBlur, + surfaceOpacity: theme.surfaceOpacity, + padding: EdgeInsets.zero, + borderRadius: theme.borderRadiusLg, + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Expanded( + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Expanded( + child: GestureDetector( + onTap: () { + panelController.open(); + }, + child: Container( + width: double.infinity, + color: Colors.transparent, + child: PlayerTrackDetails( + track: playlist.activeTrack, + color: theme.colorScheme.foreground, + ), + ), + ), + ), + Row( + children: [ + IconButton.ghost( + icon: const Icon(SpotubeIcons.skipBack), + onPressed: isFetchingActiveTrack + ? null + : audioPlayer.skipToPrevious, + ), + Consumer( + builder: (context, ref, _) { + return IconButton.ghost( + icon: isFetchingActiveTrack + ? const SizedBox( + height: 20, + width: 20, + child: CircularProgressIndicator(), + ) + : Icon( + playing + ? SpotubeIcons.pause + : SpotubeIcons.play, + ), + onPressed: Actions.handler( + context, + PlayPauseIntent(ref), + ), + ); + }, + ), + IconButton.ghost( + icon: const Icon(SpotubeIcons.skipForward), + onPressed: isFetchingActiveTrack + ? null + : audioPlayer.skipToNext, + ), + const Gap(5), + ], + ), + ], + ), + ), + ], + ), + ), + ) + : const SizedBox.shrink(), + ); + } +} diff --git a/lib/modules/player/player_queue.dart b/lib/modules/player/player_queue.dart index 369b95d2..bfb7a2e3 100644 --- a/lib/modules/player/player_queue.dart +++ b/lib/modules/player/player_queue.dart @@ -1,23 +1,24 @@ -import 'dart:ui'; - +import 'package:auto_size_text/auto_size_text.dart'; import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:scroll_to_index/scroll_to_index.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/button/back_button.dart'; +import 'package:spotube/components/dialogs/playlist_add_track_dialog.dart'; import 'package:spotube/components/fallbacks/not_found.dart'; import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; import 'package:spotube/components/track_tile/track_tile.dart'; -import 'package:spotube/extensions/artist_simple.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/hooks/controllers/use_auto_scroll_controller.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/player/player_queue_actions.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/audio_player/state.dart'; @@ -25,7 +26,7 @@ class PlayerQueue extends HookConsumerWidget { final bool floating; final AudioPlayerState playlist; - final Future Function(Track track) onJump; + final Future Function(SpotubeTrackObject track) onJump; final Future Function(String trackId) onRemove; final Future Function(int oldIndex, int newIndex) onReorder; final Future Function() onStop; @@ -52,24 +53,17 @@ class PlayerQueue extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final mediaQuery = MediaQuery.of(context); + final mediaQuery = MediaQuery.sizeOf(context); final controller = useAutoScrollController(); final searchText = useState(''); + final selectionMode = useState(false); + final selectedTrackIds = useState({}); + final isSearching = useState(false); final tracks = playlist.tracks; - final borderRadius = floating - ? const BorderRadius.only( - topLeft: Radius.circular(10), - ) - : const BorderRadius.only( - topLeft: Radius.circular(10), - topRight: Radius.circular(10), - ); - final theme = Theme.of(context); - final headlineColor = theme.textTheme.headlineSmall?.color; final filteredTracks = useMemoized( () { @@ -79,7 +73,7 @@ class PlayerQueue extends HookConsumerWidget { return tracks .map((e) => ( weightedRatio( - '${e.name!} - ${e.artists?.asString() ?? ""}', + '${e.name} - ${e.artists.asString()}', searchText.value, ), e @@ -92,217 +86,295 @@ class PlayerQueue extends HookConsumerWidget { [tracks, searchText.value], ); - useEffect(() { - if (playlist.activeTrack == null) return null; - - controller.scrollToIndex( - playlist.playlist.index, - preferPosition: AutoScrollPosition.middle, - ); - return null; - }, []); - if (tracks.isEmpty) { - return const NotFound(vertical: true); + return const NotFound(); } - return LayoutBuilder( - builder: (context, constrains) { - return ClipRRect( - borderRadius: borderRadius, - clipBehavior: Clip.hardEdge, - child: BackdropFilter( - filter: ImageFilter.blur( - sigmaX: 15, - sigmaY: 15, - ), - child: Container( - padding: const EdgeInsets.only( - top: 5.0, + return Stack( + children: [ + LayoutBuilder( + builder: (context, constrains) { + final searchBar = ConstrainedBox( + constraints: BoxConstraints( + maxHeight: 40, + maxWidth: mediaQuery.smAndDown ? mediaQuery.width - 40 : 300, ), - decoration: BoxDecoration( - color: - theme.colorScheme.surfaceContainerHighest.withOpacity(0.5), - borderRadius: borderRadius, - ), - child: CallbackShortcuts( - bindings: { - LogicalKeySet(LogicalKeyboardKey.escape): () { - if (!isSearching.value) { - Navigator.of(context).pop(); - } - isSearching.value = false; - searchText.value = ''; - } + child: TextField( + onChanged: (value) { + searchText.value = value; }, - child: InterScrollbar( - controller: controller, - child: CustomScrollView( - controller: controller, - slivers: [ - if (!floating) - SliverToBoxAdapter( - child: Center( - child: Container( - height: 5, - width: 100, - margin: const EdgeInsets.only(bottom: 5, top: 2), - decoration: BoxDecoration( - color: headlineColor, - borderRadius: BorderRadius.circular(20), - ), + placeholder: Text(context.l10n.search), + ), + ); + return CallbackShortcuts( + bindings: { + LogicalKeySet(LogicalKeyboardKey.escape): () { + if (!isSearching.value) { + Navigator.of(context).pop(); + } + isSearching.value = false; + searchText.value = ''; + } + }, + child: Column( + children: [ + if (isSearching.value && mediaQuery.smAndDown) + AppBar( + backgroundColor: Colors.transparent, + leading: [ + if (mediaQuery.smAndDown) + IconButton.ghost( + icon: const Icon( + Icons.arrow_back_ios_new_outlined, ), - ), + onPressed: () { + isSearching.value = false; + searchText.value = ''; + }, + ) + ], + surfaceBlur: 0, + surfaceOpacity: 0, + child: searchBar, + ) + else if (selectionMode.value) + AppBar( + backgroundColor: Colors.transparent, + surfaceBlur: 0, + surfaceOpacity: 0, + leading: [ + IconButton.ghost( + icon: const Icon(SpotubeIcons.close), + onPressed: () { + selectedTrackIds.value = {}; + selectionMode.value = false; + }, + ) + ], + title: SizedBox( + height: 30, + child: AutoSizeText( + '${selectedTrackIds.value.length} selected', + maxLines: 1, ), - SliverAppBar( - floating: true, - pinned: false, - snap: false, - backgroundColor: Colors.transparent, - elevation: 0, - automaticallyImplyLeading: false, - title: BackdropFilter( - filter: ImageFilter.blur( - sigmaX: 10, - sigmaY: 10, - ), - child: SizedBox( - height: kToolbarHeight, - child: mediaQuery.mdAndUp || !isSearching.value - ? Align( - alignment: Alignment.centerLeft, - child: Text( - context.l10n - .tracks_in_queue(tracks.length), - style: TextStyle( - color: headlineColor, - fontWeight: FontWeight.bold, - fontSize: 18, - ), + ), + trailing: [ + PlayerQueueActionButton( + builder: (context, close) => Column( + mainAxisSize: MainAxisSize.min, + children: [ + const Gap(12), + ButtonTile( + style: const ButtonStyle.ghost(), + leading: + const Icon(SpotubeIcons.selectionCheck), + title: Text(context.l10n.select_all), + onPressed: () { + selectedTrackIds.value = + filteredTracks.map((t) => t.id).toSet(); + Navigator.pop(context); + }, + ), + ButtonTile( + style: const ButtonStyle.ghost(), + leading: const Icon(SpotubeIcons.playlistAdd), + title: Text(context.l10n.add_to_playlist), + onPressed: () async { + final selected = filteredTracks + .where((t) => + selectedTrackIds.value.contains(t.id)) + .toList(); + close(); + if (selected.isEmpty) return; + final res = await showDialog( + context: context, + builder: (context) => + PlaylistAddTrackDialog( + tracks: selected, + openFromPlaylist: null, ), - ) - : null, + ); + if (res == true) { + selectedTrackIds.value = {}; + selectionMode.value = false; + } + }, + ), + ButtonTile( + style: const ButtonStyle.ghost(), + leading: const Icon(SpotubeIcons.trash), + title: Text(context.l10n.remove_from_queue), + onPressed: () async { + final ids = selectedTrackIds.value.toList(); + close(); + if (ids.isEmpty) return; + await Future.wait( + ids.map((id) => onRemove(id))); + if (context.mounted) { + selectedTrackIds.value = {}; + selectionMode.value = false; + } + }, + ), + const Gap(12), + ], ), ), - actions: [ - if (mediaQuery.mdAndUp || isSearching.value) - TextField( - onChanged: (value) { - searchText.value = value; - }, - decoration: InputDecoration( - hintText: context.l10n.search, - isDense: true, - prefixIcon: mediaQuery.smAndDown - ? IconButton( - icon: const Icon( - Icons.arrow_back_ios_new_outlined, - ), - onPressed: () { - isSearching.value = false; - searchText.value = ''; - }, - style: IconButton.styleFrom( - padding: EdgeInsets.zero, - minimumSize: const Size.square(20), - ), - ) - : const Icon(SpotubeIcons.filter), - constraints: BoxConstraints( - maxHeight: 40, - maxWidth: mediaQuery.smAndDown - ? mediaQuery.size.width - 40 - : 300, - ), + ], + ) + else + AppBar( + trailingGap: 0, + backgroundColor: Colors.transparent, + surfaceBlur: 0, + surfaceOpacity: 0, + title: mediaQuery.mdAndUp || !isSearching.value + ? SizedBox( + height: 30, + child: AutoSizeText( + context.l10n.tracks_in_queue(tracks.length), + maxLines: 1, ), ) - else - IconButton.filledTonal( - icon: const Icon(SpotubeIcons.filter), - onPressed: () { - isSearching.value = !isSearching.value; - }, - ), - if (mediaQuery.mdAndUp || !isSearching.value) ...[ - const SizedBox(width: 10), - FilledButton( - style: FilledButton.styleFrom( - backgroundColor: theme.scaffoldBackgroundColor - .withOpacity(0.5), - foregroundColor: - theme.textTheme.headlineSmall?.color, - ), - child: Row( - children: [ - const Icon(SpotubeIcons.playlistRemove), - const SizedBox(width: 5), - Text(context.l10n.clear_all), - ], - ), + : null, + trailing: [ + if (mediaQuery.mdAndUp) + searchBar + else + IconButton.ghost( + icon: const Icon(SpotubeIcons.filter), + onPressed: () { + isSearching.value = !isSearching.value; + }, + ), + if (!isSearching.value) ...[ + const SizedBox(width: 10), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.clear_all)) + .call, + child: IconButton.outline( + icon: const Icon(SpotubeIcons.playlistRemove), onPressed: () { onStop(); - Navigator.of(context).pop(); + closeDrawer(context); }, ), - const SizedBox(width: 10), - ], + ), + const Gap(5), + if (mediaQuery.smAndDown) + const BackButton(icon: SpotubeIcons.angleDown), ], - ), - const SliverGap(10), - SliverReorderableList( - onReorder: onReorder, - itemCount: filteredTracks.length, - onReorderStart: (index) { - HapticFeedback.selectionClick(); - }, - onReorderEnd: (index) { - HapticFeedback.selectionClick(); - }, - itemBuilder: (context, i) { - final track = filteredTracks.elementAt(i); - return AutoScrollTag( - key: ValueKey(i), - controller: controller, - index: i, - child: Material( - color: Colors.transparent, - child: TrackTile( - playlist: playlist, + ], + ), + const Divider(), + Expanded( + child: InterScrollbar( + controller: controller, + child: CustomScrollView( + controller: controller, + slivers: [ + const SliverGap(10), + SliverReorderableList( + onReorder: onReorder, + itemCount: filteredTracks.length, + onReorderStart: (index) { + HapticFeedback.selectionClick(); + }, + onReorderEnd: (index) { + HapticFeedback.selectionClick(); + }, + itemBuilder: (context, i) { + final track = filteredTracks.elementAt(i); + + void toggleSelection(String id) { + final s = {...selectedTrackIds.value}; + if (s.contains(id)) { + s.remove(id); + } else { + s.add(id); + } + selectedTrackIds.value = s; + if (selectedTrackIds.value.isEmpty) { + selectionMode.value = false; + } + } + + return AutoScrollTag( + key: ValueKey(i), + controller: controller, index: i, - track: track, - onTap: () async { - if (playlist.activeTrack?.id == track.id) { - return; - } - await onJump(track); - }, - leadingActions: [ - if (!isSearching.value && - searchText.value.isEmpty) - Padding( - padding: const EdgeInsets.only(left: 8.0), - child: ReorderableDragStartListener( - index: i, - child: const Icon( - SpotubeIcons.dragHandle, + child: TrackTile( + playlist: playlist, + index: i, + track: track, + selectionMode: selectionMode.value, + selected: + selectedTrackIds.value.contains(track.id), + onChanged: selectionMode.value + ? (_) => toggleSelection(track.id) + : null, + onTap: () async { + if (selectionMode.value) { + toggleSelection(track.id); + return; + } + if (playlist.activeTrack?.id == track.id) { + return; + } + await onJump(track); + }, + onLongPress: () { + if (!selectionMode.value) { + selectionMode.value = true; + selectedTrackIds.value = {track.id}; + } else { + toggleSelection(track.id); + } + }, + leadingActions: [ + if (!isSearching.value && + searchText.value.isEmpty && + !selectionMode.value) + Padding( + padding: + const EdgeInsets.only(left: 8.0), + child: ReorderableDragStartListener( + index: i, + child: const Icon( + SpotubeIcons.dragHandle, + ), ), ), - ), - ], - ), - ), - ); - }, + ], + ), + ); + }, + ), + const SliverSafeArea(sliver: SliverGap(100)), + ], ), - const SliverGap(100), - ], + ), ), - ), + ], ), - ), + ); + }, + ), + Positioned( + right: 20, + bottom: 20, + child: IconButton.secondary( + icon: const Icon(SpotubeIcons.angleDown), + onPressed: () { + controller.scrollToIndex( + playlist.currentIndex, + preferPosition: AutoScrollPosition.middle, + ); + }, ), - ); - }, + ) + ], ); } } diff --git a/lib/modules/player/player_queue_actions.dart b/lib/modules/player/player_queue_actions.dart new file mode 100644 index 00000000..3d1666c2 --- /dev/null +++ b/lib/modules/player/player_queue_actions.dart @@ -0,0 +1,44 @@ +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/constrains.dart'; + +class PlayerQueueActionButton extends StatelessWidget { + final Widget Function(BuildContext context, VoidCallback close) builder; + + const PlayerQueueActionButton({ + super.key, + required this.builder, + }); + + @override + Widget build(BuildContext context) { + return IconButton.ghost( + onPressed: () { + final mediaQuery = MediaQuery.sizeOf(context); + + if (mediaQuery.lgAndUp) { + showDropdown( + context: context, + builder: (context) { + return SizedBox( + width: 220 * context.theme.scaling, + child: Card( + padding: EdgeInsets.zero, + child: builder(context, () => closeOverlay(context)), + ), + ); + }, + ); + } else { + openSheet( + context: context, + builder: (context) => builder(context, () => closeSheet(context)), + position: OverlayPosition.bottom, + ); + } + }, + icon: const Icon(SpotubeIcons.moreHorizontal), + ); + } +} diff --git a/lib/modules/player/player_track_details.dart b/lib/modules/player/player_track_details.dart index 8d3b99fa..c158aed3 100644 --- a/lib/modules/player/player_track_details.dart +++ b/lib/modules/player/player_track_details.dart @@ -1,21 +1,20 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; + +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/collections/assets.gen.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/links/artist_link.dart'; import 'package:spotube/components/links/link_text.dart'; -import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/track/track.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/utils/service_utils.dart'; class PlayerTrackDetails extends HookConsumerWidget { final Color? color; - final Track? track; + final SpotubeTrackObject? track; const PlayerTrackDetails({super.key, this.color, this.track}); @override @@ -36,9 +35,9 @@ class PlayerTrackDetails extends HookConsumerWidget { child: ClipRRect( borderRadius: BorderRadius.circular(4), child: UniversalImage( - path: (track?.album?.images) + path: (track?.album.images) .asUrlString(placeholder: ImagePlaceholder.albumArt), - placeholder: Assets.albumPlaceholder.path, + placeholder: Assets.images.albumPlaceholder.path, ), ), ), @@ -48,19 +47,17 @@ class PlayerTrackDetails extends HookConsumerWidget { crossAxisAlignment: CrossAxisAlignment.start, children: [ const SizedBox(height: 4), - LinkText( + Text( playback.activeTrack?.name ?? "", - "/track/${playback.activeTrack?.id}", - push: true, overflow: TextOverflow.ellipsis, - style: theme.textTheme.bodyMedium!.copyWith( + style: theme.typography.normal.copyWith( color: color, ), ), Text( - playback.activeTrack?.artists?.asString() ?? "", + playback.activeTrack?.artists.asString() ?? "", overflow: TextOverflow.ellipsis, - style: theme.textTheme.bodySmall!.copyWith(color: color), + style: theme.typography.small.copyWith(color: color), ) ], ), @@ -72,7 +69,7 @@ class PlayerTrackDetails extends HookConsumerWidget { children: [ LinkText( playback.activeTrack?.name ?? "", - "/track/${playback.activeTrack?.id}", + TrackRoute(trackId: playback.activeTrack?.id ?? ""), push: true, overflow: TextOverflow.ellipsis, style: TextStyle(fontWeight: FontWeight.bold, color: color), @@ -80,15 +77,10 @@ class PlayerTrackDetails extends HookConsumerWidget { ArtistLink( artists: playback.activeTrack?.artists ?? [], onRouteChange: (route) { - ServiceUtils.push(context, route); + context.router.navigateNamed(route); }, - onOverflowArtistClick: () => ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": track!.id!, - }, - ), + onOverflowArtistClick: () => + context.navigateTo(TrackRoute(trackId: track!.id)), ) ], ), diff --git a/lib/modules/player/sibling_tracks_sheet.dart b/lib/modules/player/sibling_tracks_sheet.dart index 3a31d88e..b9bd7631 100644 --- a/lib/modules/player/sibling_tracks_sheet.dart +++ b/lib/modules/player/sibling_tracks_sheet.dart @@ -1,60 +1,16 @@ -import 'dart:ui'; - -import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/assets.gen.dart'; -import 'package:spotube/collections/spotube_icons.dart'; - +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/fallbacks/not_found.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/extensions/artist_simple.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/extensions/duration.dart'; -import 'package:spotube/hooks/utils/use_debounce.dart'; -import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/audio_player/querying_track_info.dart'; -import 'package:spotube/provider/server/active_sourced_track.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/video_info.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:spotube/services/sourced_track/sources/invidious.dart'; -import 'package:spotube/services/sourced_track/sources/jiosaavn.dart'; -import 'package:spotube/services/sourced_track/sources/piped.dart'; -import 'package:spotube/services/sourced_track/sources/youtube.dart'; -import 'package:spotube/utils/service_utils.dart'; - -final sourceInfoToIconMap = { - YoutubeSourceInfo: const Icon(SpotubeIcons.youtube, color: Color(0xFFFF0000)), - JioSaavnSourceInfo: Container( - height: 30, - width: 30, - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(90), - image: DecorationImage( - image: Assets.jiosaavn.provider(), - fit: BoxFit.cover, - ), - ), - ), - PipedSourceInfo: const Icon(SpotubeIcons.piped), - InvidiousSourceInfo: Container( - height: 18, - width: 18, - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(90), - image: DecorationImage( - image: Assets.invidious.provider(), - fit: BoxFit.cover, - ), - ), - ), -}; +import 'package:spotube/provider/server/sourced_track_provider.dart'; class SiblingTracksSheet extends HookConsumerWidget { final bool floating; @@ -65,272 +21,135 @@ class SiblingTracksSheet extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final theme = Theme.of(context); - final playlist = ref.watch(audioPlayerProvider); - final isFetchingActiveTrack = ref.watch(queryingTrackInfoProvider); - final preferences = ref.watch(userPreferencesProvider); - - final isSearching = useState(false); - final searchMode = useState(preferences.searchMode); - final activeTrackNotifier = ref.watch(activeSourcedTrackProvider.notifier); - final activeTrack = - ref.watch(activeSourcedTrackProvider) ?? playlist.activeTrack; - - final title = ServiceUtils.getTitle( - activeTrack?.name ?? "", - artists: activeTrack?.artists?.map((e) => e.name!).toList() ?? [], - onlyCleanArtist: true, - ).trim(); - - final defaultSearchTerm = - "$title - ${activeTrack?.artists?.asString() ?? ""}"; - final searchController = useTextEditingController( - text: defaultSearchTerm, - ); - - final searchTerm = useDebounce( - useValueListenable(searchController).text, - ); - final controller = useScrollController(); - final searchRequest = useMemoized(() async { - if (searchTerm.trim().isEmpty) { - return []; - } - if (preferences.audioSource == AudioSource.jiosaavn) { - final resultsJioSaavn = - await jiosaavnClient.search.songs(searchTerm.trim()); - final results = await Future.wait( - resultsJioSaavn.results.mapIndexed((i, song) async { - final siblingType = JioSaavnSourcedTrack.toSiblingType(song); - return siblingType.info; - })); + final activeTrack = + ref.watch(audioPlayerProvider.select((e) => e.activeTrack)); - final activeSourceInfo = (activeTrack! as SourcedTrack).sourceInfo; + if (activeTrack == null || activeTrack is! SpotubeFullTrackObject) { + return const SafeArea(child: NotFound()); + } - return results - ..removeWhere((element) => element.id == activeSourceInfo.id) - ..insert( - 0, - activeSourceInfo, - ); - } else { - final resultsYt = await youtubeClient.search.search(searchTerm.trim()); + return HookBuilder(builder: (context) { + final sourcedTrack = ref.watch(sourcedTrackProvider(activeTrack)); + final sourcedTrackNotifier = + ref.watch(sourcedTrackProvider(activeTrack).notifier); - final searchResults = await Future.wait( - resultsYt - .map(YoutubeVideoInfo.fromVideo) - .mapIndexed((i, video) async { - final siblingType = - await YoutubeSourcedTrack.toSiblingType(i, video); - return siblingType.info; - }), - ); - final activeSourceInfo = (activeTrack! as SourcedTrack).sourceInfo; - return searchResults - ..removeWhere((element) => element.id == activeSourceInfo.id) - ..insert( - 0, - activeSourceInfo, - ); - } - }, [ - searchTerm, - searchMode.value, - activeTrack, - preferences.audioSource, - ]); + final siblings = useMemoized>( + () => !sourcedTrack.isLoading + ? [ + if (sourcedTrack.asData?.value != null) + sourcedTrack.asData!.value.info, + ...?sourcedTrack.asData?.value.siblings, + ] + : [], + [sourcedTrack], + ); - final siblings = useMemoized( - () => !isFetchingActiveTrack - ? [ - (activeTrack as SourcedTrack).sourceInfo, - ...activeTrack.siblings, - ] - : [], - [activeTrack, isFetchingActiveTrack], - ); + useEffect(() { + /// Populate sibling when active track changes + if (sourcedTrack.asData?.value != null && + sourcedTrack.asData?.value.siblings.isEmpty == true) { + sourcedTrackNotifier.copyWithSibling(); + } + return null; + }, [sourcedTrack]); - final borderRadius = floating - ? BorderRadius.circular(10) - : const BorderRadius.only( - topLeft: Radius.circular(10), - topRight: Radius.circular(10), - ); - - useEffect(() { - if (activeTrack is SourcedTrack && activeTrack.siblings.isEmpty) { - activeTrackNotifier.populateSibling(); - } - return null; - }, [activeTrack]); - - final itemBuilder = useCallback( - (SourceInfo sourceInfo) { - final icon = sourceInfoToIconMap[sourceInfo.runtimeType]; - return ListTile( - title: Text(sourceInfo.title), - leading: Padding( - padding: const EdgeInsets.all(8.0), - child: UniversalImage( - path: sourceInfo.thumbnail, - height: 60, - width: 60, - ), - ), - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(5), - ), - trailing: Text(sourceInfo.duration.toHumanReadableString()), - subtitle: Row( - children: [ - if (icon != null) icon, - Text(" • ${sourceInfo.artist}"), - ], - ), - enabled: !isFetchingActiveTrack, - selected: !isFetchingActiveTrack && - sourceInfo.id == (activeTrack as SourcedTrack).sourceInfo.id, - selectedTileColor: theme.popupMenuTheme.color, - onTap: () { - if (!isFetchingActiveTrack && - sourceInfo.id != (activeTrack as SourcedTrack).sourceInfo.id) { - activeTrackNotifier.swapSibling(sourceInfo); - Navigator.of(context).pop(); - } - }, - ); - }, - [activeTrack, siblings], - ); - - final mediaQuery = MediaQuery.of(context); - return SafeArea( - child: ClipRRect( - borderRadius: borderRadius, - clipBehavior: Clip.hardEdge, - child: BackdropFilter( - filter: ImageFilter.blur( - sigmaX: 12.0, - sigmaY: 12.0, - ), - child: AnimatedSize( - duration: const Duration(milliseconds: 300), - child: Container( - height: isSearching.value && mediaQuery.smAndDown - ? mediaQuery.size.height - 50 - : mediaQuery.size.height * .6, - decoration: BoxDecoration( - borderRadius: borderRadius, - color: - theme.colorScheme.surfaceContainerHighest.withOpacity(.5), + return SafeArea( + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Padding( + padding: + const EdgeInsets.symmetric(horizontal: 8.0, vertical: 16), + child: Row( + spacing: 5, + children: [ + AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + child: Text( + context.l10n.alternative_track_sources, + ).bold()), + ], ), - child: Scaffold( - backgroundColor: Colors.transparent, - appBar: AppBar( - centerTitle: true, - title: AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: !isSearching.value - ? Text( - context.l10n.alternative_track_sources, - style: theme.textTheme.headlineSmall, - ) - : TextField( - autofocus: true, - controller: searchController, - decoration: InputDecoration( - hintText: context.l10n.search, - hintStyle: theme.textTheme.headlineSmall, - border: InputBorder.none, - ), - style: theme.textTheme.headlineSmall, - ), - ), - automaticallyImplyLeading: false, - backgroundColor: Colors.transparent, - actions: [ - if (!isSearching.value) - IconButton( - icon: const Icon(SpotubeIcons.search, size: 18), - onPressed: () { - isSearching.value = true; - }, - ) - else ...[ - if (preferences.audioSource == AudioSource.piped) - PopupMenuButton( - icon: const Icon(SpotubeIcons.filter, size: 18), - onSelected: (SearchMode mode) { - searchMode.value = mode; - }, - initialValue: searchMode.value, - itemBuilder: (context) => SearchMode.values - .map( - (e) => PopupMenuItem( - value: e, - child: Text(e.label), - ), - ) - .toList(), + ), + AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + child: sourcedTrack.isLoading + ? const SizedBox( + width: double.infinity, + child: LinearProgressIndicator(), + ) + : const SizedBox.shrink(), + ), + Expanded( + child: AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + transitionBuilder: (child, animation) => + FadeTransition(opacity: animation, child: child), + child: InterScrollbar( + controller: controller, + child: ListView.separated( + padding: const EdgeInsets.all(8.0), + controller: controller, + itemCount: siblings.length, + separatorBuilder: (context, index) => const Gap(8), + itemBuilder: (context, index) { + final sourceInfo = siblings[index]; + + return ButtonTile( + style: ButtonVariance.ghost, + padding: const EdgeInsets.symmetric(horizontal: 8), + title: Text( + sourceInfo.title, + maxLines: 2, + overflow: TextOverflow.ellipsis, ), - IconButton( - icon: const Icon(SpotubeIcons.close, size: 18), - onPressed: () { - isSearching.value = false; - }, - ), - ] - ], - ), - body: Padding( - padding: const EdgeInsets.all(8.0), - child: AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - transitionBuilder: (child, animation) => - FadeTransition(opacity: animation, child: child), - child: InterScrollbar( - controller: controller, - child: switch (isSearching.value) { - false => ListView.builder( - controller: controller, - itemCount: siblings.length, - itemBuilder: (context, index) => - itemBuilder(siblings[index]), - ), - true => FutureBuilder( - future: searchRequest, - builder: (context, snapshot) { - if (snapshot.hasError) { - return Center( - child: Text(snapshot.error.toString()), - ); - } else if (!snapshot.hasData) { - return const Center( - child: CircularProgressIndicator()); - } + leading: sourceInfo.thumbnail != null + ? UniversalImage( + path: sourceInfo.thumbnail!, + height: 60, + width: 60, + ) + : null, + trailing: + Text(sourceInfo.duration.toHumanReadableString()), + subtitle: Text( + sourceInfo.artists.join(", "), + maxLines: 1, + overflow: TextOverflow.ellipsis, + ), + enabled: !sourcedTrack.isLoading, + selected: !sourcedTrack.isLoading && + sourceInfo.id == sourcedTrack.asData?.value.info.id, + onPressed: () async { + if (!sourcedTrack.isLoading && + sourceInfo.id != + sourcedTrack.asData?.value.info.id) { + await sourcedTrackNotifier + .swapWithSibling(sourceInfo); + await ref + .read(audioPlayerProvider.notifier) + .swapActiveSource(); - return InterScrollbar( - controller: controller, - child: ListView.builder( - controller: controller, - itemCount: snapshot.data!.length, - itemBuilder: (context, index) => - itemBuilder(snapshot.data![index]), - ), - ); - }, - ), - }, - ), + if (context.mounted) { + if (MediaQuery.sizeOf(context).mdAndUp) { + closeOverlay(context); + } else { + closeDrawer(context); + } + } + } + }, + ); + }, ), ), ), ), - ), + ], ), - ), - ); + ); + }); } } diff --git a/lib/modules/player/volume_slider.dart b/lib/modules/player/volume_slider.dart index 8483143b..ee4ac9c5 100644 --- a/lib/modules/player/volume_slider.dart +++ b/lib/modules/player/volume_slider.dart @@ -1,6 +1,7 @@ import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; class VolumeSlider extends HookConsumerWidget { @@ -30,24 +31,24 @@ class VolumeSlider extends HookConsumerWidget { } } }, - child: SliderTheme( - data: const SliderThemeData( - showValueIndicator: ShowValueIndicator.always, - ), + child: SizedBox( + height: 20, + width: 100, child: Slider( min: 0, max: 1, - label: (value * 100).toStringAsFixed(0), - value: value, - onChanged: onChanged, + value: SliderValue.single(value), + onChanged: (v) => onChanged(v.value), ), ), ); + return Row( mainAxisAlignment: !fullWidth ? MainAxisAlignment.center : MainAxisAlignment.start, children: [ IconButton( + variance: ButtonVariance.ghost, icon: Icon( value == 0 ? SpotubeIcons.volumeMute diff --git a/lib/modules/playlist/playlist_card.dart b/lib/modules/playlist/playlist_card.dart index df683a80..1d221a33 100644 --- a/lib/modules/playlist/playlist_card.dart +++ b/lib/modules/playlist/playlist_card.dart @@ -1,27 +1,37 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/dialogs/select_device_dialog.dart'; -import 'package:spotube/components/playbutton_card.dart'; +import 'package:spotube/components/playbutton_view/playbutton_card.dart'; +import 'package:spotube/components/playbutton_view/playbutton_tile.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; import 'package:spotube/models/connect/connect.dart'; -import 'package:spotube/pages/playlist/playlist.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/querying_track_info.dart'; import 'package:spotube/provider/connect/connect.dart'; import 'package:spotube/provider/history/history.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/library/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/playlist.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; -import 'package:spotube/utils/service_utils.dart'; class PlaylistCard extends HookConsumerWidget { - final PlaylistSimple playlist; + final SpotubeSimplePlaylistObject playlist; + final bool _isTile; + const PlaylistCard( this.playlist, { super.key, - }); + }) : _isTile = false; + + const PlaylistCard.tile( + this.playlist, { + super.key, + }) : _isTile = true; + @override Widget build(BuildContext context, ref) { final playlistQueue = ref.watch(audioPlayerProvider); @@ -31,125 +41,184 @@ class PlaylistCard extends HookConsumerWidget { final playing = useStream(audioPlayer.playingStream).data ?? audioPlayer.isPlaying; - bool isPlaylistPlaying = useMemoized( - () => playlistQueue.containsCollection(playlist.id!), + + final isPlaylistPlaying = useMemoized( + () => playlistQueue.containsCollection(playlist.id), [playlistQueue, playlist.id], ); final updating = useState(false); - final me = ref.watch(meProvider); + final me = ref.watch(metadataPluginUserProvider); - Future> fetchInitialTracks() async { + final fetchInitialTracks = useCallback(() async { if (playlist.id == 'user-liked-tracks') { - return await ref.read(likedTracksProvider.future); + final tracks = await ref.read(metadataPluginSavedTracksProvider.future); + return tracks.items; } - final result = - await ref.read(playlistTracksProvider(playlist.id!).future); + final result = await ref + .read(metadataPluginPlaylistTracksProvider(playlist.id).future); return result.items; - } + }, [playlist.id, ref]); - Future> fetchAllTracks() async { - final initialTracks = await fetchInitialTracks(); + final fetchAllTracks = useCallback(() async { + await fetchInitialTracks(); if (playlist.id == 'user-liked-tracks') { - return initialTracks; + return ref.read(metadataPluginSavedTracksProvider.notifier).fetchAll(); } - return ref.read(playlistTracksProvider(playlist.id!).notifier).fetchAll(); + return ref + .read(metadataPluginPlaylistTracksProvider(playlist.id).notifier) + .fetchAll(); + }, [playlist.id, ref, fetchInitialTracks]); + + final onTap = useCallback(() { + context.navigateTo(PlaylistRoute(id: playlist.id, playlist: playlist)); + }, [context, playlist]); + + final onPlaybuttonPressed = useCallback(() async { + try { + updating.value = true; + if (isPlaylistPlaying && playing) { + return audioPlayer.pause(); + } else if (isPlaylistPlaying && !playing) { + return audioPlayer.resume(); + } + + final fetchedInitialTracks = await fetchInitialTracks(); + + if (fetchedInitialTracks.isEmpty || !context.mounted) return; + + final isRemoteDevice = await showSelectDeviceDialog(context, ref); + if (isRemoteDevice == null) return; + if (isRemoteDevice) { + final remotePlayback = ref.read(connectProvider.notifier); + final allTracks = await fetchAllTracks(); + await remotePlayback.load( + WebSocketLoadEventData.playlist( + tracks: allTracks, + collection: playlist, + ), + ); + } else { + await playlistNotifier.load(fetchedInitialTracks, autoPlay: true); + playlistNotifier.addCollection(playlist.id); + historyNotifier.addPlaylists([playlist]); + + final allTracks = await fetchAllTracks(); + + await playlistNotifier + .addTracks(allTracks.sublist(fetchedInitialTracks.length)); + } + } finally { + if (context.mounted) { + updating.value = false; + } + } + }, [ + isPlaylistPlaying, + playing, + fetchInitialTracks, + context, + showSelectDeviceDialog, + ref, + connectProvider, + fetchAllTracks, + playlistNotifier, + playlist.id, + historyNotifier, + playlist, + updating + ]); + + final onAddToQueuePressed = useCallback(() async { + updating.value = true; + try { + if (isPlaylistPlaying) return; + + final fetchedInitialTracks = await fetchAllTracks(); + + if (fetchedInitialTracks.isEmpty) return; + + playlistNotifier.addTracks(fetchedInitialTracks); + playlistNotifier.addCollection(playlist.id); + historyNotifier.addPlaylists([playlist]); + if (context.mounted) { + showToast( + context: context, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + content: Text( + context.l10n + .added_num_tracks_to_queue(fetchedInitialTracks.length), + ), + trailing: Button.outline( + child: Text(context.l10n.undo), + onPressed: () { + playlistNotifier + .removeTracks(fetchedInitialTracks.map((e) => e.id)); + }, + ), + ), + ); + }, + ); + } + } finally { + updating.value = false; + } + }, [ + isPlaylistPlaying, + fetchAllTracks, + playlistNotifier, + playlist.id, + historyNotifier, + playlist, + context, + updating + ]); + + final imageUrl = useMemoized( + () => playlist.images.from200PxTo300PxOrSmallestImage( + ImagePlaceholder.collection, + ), + [playlist.images], + ); + + final isLoading = + (isPlaylistPlaying && isFetchingActiveTrack) || updating.value; + final isOwner = playlist.owner.id == me.asData?.value?.id && + me.asData?.value?.id != null; + + if (_isTile) { + return PlaybuttonTile( + title: playlist.name, + description: playlist.description, + image: null, + imageUrl: imageUrl, + isPlaying: isPlaylistPlaying, + isLoading: isLoading, + isOwner: isOwner, + onTap: onTap, + onPlaybuttonPressed: onPlaybuttonPressed, + onAddToQueuePressed: onAddToQueuePressed, + ); } return PlaybuttonCard( - margin: const EdgeInsets.symmetric(horizontal: 10), - title: playlist.name!, + title: playlist.name, description: playlist.description, - imageUrl: playlist.images.asUrlString( - placeholder: ImagePlaceholder.collection, - ), + image: null, + imageUrl: imageUrl, isPlaying: isPlaylistPlaying, - isLoading: (isPlaylistPlaying && isFetchingActiveTrack) || updating.value, - isOwner: playlist.owner?.id == me.asData?.value.id && - me.asData?.value.id != null, - onTap: () { - ServiceUtils.pushNamed( - context, - PlaylistPage.name, - pathParameters: { - "id": playlist.id!, - }, - extra: playlist, - ); - }, - onPlaybuttonPressed: () async { - try { - updating.value = true; - if (isPlaylistPlaying && playing) { - return audioPlayer.pause(); - } else if (isPlaylistPlaying && !playing) { - return audioPlayer.resume(); - } - - final fetchedInitialTracks = await fetchInitialTracks(); - - if (fetchedInitialTracks.isEmpty || !context.mounted) return; - - final isRemoteDevice = await showSelectDeviceDialog(context, ref); - if (isRemoteDevice) { - final remotePlayback = ref.read(connectProvider.notifier); - final allTracks = await fetchAllTracks(); - await remotePlayback.load( - WebSocketLoadEventData.playlist( - tracks: allTracks, - collection: playlist, - ), - ); - } else { - await playlistNotifier.load(fetchedInitialTracks, autoPlay: true); - playlistNotifier.addCollection(playlist.id!); - historyNotifier.addPlaylists([playlist]); - - final allTracks = await fetchAllTracks(); - - await playlistNotifier - .addTracks(allTracks.sublist(fetchedInitialTracks.length)); - } - } finally { - if (context.mounted) { - updating.value = false; - } - } - }, - onAddToQueuePressed: () async { - updating.value = true; - try { - if (isPlaylistPlaying) return; - - final fetchedInitialTracks = await fetchAllTracks(); - - if (fetchedInitialTracks.isEmpty) return; - - playlistNotifier.addTracks(fetchedInitialTracks); - playlistNotifier.addCollection(playlist.id!); - historyNotifier.addPlaylists([playlist]); - if (context.mounted) { - final snackbar = SnackBar( - content: Text(context.l10n - .added_num_tracks_to_queue(fetchedInitialTracks.length)), - action: SnackBarAction( - label: "Undo", - onPressed: () { - playlistNotifier - .removeTracks(fetchedInitialTracks.map((e) => e.id!)); - }, - ), - ); - ScaffoldMessenger.maybeOf(context)?.showSnackBar(snackbar); - } - } finally { - updating.value = false; - } - }, + isLoading: isLoading, + isOwner: isOwner, + onTap: onTap, + onPlaybuttonPressed: onPlaybuttonPressed, + onAddToQueuePressed: onAddToQueuePressed, ); } } diff --git a/lib/modules/playlist/playlist_create_dialog.dart b/lib/modules/playlist/playlist_create_dialog.dart index 78680a1c..0fdcf081 100644 --- a/lib/modules/playlist/playlist_create_dialog.dart +++ b/lib/modules/playlist/playlist_create_dialog.dart @@ -1,263 +1,283 @@ import 'dart:convert'; import 'dart:io'; +import 'package:auto_route/auto_route.dart'; import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:form_builder_validators/form_builder_validators.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:form_validator/form_validator.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:image_picker/image_picker.dart'; -import 'package:spotify/spotify.dart'; +import 'package:path/path.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/form/checkbox_form_field.dart'; +import 'package:spotube/components/form/text_form_field.dart'; import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/extensions/string.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; -import 'package:spotube/provider/spotify_provider.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:spotube/provider/metadata_plugin/playlist/playlist.dart'; class PlaylistCreateDialog extends HookConsumerWidget { /// Track ids to add to the playlist final List trackIds; final String? playlistId; - PlaylistCreateDialog({ + const PlaylistCreateDialog({ super.key, this.trackIds = const [], this.playlistId, }); - final formKey = GlobalKey(); - @override Widget build(BuildContext context, ref) { - return ScaffoldMessenger( - child: Scaffold( - backgroundColor: Colors.transparent, - body: HookBuilder(builder: (context) { - final userPlaylists = ref.watch(favoritePlaylistsProvider); - final playlist = ref.watch(playlistProvider(playlistId ?? "")); - final playlistNotifier = - ref.watch(playlistProvider(playlistId ?? "").notifier); + final userPlaylists = ref.watch(metadataPluginSavedPlaylistsProvider); + final playlist = + ref.watch(metadataPluginPlaylistProvider(playlistId ?? "")); + final playlistNotifier = + ref.watch(metadataPluginPlaylistProvider(playlistId ?? "").notifier); - final updatingPlaylist = useMemoized( - () => userPlaylists.asData?.value.items - .firstWhereOrNull((playlist) => playlist.id == playlistId), - [ - userPlaylists.asData?.value.items, - playlistId, - ], - ); + final isSubmitting = useState(false); - final playlistName = useTextEditingController( - text: updatingPlaylist?.name, - ); - final description = useTextEditingController( - text: updatingPlaylist?.description?.unescapeHtml(), - ); - final public = useState( - updatingPlaylist?.public ?? false, - ); - final collaborative = useState( - updatingPlaylist?.collaborative ?? false, - ); - final image = useState(null); + final formKey = useMemoized(() => GlobalKey(), []); - final isUpdatingPlaylist = playlistId != null; + final updatingPlaylist = useMemoized( + () => userPlaylists.asData?.value.items + .firstWhereOrNull((playlist) => playlist.id == playlistId), + [ + userPlaylists.asData?.value.items, + playlistId, + ], + ); - final l10n = context.l10n; - final theme = Theme.of(context); - final scaffold = ScaffoldMessenger.of(context); + final isUpdatingPlaylist = playlistId != null; - final onError = useCallback((error) { - if (error is SpotifyError || error is SpotifyException) { - scaffold.showSnackBar( - SnackBar( - content: Text( - l10n.error(error.message ?? context.l10n.epic_failure), - style: theme.textTheme.bodyMedium!.copyWith( - color: theme.colorScheme.onError, - ), - ), - backgroundColor: theme.colorScheme.error, + final l10n = context.l10n; + final theme = Theme.of(context); + + useEffect(() { + if (playlist.asData?.value != null) { + formKey.currentState?.patchValue({ + 'playlistName': playlist.asData!.value.name, + 'description': playlist.asData!.value.description, + 'public': playlist.asData!.value.public, + 'collaborative': playlist.asData!.value.collaborative, + }); + } + + return; + }, [playlist]); + + final onError = useCallback((error) { + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + title: Text( + l10n.error(l10n.epic_failure), + style: theme.typography.normal.copyWith( + color: theme.colorScheme.destructive, ), - ); - } - }, [scaffold, l10n, theme]); - - Future onCreate() async { - if (!formKey.currentState!.validate()) return; - - final PlaylistInput payload = ( - playlistName: playlistName.text, - collaborative: collaborative.value, - public: public.value, - description: description.text, - base64Image: image.value?.path != null - ? await image.value! - .readAsBytes() - .then((bytes) => base64Encode(bytes)) - : null, - ); - - if (isUpdatingPlaylist) { - await playlistNotifier.modify(payload, onError); - } else { - await playlistNotifier.create(payload, onError); - } - - if (context.mounted && - !ref.read(playlistProvider(playlistId ?? "")).hasError) { - context.pop(); - } - } - - return AlertDialog( - title: Text( - isUpdatingPlaylist - ? context.l10n.update_playlist - : context.l10n.create_a_playlist, + ), ), - actions: [ - OutlinedButton( - child: Text(context.l10n.cancel), - onPressed: () { - Navigator.pop(context); + ); + }, + ); + }, [l10n, theme]); + + Future onCreate() async { + if (!formKey.currentState!.saveAndValidate()) return; + + try { + isSubmitting.value = true; + final values = formKey.currentState!.value; + + final payload = ( + playlistName: values['playlistName'], + collaborative: values['collaborative'], + public: values['public'], + description: values['description'], + base64Image: (values['image'] as XFile?)?.path != null + ? await (values['image'] as XFile) + .readAsBytes() + .then((bytes) => base64Encode(bytes)) + : null, + ); + + if (isUpdatingPlaylist) { + await playlistNotifier.modify( + name: payload.playlistName, + description: payload.description, + public: payload.public, + collaborative: payload.collaborative, + onError: onError, + ); + } else { + await playlistNotifier.create( + name: payload.playlistName, + description: payload.description, + public: payload.public, + collaborative: payload.collaborative, + onError: onError, + ); + } + + if (trackIds.isNotEmpty) { + await playlistNotifier.addTracks(trackIds, onError); + } + } finally { + isSubmitting.value = false; + if (context.mounted && + !ref + .read(metadataPluginPlaylistProvider(playlistId ?? "")) + .hasError) { + context.router.maybePop( + await ref + .read(metadataPluginPlaylistProvider(playlistId ?? "").future), + ); + } + } + } + + return AlertDialog( + title: Text( + isUpdatingPlaylist + ? context.l10n.update_playlist + : context.l10n.create_a_playlist, + ), + actions: [ + Button.outline( + child: Text(context.l10n.cancel), + onPressed: () { + Navigator.pop(context); + }, + ), + Button.primary( + onPressed: onCreate, + enabled: !playlist.isLoading & !isSubmitting.value, + child: Text( + isUpdatingPlaylist ? context.l10n.update : context.l10n.create, + ), + ), + ], + content: Container( + width: MediaQuery.of(context).size.width, + constraints: const BoxConstraints(maxWidth: 500), + child: FormBuilder( + key: formKey, + initialValue: { + 'playlistName': updatingPlaylist?.name, + 'description': updatingPlaylist?.description, + 'public': playlist.asData?.value.public ?? false, + 'collaborative': playlist.asData?.value.collaborative ?? false, + }, + child: ListView( + shrinkWrap: true, + children: [ + FormBuilderField( + name: 'image', + validator: (value) { + if (value == null) return null; + final file = File(value.path); + + if (file.lengthSync() > 256000) { + return "Image size should be less than 256kb"; + } + + if (extension(file.path) != ".png") { + return "Image should be in PNG format"; + } + return null; + }, + builder: (field) { + return Column( + spacing: 10, + children: [ + UniversalImage( + path: field.value?.path ?? + (updatingPlaylist?.images).asUrlString( + placeholder: ImagePlaceholder.collection, + ), + height: 200, + ), + Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Button.secondary( + leading: const Icon(SpotubeIcons.edit), + child: Text( + field.value?.path != null || + updatingPlaylist?.images != null + ? context.l10n.change_cover + : context.l10n.add_cover, + ), + onPressed: () async { + final imageFile = await ImagePicker().pickImage( + source: ImageSource.gallery, + ); + + if (imageFile != null) { + field.didChange(imageFile); + field.validate(); + field.save(); + } + }, + ), + const SizedBox(width: 10), + IconButton.destructive( + icon: const Icon(SpotubeIcons.trash), + enabled: field.value != null, + onPressed: () { + field.didChange(null); + field.validate(); + field.save(); + }, + ), + ], + ), + if (field.hasError) + Text( + field.errorText ?? "", + style: theme.typography.normal.copyWith( + color: theme.colorScheme.destructive, + ), + ) + ], + ); }, ), - FilledButton( - onPressed: playlist.isLoading ? null : onCreate, - child: Text( - isUpdatingPlaylist - ? context.l10n.update - : context.l10n.create, - ), + const Gap(20), + TextFormBuilderField( + name: 'playlistName', + label: Text(context.l10n.playlist_name), + placeholder: Text(context.l10n.name_of_playlist), + validator: FormBuilderValidators.required(), + ), + const Gap(20), + TextFormBuilderField( + name: 'description', + label: Text(context.l10n.description), + validator: FormBuilderValidators.required(), + placeholder: Text(context.l10n.description), + keyboardType: TextInputType.multiline, + maxLines: 5, + ), + const Gap(20), + CheckboxFormBuilderField( + name: 'public', + trailing: Text(context.l10n.public), + ), + const Gap(10), + CheckboxFormBuilderField( + name: 'collaborative', + trailing: Text(context.l10n.collaborative), ), ], - insetPadding: const EdgeInsets.all(8), - content: Container( - width: MediaQuery.of(context).size.width, - constraints: const BoxConstraints(maxWidth: 500), - child: Form( - key: formKey, - child: ListView( - shrinkWrap: true, - children: [ - FormField( - initialValue: image.value, - onSaved: (newValue) { - image.value = newValue; - }, - validator: (value) { - if (value == null) return null; - final file = File(value.path); - - if (file.lengthSync() > 256000) { - return "Image size should be less than 256kb"; - } - return null; - }, - builder: (field) { - return Column( - children: [ - UniversalImage( - path: field.value?.path ?? - (updatingPlaylist?.images).asUrlString( - placeholder: ImagePlaceholder.collection, - ), - height: 200, - ), - const SizedBox(height: 10), - Row( - mainAxisAlignment: MainAxisAlignment.center, - children: [ - FilledButton.icon( - icon: const Icon(SpotubeIcons.edit), - label: Text( - field.value?.path != null || - updatingPlaylist?.images != null - ? context.l10n.change_cover - : context.l10n.add_cover, - ), - onPressed: () async { - final imageFile = await ImagePicker() - .pickImage( - source: ImageSource.gallery); - - if (imageFile != null) { - field.didChange(imageFile); - field.validate(); - field.save(); - } - }, - ), - const SizedBox(width: 10), - IconButton.filled( - icon: const Icon(SpotubeIcons.trash), - style: IconButton.styleFrom( - backgroundColor: - theme.colorScheme.errorContainer, - foregroundColor: theme.colorScheme.error, - ), - onPressed: field.value == null - ? null - : () { - field.didChange(null); - field.validate(); - field.save(); - }, - ), - ], - ), - if (field.hasError) - Text( - field.errorText ?? "", - style: theme.textTheme.bodyMedium!.copyWith( - color: theme.colorScheme.error, - ), - ) - ], - ); - }), - const SizedBox(height: 10), - TextFormField( - controller: playlistName, - decoration: InputDecoration( - hintText: context.l10n.name_of_playlist, - labelText: context.l10n.name_of_playlist, - ), - validator: ValidationBuilder().required().build(), - ), - const SizedBox(height: 10), - TextFormField( - controller: description, - decoration: InputDecoration( - hintText: context.l10n.description, - ), - keyboardType: TextInputType.multiline, - validator: ValidationBuilder().required().build(), - maxLines: 5, - ), - const SizedBox(height: 10), - CheckboxListTile( - title: Text(context.l10n.public), - value: public.value, - onChanged: (val) => public.value = val ?? false, - ), - const SizedBox(height: 10), - CheckboxListTile( - title: Text(context.l10n.collaborative), - value: collaborative.value, - onChanged: (val) => collaborative.value = val ?? false, - ), - ], - ), - ), - ), - ); - }), + ), + ), ), ); } @@ -266,35 +286,22 @@ class PlaylistCreateDialog extends HookConsumerWidget { class PlaylistCreateDialogButton extends HookConsumerWidget { const PlaylistCreateDialogButton({super.key}); - showPlaylistDialog(BuildContext context, SpotifyApi spotify) { + showPlaylistDialog(BuildContext context) { showDialog( context: context, - builder: (context) => PlaylistCreateDialog(), + alignment: Alignment.center, + builder: (context) => const ToastLayer( + child: PlaylistCreateDialog(), + ), ); } @override Widget build(BuildContext context, ref) { - final mediaQuery = MediaQuery.of(context); - final spotify = ref.watch(spotifyProvider); - - if (mediaQuery.smAndDown) { - return ElevatedButton( - style: FilledButton.styleFrom( - foregroundColor: Theme.of(context).colorScheme.primary, - ), - child: const Icon(SpotubeIcons.addFilled), - onPressed: () => showPlaylistDialog(context, spotify), - ); - } - - return FilledButton.tonalIcon( - style: FilledButton.styleFrom( - foregroundColor: Theme.of(context).colorScheme.primary, - ), - icon: const Icon(SpotubeIcons.addFilled), - label: Text(context.l10n.create_playlist), - onPressed: () => showPlaylistDialog(context, spotify), + return Button.secondary( + leading: const Icon(SpotubeIcons.addFilled), + child: Text(context.l10n.playlist), + onPressed: () => showPlaylistDialog(context), ); } } diff --git a/lib/modules/root/bottom_player.dart b/lib/modules/root/bottom_player.dart index a2f45449..33497d8d 100644 --- a/lib/modules/root/bottom_player.dart +++ b/lib/modules/root/bottom_player.dart @@ -1,12 +1,14 @@ -import 'dart:ui'; - +import 'package:auto_route/auto_route.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/assets.gen.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/player/player_actions.dart'; import 'package:spotube/modules/player/player_overlay.dart'; import 'package:spotube/modules/player/player_track_details.dart'; @@ -14,9 +16,6 @@ import 'package:spotube/modules/player/player_controls.dart'; import 'package:spotube/modules/player/volume_slider.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; -import 'package:flutter/material.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; @@ -36,21 +35,13 @@ class BottomPlayer extends HookConsumerWidget { final mediaQuery = MediaQuery.of(context); String albumArt = useMemoized( - () => playlist.activeTrack?.album?.images?.isNotEmpty == true - ? (playlist.activeTrack?.album?.images).asUrlString( - index: (playlist.activeTrack?.album?.images?.length ?? 1) - 1, + () => playlist.activeTrack?.album.images.isNotEmpty == true + ? (playlist.activeTrack?.album.images).asUrlString( + index: (playlist.activeTrack?.album.images.length ?? 1) - 1, placeholder: ImagePlaceholder.albumArt, ) - : Assets.albumPlaceholder.path, - [playlist.activeTrack?.album?.images], - ); - - final theme = Theme.of(context); - final bg = theme.colorScheme.surfaceContainerHighest; - - final bgColor = useBrightnessValue( - Color.lerp(bg, Colors.white, 0.7), - Color.lerp(bg, Colors.black, 0.45)!, + : Assets.images.albumPlaceholder.path, + [playlist.activeTrack?.album.images], ); // returning an empty non spacious Container as the overlay will take @@ -60,84 +51,82 @@ class BottomPlayer extends HookConsumerWidget { return PlayerOverlay(albumArt: albumArt); } - return ClipRect( - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 15, sigmaY: 15), - child: DecoratedBox( - decoration: BoxDecoration(color: bgColor?.withOpacity(0.8)), - child: Material( - type: MaterialType.transparency, - textStyle: theme.textTheme.bodyMedium!, - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Expanded( - child: PlayerTrackDetails(track: playlist.activeTrack), - ), - // controls - const Flexible( - flex: 3, - child: Padding( - padding: EdgeInsets.only(top: 5), - child: PlayerControls(), - ), - ), - // add to saved tracks - Column( - children: [ - PlayerActions( - extraActions: [ - IconButton( - tooltip: context.l10n.mini_player, - icon: const Icon(SpotubeIcons.miniPlayer), - onPressed: () async { - if (!kIsDesktop) return; - - final prevSize = await windowManager.getSize(); - await windowManager.setMinimumSize( - const Size(300, 300), - ); - await windowManager.setAlwaysOnTop(true); - if (!kIsLinux) { - await windowManager.setHasShadow(false); - } - await windowManager - .setAlignment(Alignment.topRight); - await windowManager.setSize(const Size(400, 500)); - await Future.delayed( - const Duration(milliseconds: 100), - () async { - GoRouter.of(context).go( - '/mini-player', - extra: prevSize, - ); - }, - ); - }, - ), - ], - ), - Container( - height: 40, - constraints: const BoxConstraints(maxWidth: 250), - padding: const EdgeInsets.only(right: 10), - child: Consumer(builder: (context, ref, _) { - final volume = ref.watch(volumeProvider); - return VolumeSlider( - fullWidth: true, - value: volume, - onChanged: (value) { - ref.read(volumeProvider.notifier).setVolume(value); - }, - ); - }), - ) - ], - ), - ], + return SurfaceCard( + borderRadius: BorderRadius.zero, + surfaceBlur: context.theme.surfaceBlur, + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Expanded( + child: PlayerTrackDetails(track: playlist.activeTrack), + ), + // controls + const Flexible( + flex: 3, + child: Padding( + padding: EdgeInsets.only(top: 5), + child: PlayerControls(), ), ), - ), + // add to saved tracks + Column( + mainAxisSize: MainAxisSize.min, + children: [ + PlayerActions( + extraActions: [ + Tooltip( + tooltip: + TooltipContainer(child: Text(context.l10n.mini_player)) + .call, + child: IconButton( + variance: ButtonVariance.ghost, + icon: const Icon(SpotubeIcons.miniPlayer), + onPressed: () async { + if (!kIsDesktop) return; + + final prevSize = await windowManager.getSize(); + await windowManager.setMinimumSize( + const Size(300, 300), + ); + await windowManager.setAlwaysOnTop(true); + if (!kIsLinux) { + await windowManager.setHasShadow(false); + } + await windowManager.setAlignment(Alignment.topRight); + await windowManager.setSize(const Size(400, 500)); + await Future.delayed( + const Duration(milliseconds: 100), + () async { + if (context.mounted) { + context.navigateTo( + MiniLyricsRoute(prevSize: prevSize), + ); + } + }, + ); + }, + ), + ), + ], + ), + Container( + height: 40, + constraints: const BoxConstraints(maxWidth: 250), + padding: const EdgeInsets.only(right: 10), + child: Consumer(builder: (context, ref, _) { + final volume = ref.watch(volumeProvider); + return VolumeSlider( + fullWidth: true, + value: volume, + onChanged: (value) { + ref.read(volumeProvider.notifier).setVolume(value); + }, + ); + }), + ) + ], + ), + ], ), ); } diff --git a/lib/modules/root/sidebar.dart b/lib/modules/root/sidebar.dart deleted file mode 100644 index f29644fb..00000000 --- a/lib/modules/root/sidebar.dart +++ /dev/null @@ -1,319 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:flutter/material.dart'; -import 'package:sidebarx/sidebarx.dart'; - -import 'package:spotube/collections/assets.gen.dart'; -import 'package:spotube/collections/side_bar_tiles.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/modules/connect/connect_device.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; -import 'package:spotube/hooks/controllers/use_sidebarx_controller.dart'; -import 'package:spotube/pages/profile/profile.dart'; -import 'package:spotube/pages/settings/settings.dart'; -import 'package:spotube/provider/download_manager_provider.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -import 'package:spotube/utils/platform.dart'; -import 'package:spotube/utils/service_utils.dart'; -import 'package:window_manager/window_manager.dart'; - -class Sidebar extends HookConsumerWidget { - final Widget child; - - const Sidebar({ - required this.child, - super.key, - }); - - static Widget brandLogo() { - return Container( - decoration: BoxDecoration( - color: Colors.black, - borderRadius: BorderRadius.circular(50), - ), - child: Assets.spotubeLogoPng.image(height: 50), - ); - } - - @override - Widget build(BuildContext context, WidgetRef ref) { - final routerState = GoRouterState.of(context); - final mediaQuery = MediaQuery.of(context); - - final downloadCount = ref.watch(downloadManagerProvider).$downloadCount; - - final layoutMode = - ref.watch(userPreferencesProvider.select((s) => s.layoutMode)); - - final sidebarTileList = useMemoized( - () => getSidebarTileList(context.l10n), - [context.l10n], - ); - - final selectedIndex = sidebarTileList.indexWhere( - (e) => routerState.namedLocation(e.name) == routerState.matchedLocation, - ); - - final controller = useSidebarXController( - selectedIndex: selectedIndex, - extended: mediaQuery.lgAndUp, - ); - - final theme = Theme.of(context); - final bg = theme.colorScheme.surfaceContainerHighest; - - final bgColor = useBrightnessValue( - Color.lerp(bg, Colors.white, 0.6), - Color.lerp(bg, Colors.black, 0.45)!, - ); - - useEffect(() { - if (!context.mounted) return; - if (mediaQuery.lgAndUp && !controller.extended) { - controller.setExtended(true); - } else if (mediaQuery.mdAndDown && controller.extended) { - controller.setExtended(false); - } - return null; - }, [mediaQuery, controller]); - - useEffect(() { - if (controller.selectedIndex != selectedIndex) { - controller.selectIndex(selectedIndex); - } - return null; - }, [selectedIndex]); - - if (layoutMode == LayoutMode.compact || - (mediaQuery.smAndDown && layoutMode == LayoutMode.adaptive)) { - return Scaffold(body: child); - } - - return Row( - children: [ - SafeArea( - child: SidebarX( - controller: controller, - items: sidebarTileList.mapIndexed( - (index, e) { - return SidebarXItem( - onTap: () { - context.goNamed(e.name); - }, - iconBuilder: (selected, hovered) { - return Badge( - backgroundColor: theme.colorScheme.primary, - isLabelVisible: e.title == "Library" && downloadCount > 0, - label: Text( - downloadCount.toString(), - style: const TextStyle( - color: Colors.white, - fontSize: 10, - ), - ), - child: Icon( - e.icon, - color: selected || hovered - ? theme.colorScheme.primary - : null, - ), - ); - }, - label: e.title, - ); - }, - ).toList(), - headerBuilder: (_, __) => const SidebarHeader(), - footerBuilder: (_, __) => const Padding( - padding: EdgeInsets.only(bottom: 5), - child: SidebarFooter(), - ), - showToggleButton: false, - theme: SidebarXTheme( - width: 50, - margin: EdgeInsets.only(bottom: 10, top: kIsMacOS ? 35 : 5), - selectedItemDecoration: BoxDecoration( - borderRadius: BorderRadius.circular(10), - color: theme.colorScheme.primary.withOpacity(0.1), - ), - selectedIconTheme: IconThemeData( - color: theme.colorScheme.primary, - ), - ), - extendedTheme: SidebarXTheme( - width: 250, - margin: EdgeInsets.only( - bottom: 10, - left: 0, - top: kIsMacOS ? 0 : 5, - ), - padding: const EdgeInsets.symmetric(horizontal: 6), - decoration: BoxDecoration( - color: bgColor?.withOpacity(0.8), - borderRadius: const BorderRadius.only( - topRight: Radius.circular(10), - bottomRight: Radius.circular(10), - ), - ), - selectedItemDecoration: BoxDecoration( - borderRadius: BorderRadius.circular(10), - color: theme.colorScheme.primary.withOpacity(0.1), - ), - selectedIconTheme: IconThemeData( - color: theme.colorScheme.primary, - ), - selectedTextStyle: theme.textTheme.bodyMedium?.copyWith( - color: theme.colorScheme.primary, - fontWeight: FontWeight.w600, - ), - itemTextPadding: const EdgeInsets.only(left: 10), - selectedItemTextPadding: const EdgeInsets.only(left: 10), - hoverTextStyle: theme.textTheme.bodyMedium?.copyWith( - color: theme.colorScheme.primary, - ), - ), - ), - ), - Expanded(child: child) - ], - ); - } -} - -class SidebarHeader extends HookWidget { - const SidebarHeader({super.key}); - - @override - Widget build(BuildContext context) { - final mediaQuery = MediaQuery.of(context); - final theme = Theme.of(context); - - if (mediaQuery.mdAndDown) { - return Container( - height: 40, - width: 40, - margin: const EdgeInsets.only(bottom: 5), - child: Sidebar.brandLogo(), - ); - } - - return DragToMoveArea( - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Column( - children: [ - if (kIsMacOS) const SizedBox(height: 25), - Row( - children: [ - Sidebar.brandLogo(), - const SizedBox(width: 10), - Text( - "Spotube", - style: theme.textTheme.titleLarge, - ), - ], - ), - ], - ), - ), - ); - } -} - -class SidebarFooter extends HookConsumerWidget { - const SidebarFooter({ - super.key, - }); - - @override - Widget build(BuildContext context, ref) { - final theme = Theme.of(context); - final mediaQuery = MediaQuery.of(context); - final me = ref.watch(meProvider); - final data = me.asData?.value; - - final avatarImg = (data?.images).asUrlString( - index: (data?.images?.length ?? 1) - 1, - placeholder: ImagePlaceholder.artist, - ); - - final auth = ref.watch(authenticationProvider); - - if (mediaQuery.mdAndDown) { - return IconButton( - icon: const Icon(SpotubeIcons.settings), - onPressed: () => ServiceUtils.navigateNamed(context, SettingsPage.name), - ); - } - - return Container( - padding: const EdgeInsets.only(left: 12), - width: 250, - child: Column( - children: [ - const ConnectDeviceButton.sidebar(), - const Gap(10), - Row( - mainAxisSize: MainAxisSize.min, - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - if (auth.asData?.value != null && data == null) - const CircularProgressIndicator() - else if (data != null) - Flexible( - child: InkWell( - onTap: () { - ServiceUtils.pushNamed(context, ProfilePage.name); - }, - borderRadius: BorderRadius.circular(30), - child: Row( - children: [ - CircleAvatar( - backgroundImage: - UniversalImage.imageProvider(avatarImg), - onBackgroundImageError: (exception, stackTrace) => - Assets.userPlaceholder.image( - height: 16, - width: 16, - ), - ), - const SizedBox(width: 10), - Flexible( - child: Text( - data.displayName ?? context.l10n.guest, - maxLines: 1, - softWrap: false, - overflow: TextOverflow.fade, - style: theme.textTheme.bodyMedium - ?.copyWith(fontWeight: FontWeight.bold), - ), - ), - ], - ), - ), - ), - IconButton( - icon: const Icon(SpotubeIcons.settings), - onPressed: () { - ServiceUtils.pushNamed(context, SettingsPage.name); - }, - ), - ], - ), - ], - ), - ); - } -} diff --git a/lib/modules/root/sidebar/sidebar.dart b/lib/modules/root/sidebar/sidebar.dart new file mode 100644 index 00000000..1538d624 --- /dev/null +++ b/lib/modules/root/sidebar/sidebar.dart @@ -0,0 +1,134 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +import 'package:spotube/collections/side_bar_tiles.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/root/sidebar/sidebar_footer.dart'; + +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; + +class Sidebar extends HookConsumerWidget { + final Widget child; + + const Sidebar({ + required this.child, + super.key, + }); + + @override + Widget build(BuildContext context, WidgetRef ref) { + final ThemeData(:colorScheme) = Theme.of(context); + final mediaQuery = MediaQuery.sizeOf(context); + + final layoutMode = + ref.watch(userPreferencesProvider.select((s) => s.layoutMode)); + + final sidebarTileList = useMemoized( + () => getSidebarTileList(context.l10n), + [context.l10n], + ); + + final sidebarLibraryTileList = useMemoized( + () => getSidebarLibraryTileList(context.l10n), + [context.l10n], + ); + + final tileList = [...sidebarTileList, ...sidebarLibraryTileList]; + + final router = context.watchRouter; + + final selectedIndex = tileList.indexWhere( + (e) => router.currentPath.startsWith(e.pathPrefix), + ); + + if (layoutMode == LayoutMode.compact || + (mediaQuery.smAndDown && layoutMode == LayoutMode.adaptive)) { + return child; + } + + final navigationButtons = [ + NavigationLabel( + child: mediaQuery.lgAndUp + ? DefaultTextStyle( + style: TextStyle( + fontFamily: "Cookie", + fontSize: 30, + letterSpacing: 1.8, + color: colorScheme.foreground, + ), + child: const Text("Spotube"), + ) + : const Text(""), + ), + for (final tile in sidebarTileList) + NavigationButton( + style: router.currentPath.startsWith(tile.pathPrefix) + ? const ButtonStyle.secondary() + : null, + label: mediaQuery.lgAndUp ? Text(tile.title) : null, + child: Tooltip( + tooltip: TooltipContainer(child: Text(tile.title)).call, + child: Icon(tile.icon), + ), + onPressed: () { + context.navigateTo(tile.route); + }, + ), + const NavigationDivider(), + if (mediaQuery.lgAndUp) + NavigationLabel(child: Text(context.l10n.library)), + for (final tile in sidebarLibraryTileList) + NavigationButton( + style: router.currentPath.startsWith(tile.pathPrefix) + ? const ButtonStyle.secondary() + : null, + label: mediaQuery.lgAndUp ? Text(tile.title) : null, + onPressed: () { + context.navigateTo(tile.route); + }, + child: Tooltip( + tooltip: TooltipContainer(child: Text(tile.title)).call, + child: Icon(tile.icon), + ), + ), + ]; + + return Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Column( + children: [ + Expanded( + child: mediaQuery.lgAndUp + ? NavigationSidebar( + index: selectedIndex, + onSelected: (index) { + final tile = tileList[index]; + context.navigateTo(tile.route); + }, + children: navigationButtons, + ) + : NavigationRail( + alignment: NavigationRailAlignment.start, + index: selectedIndex, + onSelected: (index) { + final tile = tileList[index]; + context.navigateTo(tile.route); + }, + children: navigationButtons, + ), + ), + const SidebarFooter(), + if (mediaQuery.lgAndUp) const Gap(130) else const Gap(65), + ], + ), + const VerticalDivider(), + Expanded(child: child), + ], + ); + } +} diff --git a/lib/modules/root/sidebar/sidebar_footer.dart b/lib/modules/root/sidebar/sidebar_footer.dart new file mode 100644 index 00000000..0f8ac9d8 --- /dev/null +++ b/lib/modules/root/sidebar/sidebar_footer.dart @@ -0,0 +1,144 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/material.dart' show Badge; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/image/universal_image.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/connect/connect_device.dart'; +import 'package:spotube/provider/download_manager_provider.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; + +class SidebarFooter extends HookConsumerWidget implements NavigationBarItem { + const SidebarFooter({ + super.key, + }); + + @override + Widget build(BuildContext context, ref) { + final theme = Theme.of(context); + final router = AutoRouter.of(context, watch: true); + final mediaQuery = MediaQuery.of(context); + final downloadCount = ref + .watch(downloadManagerProvider) + .where((e) => + e.status == DownloadStatus.downloading || + e.status == DownloadStatus.queued) + .length; + final userSnapshot = ref.watch(metadataPluginUserProvider); + final data = userSnapshot.asData?.value; + + final avatarImg = (data?.images).asUrlString( + index: (data?.images.length ?? 1) - 1, + placeholder: ImagePlaceholder.artist, + ); + + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); + + if (mediaQuery.mdAndDown) { + return Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Badge( + isLabelVisible: downloadCount > 0, + label: Text(downloadCount.toString()), + child: IconButton( + variance: router.topRoute.name == UserDownloadsRoute.name + ? ButtonVariance.secondary + : ButtonVariance.ghost, + icon: const Icon(SpotubeIcons.download), + onPressed: () => context.navigateTo(const UserDownloadsRoute()), + ), + ), + const ConnectDeviceButton.sidebar(), + IconButton( + variance: ButtonVariance.ghost, + icon: const Icon(SpotubeIcons.settings), + onPressed: () => context.navigateTo(const SettingsRoute()), + ), + ], + ); + } + + return Container( + padding: const EdgeInsets.only(left: 12), + width: 180, + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + SizedBox( + width: double.infinity, + child: Button( + style: router.topRoute.name == UserDownloadsRoute.name + ? ButtonVariance.secondary + : ButtonVariance.outline, + onPressed: () { + context.navigateTo(const UserDownloadsRoute()); + }, + leading: const Icon(SpotubeIcons.download), + trailing: downloadCount > 0 + ? PrimaryBadge( + child: Text(downloadCount.toString()), + ) + : null, + child: Text(context.l10n.downloads), + ), + ), + const ConnectDeviceButton.sidebar(), + Row( + mainAxisSize: MainAxisSize.min, + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + if (authenticated.asData?.value == true && data == null) + const CircularProgressIndicator() + else if (data != null) + Flexible( + child: GestureDetector( + onTap: () { + context.navigateTo(const ProfileRoute()); + }, + child: Row( + children: [ + Avatar( + initials: Avatar.getInitials(data.name), + provider: UniversalImage.imageProvider(avatarImg), + ), + const SizedBox(width: 10), + Flexible( + child: Text( + data.name, + maxLines: 1, + softWrap: false, + overflow: TextOverflow.fade, + style: theme.typography.normal + .copyWith(fontWeight: FontWeight.bold), + ), + ), + ], + ), + ), + ), + IconButton( + variance: ButtonVariance.ghost, + icon: const Icon(SpotubeIcons.settings), + onPressed: () { + context.navigateTo(const SettingsRoute()); + }, + ), + ], + ), + ], + ), + ); + } + + @override + bool get selectable => false; +} diff --git a/lib/modules/root/spotube_navigation_bar.dart b/lib/modules/root/spotube_navigation_bar.dart index 978891b8..47ea3ca3 100644 --- a/lib/modules/root/spotube_navigation_bar.dart +++ b/lib/modules/root/spotube_navigation_bar.dart @@ -1,21 +1,19 @@ -import 'dart:ui'; +import 'dart:math'; -import 'package:curved_navigation_bar/curved_navigation_bar.dart'; -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/material.dart' show Badge; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/side_bar_tiles.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/hooks/utils/use_brightness_value.dart'; import 'package:spotube/models/database/database.dart'; import 'package:spotube/provider/download_manager_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:spotube/utils/service_utils.dart'; - final navigationPanelHeight = StateProvider((ref) => 50); class SpotubeNavigationBar extends HookConsumerWidget { @@ -25,19 +23,17 @@ class SpotubeNavigationBar extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final theme = Theme.of(context); - final routerState = GoRouterState.of(context); - - final downloadCount = ref.watch(downloadManagerProvider).$downloadCount; final mediaQuery = MediaQuery.of(context); + + final downloadCount = ref + .watch(downloadManagerProvider) + .where((e) => + e.status == DownloadStatus.downloading || + e.status == DownloadStatus.queued) + .length; final layoutMode = ref.watch(userPreferencesProvider.select((s) => s.layoutMode)); - final buttonColor = useBrightnessValue( - theme.colorScheme.inversePrimary, - theme.colorScheme.primary.withOpacity(0.2), - ); - final navbarTileList = useMemoized( () => getNavbarTileList(context.l10n), [context.l10n], @@ -45,13 +41,13 @@ class SpotubeNavigationBar extends HookConsumerWidget { final panelHeight = ref.watch(navigationPanelHeight); - final selectedIndex = useMemoized(() { - final index = navbarTileList.indexWhere( - (e) => routerState.namedLocation(e.name) == routerState.matchedLocation, - ); - - return index == -1 ? 0 : index; - }, [navbarTileList, routerState.matchedLocation]); + final router = context.watchRouter; + final selectedIndex = max( + 0, + navbarTileList.indexWhere( + (e) => router.currentPath.startsWith(e.pathPrefix), + ), + ); if (layoutMode == LayoutMode.extended || (mediaQuery.mdAndUp && layoutMode == LayoutMode.adaptive) || @@ -62,40 +58,32 @@ class SpotubeNavigationBar extends HookConsumerWidget { return AnimatedContainer( duration: const Duration(milliseconds: 100), height: panelHeight, - child: ClipRect( - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 15, sigmaY: 15), - child: CurvedNavigationBar( - backgroundColor: - theme.colorScheme.secondaryContainer.withOpacity(0.72), - buttonBackgroundColor: buttonColor, - color: theme.colorScheme.surface, - height: panelHeight, - animationDuration: const Duration(milliseconds: 350), - items: navbarTileList.map( - (e) { - /// Using this [Builder] as an workaround for the first item's - /// icon color not updating unless navigating to another page - return Builder(builder: (context) { - return MouseRegion( - cursor: SystemMouseCursors.click, + child: SingleChildScrollView( + child: Column( + children: [ + const Divider(), + NavigationBar( + index: selectedIndex, + surfaceBlur: context.theme.surfaceBlur, + surfaceOpacity: context.theme.surfaceOpacity, + children: [ + for (final tile in navbarTileList) + NavigationButton( + style: navbarTileList[selectedIndex] == tile + ? const ButtonStyle.fixed(density: ButtonDensity.icon) + : const ButtonStyle.muted(density: ButtonDensity.icon), child: Badge( - isLabelVisible: e.id == "library" && downloadCount > 0, + isLabelVisible: tile.id == "library" && downloadCount > 0, label: Text(downloadCount.toString()), - child: Icon( - e.icon, - color: Theme.of(context).colorScheme.primary, - ), + child: Icon(tile.icon), ), - ); - }); - }, - ).toList(), - index: selectedIndex, - onTap: (i) { - ServiceUtils.navigateNamed(context, navbarTileList[i].name); - }, - ), + onPressed: () { + context.navigateTo(tile.route); + }, + ) + ], + ), + ], ), ), ); diff --git a/lib/modules/root/update_dialog.dart b/lib/modules/root/update_dialog.dart index 27b857df..4aa2fd13 100644 --- a/lib/modules/root/update_dialog.dart +++ b/lib/modules/root/update_dialog.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/links/anchor_button.dart'; import 'package:url_launcher/url_launcher_string.dart'; import 'package:spotube/extensions/context.dart'; @@ -19,7 +19,7 @@ class RootAppUpdateDialog extends StatelessWidget { return AlertDialog( title: Text(context.l10n.spotube_has_an_update), actions: [ - FilledButton( + Button.primary( child: Text(context.l10n.download_now), onPressed: () => launchUrlString( nightlyBuildNum != null ? nightlyUrl : url, diff --git a/lib/modules/root/use_global_subscriptions.dart b/lib/modules/root/use_global_subscriptions.dart new file mode 100644 index 00000000..9a492d31 --- /dev/null +++ b/lib/modules/root/use_global_subscriptions.dart @@ -0,0 +1,146 @@ +import 'dart:async'; + +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/metadata_plugins/plugin_update_available_dialog.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/updater/update_checker.dart'; +import 'package:spotube/provider/server/routes/connect.dart'; +import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/services/connectivity_adapter.dart'; +import 'package:spotube/utils/service_utils.dart'; + +void useGlobalSubscriptions(WidgetRef ref) { + final context = useContext(); + final theme = Theme.of(context); + final connectRoutes = ref.watch(serverConnectRoutesProvider); + + useEffect(() { + WidgetsBinding.instance.addPostFrameCallback((_) async { + ServiceUtils.checkForUpdates(context, ref); + + final pluginUpdate = + await ref.read(metadataPluginUpdateCheckerProvider.future); + + if (pluginUpdate != null) { + final pluginConfig = await ref.read(metadataPluginsProvider.future); + if (context.mounted) { + showDialog( + context: context, + builder: (context) => MetadataPluginUpdateAvailableDialog( + plugin: pluginConfig.defaultMetadataPluginConfig!, + update: pluginUpdate, + ), + ); + } + } + }); + + StreamSubscription? audioPlayerSubscription; + bool pausedByStream = false; + + final subscriptions = [ + ConnectionCheckerService.instance.onConnectivityChanged + .listen((connected) async { + audioPlayerSubscription?.cancel(); + + /// Pausing or resuming based on connectivity to avoid MPV skipping + /// audio while retrying to connect + if (audioPlayer.currentIndex >= 0) { + if (connected && audioPlayer.isPaused && pausedByStream) { + await audioPlayer.resume(); + pausedByStream = false; + } else if (!connected && audioPlayer.isPlaying) { + if ((audioPlayer.bufferedPosition - const Duration(seconds: 1)) <= + audioPlayer.position) { + await audioPlayer.pause(); + pausedByStream = true; + } else { + audioPlayerSubscription = + audioPlayer.positionStream.listen((position) async { + if (ConnectionCheckerService.instance.isConnectedSync) return; + + final bufferedPosition = + audioPlayer.bufferedPosition - const Duration(seconds: 1); + final duration = + audioPlayer.duration - const Duration(seconds: 1); + + if (bufferedPosition <= position || position >= duration) { + audioPlayer.pause(); + pausedByStream = true; + } + }); + } + } + } + + // Show notification for connection related issues + if (!context.mounted) return; + + showToast( + context: context, + location: ToastLocation.bottomCenter, + builder: (context, overlay) { + if (connected) { + return SurfaceCard( + child: Basic( + leading: const Icon(SpotubeIcons.wifi), + title: Text(context.l10n.connection_restored), + ), + ); + } + + return SurfaceCard( + fillColor: theme.colorScheme.destructive, + filled: true, + child: Basic( + leading: Icon( + SpotubeIcons.noWifi, + color: theme.colorScheme.destructiveForeground, + ), + trailing: Text( + context.l10n.you_are_offline, + style: TextStyle( + color: theme.colorScheme.destructiveForeground, + ), + ), + ), + ); + }, + ); + }), + connectRoutes.connectClientStream.listen((clientOrigin) { + if (!context.mounted) return; + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + fillColor: Colors.yellow[600], + filled: true, + child: Basic( + leading: const Icon( + SpotubeIcons.error, + color: Colors.black, + ), + title: Text( + context.l10n.connect_client_alert(clientOrigin), + style: const TextStyle(color: Colors.black), + ), + ), + ); + }, + ); + }) + ]; + + return () { + for (final subscription in subscriptions) { + subscription.cancel(); + } + }; + }, []); +} diff --git a/lib/modules/search/loading.dart b/lib/modules/search/loading.dart new file mode 100644 index 00000000..8ca2820f --- /dev/null +++ b/lib/modules/search/loading.dart @@ -0,0 +1,68 @@ +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/pages/search/search.dart'; + +class SearchPlaceholder extends HookConsumerWidget { + final AsyncValue snapshot; + final Widget child; + const SearchPlaceholder({ + super.key, + required this.child, + required this.snapshot, + }); + + @override + Widget build(BuildContext context, ref) { + final theme = context.theme; + final mediaQuery = MediaQuery.sizeOf(context); + + final searchTerm = ref.watch(searchTermStateProvider); + + return switch ((searchTerm.isEmpty, snapshot.isLoading)) { + (true, false) => Column( + children: [ + SizedBox( + height: mediaQuery.height * 0.2, + ), + Undraw( + illustration: UndrawIllustration.explore, + color: theme.colorScheme.primary, + height: 200 * theme.scaling, + ), + const SizedBox(height: 20), + Text(context.l10n.search_to_get_results).large(), + ], + ), + (false, true) => Container( + constraints: BoxConstraints( + maxWidth: + mediaQuery.lgAndUp ? mediaQuery.width * 0.5 : mediaQuery.width, + ), + padding: const EdgeInsets.symmetric( + horizontal: 20, + ), + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Text( + context.l10n.crunching_results, + style: TextStyle( + fontSize: 20, + fontWeight: FontWeight.w900, + color: theme.colorScheme.foreground.withValues(alpha: 0.7), + ), + ), + const SizedBox(height: 20), + const LinearProgressIndicator(), + ], + ), + ), + _ => child, + }; + } +} diff --git a/lib/modules/search/sections/albums.dart b/lib/modules/search/sections/albums.dart new file mode 100644 index 00000000..e8bc71fc --- /dev/null +++ b/lib/modules/search/sections/albums.dart @@ -0,0 +1,28 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/metadata_plugin/search/all.dart'; + +class SearchAlbumsSection extends HookConsumerWidget { + const SearchAlbumsSection({ + super.key, + }); + + @override + Widget build(BuildContext context, ref) { + final searchTerm = ref.watch(searchTermStateProvider); + final search = ref.watch(metadataPluginSearchAllProvider(searchTerm)); + final albums = search.asData?.value.albums ?? []; + + return HorizontalPlaybuttonCardView( + isLoadingNextPage: false, + hasNextPage: false, + items: albums, + onFetchMore: () {}, + title: Text(context.l10n.albums), + ); + } +} diff --git a/lib/modules/search/sections/artists.dart b/lib/modules/search/sections/artists.dart new file mode 100644 index 00000000..9da3702c --- /dev/null +++ b/lib/modules/search/sections/artists.dart @@ -0,0 +1,29 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; + +import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/metadata_plugin/search/all.dart'; + +class SearchArtistsSection extends HookConsumerWidget { + const SearchArtistsSection({ + super.key, + }); + + @override + Widget build(BuildContext context, ref) { + final searchTerm = ref.watch(searchTermStateProvider); + final search = ref.watch(metadataPluginSearchAllProvider(searchTerm)); + + final artists = search.asData?.value.artists ?? []; + + return HorizontalPlaybuttonCardView( + isLoadingNextPage: false, + hasNextPage: false, + items: artists, + onFetchMore: () {}, + title: Text(context.l10n.artists), + ); + } +} diff --git a/lib/modules/search/sections/playlists.dart b/lib/modules/search/sections/playlists.dart new file mode 100644 index 00000000..7e03bdeb --- /dev/null +++ b/lib/modules/search/sections/playlists.dart @@ -0,0 +1,28 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/metadata_plugin/search/all.dart'; + +class SearchPlaylistsSection extends HookConsumerWidget { + const SearchPlaylistsSection({ + super.key, + }); + + @override + Widget build(BuildContext context, ref) { + final searchTerm = ref.watch(searchTermStateProvider); + final playlistsQuery = + ref.watch(metadataPluginSearchAllProvider(searchTerm)); + final playlists = playlistsQuery.asData?.value.playlists ?? []; + + return HorizontalPlaybuttonCardView( + isLoadingNextPage: false, + hasNextPage: false, + items: playlists, + onFetchMore: () {}, + title: Text(context.l10n.playlists), + ); + } +} diff --git a/lib/pages/search/sections/tracks.dart b/lib/modules/search/sections/tracks.dart similarity index 76% rename from lib/pages/search/sections/tracks.dart rename to lib/modules/search/sections/tracks.dart index 6ec8f685..6bc60045 100644 --- a/lib/pages/search/sections/tracks.dart +++ b/lib/modules/search/sections/tracks.dart @@ -1,15 +1,16 @@ import 'package:collection/collection.dart'; -import 'package:flutter/material.dart' hide Page; + import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/dialogs/prompt_dialog.dart'; import 'package:spotube/components/dialogs/select_device_dialog.dart'; import 'package:spotube/components/track_tile/track_tile.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/pages/search/search.dart'; import 'package:spotube/provider/connect/connect.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/search/all.dart'; class SearchTracksSection extends HookConsumerWidget { const SearchTracksSection({ @@ -18,12 +19,9 @@ class SearchTracksSection extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final searchTrack = ref.watch(searchProvider(SearchType.track)); - - final searchTrackNotifier = - ref.watch(searchProvider(SearchType.track).notifier); - - final tracks = searchTrack.asData?.value.items.cast() ?? []; + final searchTerm = ref.watch(searchTermStateProvider); + final search = ref.watch(metadataPluginSearchAllProvider(searchTerm)); + final tracks = search.asData?.value.tracks ?? []; final playlistNotifier = ref.watch(audioPlayerProvider.notifier); final playlist = ref.watch(audioPlayerProvider); final theme = Theme.of(context); @@ -37,13 +35,11 @@ class SearchTracksSection extends HookConsumerWidget { padding: const EdgeInsets.symmetric(horizontal: 8), child: Text( context.l10n.songs, - style: theme.textTheme.titleLarge!, + style: theme.typography.h4, ), ), - if (searchTrack.isLoading) + if (search.isLoading) const CircularProgressIndicator() - else if (searchTrack.hasError) - Text(searchTrack.error.toString()) else ...tracks.mapIndexed((i, track) { return TrackTile( @@ -54,6 +50,8 @@ class SearchTracksSection extends HookConsumerWidget { final isRemoteDevice = await showSelectDeviceDialog(context, ref); + if (isRemoteDevice == null) return; + if (isRemoteDevice) { final remotePlayback = ref.read(connectProvider.notifier); final remotePlaylist = ref.read(queueProvider); @@ -66,7 +64,7 @@ class SearchTracksSection extends HookConsumerWidget { ? await showPromptDialog( context: context, title: context.l10n.playing_track( - track.name!, + track.name, ), message: context.l10n.queue_clear_alert( playlist.tracks.length, @@ -89,7 +87,7 @@ class SearchTracksSection extends HookConsumerWidget { ? await showPromptDialog( context: context, title: context.l10n.playing_track( - track.name!, + track.name, ), message: context.l10n.queue_clear_alert( playlist.tracks.length, @@ -108,17 +106,6 @@ class SearchTracksSection extends HookConsumerWidget { }, ); }), - if (searchTrack.asData?.value.hasMore == true && tracks.isNotEmpty) - Center( - child: TextButton( - onPressed: searchTrack.isLoadingNextPage - ? null - : searchTrackNotifier.fetchMore, - child: searchTrack.isLoadingNextPage - ? const CircularProgressIndicator() - : Text(context.l10n.load_more), - ), - ) ], ); } diff --git a/lib/modules/settings/color_scheme_picker_dialog.dart b/lib/modules/settings/color_scheme_picker_dialog.dart index f2933505..9469ff00 100644 --- a/lib/modules/settings/color_scheme_picker_dialog.dart +++ b/lib/modules/settings/color_scheme_picker_dialog.dart @@ -1,10 +1,11 @@ -import 'package:flutter/material.dart'; +import 'package:collection/collection.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:system_theme/system_theme.dart'; class SpotubeColor extends Color { final String name; @@ -20,28 +21,38 @@ class SpotubeColor extends Color { @override String toString() { - return "$name:$value"; + return "$name:${toARGB32()}"; } } final Set colorsMap = { - SpotubeColor(SystemTheme.accentColor.accent.value, name: "System"), - SpotubeColor(Colors.red.value, name: "Red"), - SpotubeColor(Colors.pink.value, name: "Pink"), - SpotubeColor(Colors.purple.value, name: "Purple"), - SpotubeColor(Colors.deepPurple.value, name: "DeepPurple"), - SpotubeColor(Colors.indigo.value, name: "Indigo"), - SpotubeColor(Colors.blue.value, name: "Blue"), - SpotubeColor(Colors.lightBlue.value, name: "LightBlue"), - SpotubeColor(Colors.cyan.value, name: "Cyan"), - SpotubeColor(Colors.teal.value, name: "Teal"), - SpotubeColor(Colors.green.value, name: "Green"), - SpotubeColor(Colors.lightGreen.value, name: "LightGreen"), - SpotubeColor(Colors.yellow.value, name: "Yellow"), - SpotubeColor(Colors.amber.value, name: "Amber"), - SpotubeColor(Colors.orange.value, name: "Orange"), - SpotubeColor(Colors.deepOrange.value, name: "DeepOrange"), - SpotubeColor(Colors.brown.value, name: "Brown"), + SpotubeColor(Colors.slate.value, name: "slate"), + SpotubeColor(Colors.gray.value, name: "gray"), + SpotubeColor(Colors.zinc.value, name: "zinc"), + SpotubeColor(Colors.neutral.value, name: "neutral"), + SpotubeColor(Colors.stone.value, name: "stone"), + SpotubeColor(Colors.red.value, name: "red"), + SpotubeColor(Colors.orange.value, name: "orange"), + SpotubeColor(Colors.yellow.value, name: "yellow"), + SpotubeColor(Colors.green.value, name: "green"), + SpotubeColor(Colors.blue.value, name: "blue"), + SpotubeColor(Colors.violet.value, name: "violet"), + SpotubeColor(Colors.rose.value, name: "rose"), +}; + +final colorSchemeMap = { + "slate": LegacyColorSchemes.slate, + "gray": LegacyColorSchemes.gray, + "zinc": LegacyColorSchemes.zinc, + "neutral": LegacyColorSchemes.neutral, + "stone": LegacyColorSchemes.stone, + "red": LegacyColorSchemes.red, + "orange": LegacyColorSchemes.orange, + "yellow": LegacyColorSchemes.yellow, + "green": LegacyColorSchemes.green, + "blue": LegacyColorSchemes.blue, + "violet": LegacyColorSchemes.violet, + "rose": LegacyColorSchemes.rose, }; class ColorSchemePickerDialog extends HookConsumerWidget { @@ -51,180 +62,93 @@ class ColorSchemePickerDialog extends HookConsumerWidget { Widget build(BuildContext context, ref) { final preferences = ref.watch(userPreferencesProvider); final preferencesNotifier = ref.watch(userPreferencesProvider.notifier); - final scheme = preferences.accentColorScheme; - final active = useState(colorsMap.firstWhere( - (element) { - return scheme.name == element.name; - }, - ).name); - onOk() { - preferencesNotifier.setAccentColorScheme( - colorsMap.firstWhere( - (element) { - return element.name == active.value; - }, - ), - ); - Navigator.pop(context); - } + final scheme = preferences.accentColorScheme; + final active = useState( + colorsMap.firstWhereOrNull( + (element) { + return scheme.name == element.name; + }, + )?.name, + ); return AlertDialog( - title: Text(context.l10n.pick_color_scheme), + title: Text( + context.l10n.pick_color_scheme, + style: TextStyle(color: context.theme.colorScheme.foreground), + ).large(), actions: [ - OutlinedButton( + Button.outline( child: Text(context.l10n.cancel), onPressed: () { Navigator.pop(context); }, ), - FilledButton( - onPressed: onOk, + Button.primary( + onPressed: () { + Navigator.pop(context); + }, child: Text(context.l10n.save), ), ], content: SizedBox( height: 200, width: 400, - child: ListView.separated( - separatorBuilder: (context, index) { - return const SizedBox(height: 10); - }, - itemCount: colorsMap.length, - itemBuilder: (context, index) { - final color = colorsMap.elementAt(index); - return ColorTile( - color: color, - isActive: active.value == color.name, - onPressed: () { - active.value = color.name; - }, - tooltip: color.name, - ); - }, + child: Wrap( + spacing: 8, + runSpacing: 8, + children: colorsMap.map( + (color) { + return ColorChip( + name: color.name, + color: color, + isActive: color.name == active.value, + onPressed: () { + active.value = color.name; + preferencesNotifier.setAccentColorScheme( + colorsMap.firstWhere( + (element) { + return element.name == color.name; + }, + ), + ); + }, + ); + }, + ).toList(), ), ), ); } } -class ColorTile extends StatelessWidget { +class ColorChip extends StatelessWidget { + final String name; final Color color; final bool isActive; - final void Function()? onPressed; - final String? tooltip; - final bool isCompact; - const ColorTile({ - required this.color, - this.isActive = false, - this.onPressed, - this.tooltip = "", - this.isCompact = false, + final VoidCallback onPressed; + const ColorChip({ super.key, + required this.name, + required this.color, + required this.isActive, + required this.onPressed, }); - factory ColorTile.compact({ - required Color color, - bool isActive = false, - void Function()? onPressed, - String? tooltip = "", - Key? key, - }) { - return ColorTile( - color: color, - isActive: isActive, - onPressed: onPressed, - tooltip: tooltip, - isCompact: true, - key: key, - ); - } - @override Widget build(BuildContext context) { - final theme = Theme.of(context); - - final lead = Container( - height: 40, - width: 40, - decoration: BoxDecoration( - border: isActive - ? Border.fromBorderSide( - BorderSide( - color: Color.lerp( - theme.colorScheme.primary, - theme.colorScheme.onPrimary, - 0.5, - )!, - width: 4, - ), - ) - : null, - borderRadius: BorderRadius.circular(15), - color: color, - ), - ); - - if (isCompact) { - return GestureDetector( - onTap: onPressed, - child: lead, - ); - } - - final colorScheme = ColorScheme.fromSeed(seedColor: color); - - final palette = [ - colorScheme.primary, - colorScheme.inversePrimary, - colorScheme.primaryContainer, - colorScheme.secondary, - colorScheme.secondaryContainer, - colorScheme.surface, - colorScheme.surface, - colorScheme.surfaceContainerHighest, - colorScheme.onPrimary, - colorScheme.onSurface, - ]; - - return GestureDetector( - onTap: onPressed, - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Row( - children: [ - lead, - const SizedBox(width: 10), - Text( - tooltip!, - style: theme.textTheme.bodyLarge?.copyWith( - color: theme.colorScheme.primary, - fontWeight: FontWeight.w600, - ), - ), - ], - ), - const SizedBox(height: 10), - Wrap( - alignment: WrapAlignment.start, - spacing: 10, - runSpacing: 10, - children: [ - ...palette.map( - (e) => Container( - height: 20, - width: 20, - decoration: BoxDecoration( - color: e, - borderRadius: BorderRadius.circular(5), - ), - ), - ), - ], - ), - ], + return Chip( + leading: Container( + width: 20, + height: 20, + decoration: BoxDecoration( + color: color, + borderRadius: BorderRadius.circular(10), + ), ), + onPressed: onPressed, + style: isActive ? ButtonVariance.primary : ButtonVariance.outline, + child: Text(name), ); } } diff --git a/lib/modules/settings/playback/edit_connect_port_dialog.dart b/lib/modules/settings/playback/edit_connect_port_dialog.dart new file mode 100644 index 00000000..587f4388 --- /dev/null +++ b/lib/modules/settings/playback/edit_connect_port_dialog.dart @@ -0,0 +1,97 @@ +import 'package:flutter/services.dart'; +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:form_builder_validators/form_builder_validators.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/form/text_form_field.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart'; +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; + +class SettingsPlaybackEditConnectPortDialog extends HookConsumerWidget { + const SettingsPlaybackEditConnectPortDialog({super.key}); + + @override + Widget build(BuildContext context, ref) { + final connectPort = ref.watch( + userPreferencesProvider.select((s) => s.connectPort), + ); + final controller = useShadcnTextEditingController( + text: connectPort.toString(), + ); + final formKey = useMemoized(() => GlobalKey(), []); + + return ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 400), + child: Alert( + title: Text(context.l10n.edit_port).h4(), + content: FormBuilder( + key: formKey, + child: Column( + children: [ + const Gap(10), + TextFormBuilderField( + name: "port", + controller: controller, + placeholder: const Text("3000"), + validator: FormBuilderValidators.integer(radix: 10), + keyboardType: TextInputType.number, + inputFormatters: [ + // Allow only signed integers + TextInputFormatter.withFunction( + (oldValue, newValue) { + if (newValue.text.isEmpty) { + return const TextEditingValue(); + } + if (newValue.text.length == 1 && newValue.text == "-") { + return newValue; + } + + final intValue = int.tryParse(newValue.text); + if (intValue == null) { + return oldValue; + } + return newValue; + }, + ), + ], + ), + const Gap(5), + Text(context.l10n.port_helper_msg).small.muted, + const Gap(20), + Row( + children: [ + Expanded( + child: Button.secondary( + onPressed: () { + Navigator.of(context).pop(); + }, + child: Text(context.l10n.cancel), + ), + ), + const Gap(10), + Expanded( + child: Button.primary( + onPressed: () { + if (!formKey.currentState!.saveAndValidate()) { + return; + } + final port = int.parse(controller.text); + ref + .read(userPreferencesProvider.notifier) + .setConnectPort(port); + Navigator.of(context).pop(); + }, + child: Text(context.l10n.save), + ), + ), + ], + ), + ], + ), + ), + ), + ); + } +} diff --git a/lib/modules/settings/playback/edit_instance_url_dialog.dart b/lib/modules/settings/playback/edit_instance_url_dialog.dart new file mode 100644 index 00000000..b2dda411 --- /dev/null +++ b/lib/modules/settings/playback/edit_instance_url_dialog.dart @@ -0,0 +1,75 @@ +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:form_builder_validators/form_builder_validators.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/form/text_form_field.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart'; + +class SettingsPlaybackEditInstanceUrlDialog extends HookConsumerWidget { + final String title; + final String? initialValue; + final ValueChanged onSave; + + const SettingsPlaybackEditInstanceUrlDialog({ + super.key, + required this.title, + required this.onSave, + this.initialValue, + }); + + @override + Widget build(BuildContext context, ref) { + final controller = useShadcnTextEditingController( + text: initialValue, + ); + final formKey = useMemoized(() => GlobalKey(), []); + + return Alert( + title: Text(title).h4(), + content: FormBuilder( + key: formKey, + child: Column( + children: [ + const Gap(10), + TextFormBuilderField( + name: "url", + controller: controller, + placeholder: Text(title), + validator: FormBuilderValidators.url(), + ), + const Gap(10), + Row( + children: [ + Expanded( + child: Button.secondary( + onPressed: () { + Navigator.of(context).pop(); + }, + child: Text(context.l10n.cancel), + ), + ), + const Gap(10), + Expanded( + child: Button.primary( + onPressed: () { + if (!formKey.currentState!.saveAndValidate()) { + return; + } + onSave( + controller.text, + ); + Navigator.of(context).pop(); + }, + child: Text(context.l10n.save), + ), + ), + ], + ) + ], + ), + ), + ); + } +} diff --git a/lib/modules/settings/section_card_with_heading.dart b/lib/modules/settings/section_card_with_heading.dart index 87060579..c7bc1f26 100644 --- a/lib/modules/settings/section_card_with_heading.dart +++ b/lib/modules/settings/section_card_with_heading.dart @@ -1,4 +1,6 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' show ListTileTheme, ListTileThemeData; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide Theme, ThemeData; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; class SectionCardWithHeading extends StatelessWidget { final String heading; @@ -11,27 +13,43 @@ class SectionCardWithHeading extends StatelessWidget { @override Widget build(BuildContext context) { - return Column( - crossAxisAlignment: CrossAxisAlignment.start, - mainAxisSize: MainAxisSize.min, - children: [ - Padding( - padding: const EdgeInsets.symmetric(horizontal: 8.0), - child: Text( - heading, - style: Theme.of(context).textTheme.titleLarge?.copyWith( - fontWeight: FontWeight.w600, - ), + return ListTileTheme( + data: ListTileThemeData( + shape: RoundedRectangleBorder( + borderRadius: context.theme.borderRadiusLg, + side: BorderSide( + color: context.theme.colorScheme.border, + width: .5, ), ), - Padding( - padding: const EdgeInsets.all(8.0), - child: Card( - clipBehavior: Clip.antiAliasWithSaveLayer, - child: Column(mainAxisSize: MainAxisSize.min, children: children), + textColor: context.theme.colorScheme.foreground, + iconColor: context.theme.colorScheme.foreground, + selectedColor: context.theme.colorScheme.accent, + subtitleTextStyle: context.theme.typography.xSmall, + ), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + Padding( + padding: const EdgeInsets.symmetric(horizontal: 8.0), + child: Text( + heading, + style: context.theme.typography.large.copyWith( + color: context.theme.colorScheme.foreground, + ), + ), ), - ), - ], + Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.stretch, + children: children, + ).gap(8.0), + ), + ], + ), ); } } diff --git a/lib/modules/settings/youtube_engine_not_installed_dialog.dart b/lib/modules/settings/youtube_engine_not_installed_dialog.dart new file mode 100644 index 00000000..b993dd1b --- /dev/null +++ b/lib/modules/settings/youtube_engine_not_installed_dialog.dart @@ -0,0 +1,122 @@ +import 'dart:io'; + +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/form/text_form_field.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/services/kv_store/kv_store.dart'; +import 'package:spotube/utils/platform.dart'; +import 'package:url_launcher/url_launcher.dart'; +import 'package:yt_dlp_dart/yt_dlp_dart.dart'; + +const engineDownloadUrls = { + YoutubeClientEngine.ytDlp: + "https://github.com/yt-dlp/yt-dlp?tab=readme-ov-file#installation", +}; + +class YouTubeEngineNotInstalledDialog extends HookConsumerWidget { + final YoutubeClientEngine engine; + const YouTubeEngineNotInstalledDialog({ + super.key, + required this.engine, + }); + + @override + Widget build(BuildContext context, ref) { + final controller = useShadcnTextEditingController(); + final formKey = useMemoized(() => GlobalKey(), []); + + return AlertDialog( + title: Row( + spacing: 8, + mainAxisSize: MainAxisSize.min, + children: [ + const Icon(SpotubeIcons.error, color: Colors.red), + Text( + context.l10n.youtube_engine_not_installed_title(engine.label), + style: const TextStyle(color: Colors.red), + ), + ], + ), + content: ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 400), + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 8, + children: [ + Text( + context.l10n.youtube_engine_not_installed_message(engine.label), + ), + if (engineDownloadUrls[engine] != null) + Row( + mainAxisSize: MainAxisSize.min, + children: [ + Text("${context.l10n.download}:"), + Button.link( + child: Text(engineDownloadUrls[engine]!.split("?").first), + onPressed: () async { + launchUrl(Uri.parse(engineDownloadUrls[engine]!)); + }, + ), + ], + ), + Text(context.l10n.youtube_engine_set_path(engine.label)), + const Gap(8), + FormBuilder( + key: formKey, + child: TextFormBuilderField( + name: "path", + controller: controller, + placeholder: Text(switch (context.theme.platform) { + TargetPlatform.macOS => "e.g. /opt/homebrew/bin/yt-dlp", + TargetPlatform.windows => + r"e.g. C:\Program Files\yt-dlp\yt-dlp.exe", + _ => "e.g. /home/user/.local/bin/yt-dlp", + }), + ), + ), + if (kIsMacOS || kIsLinux) + Text(context.l10n.youtube_engine_unix_issue_message), + ], + ), + ), + actions: [ + Button.text( + onPressed: () { + if (!context.mounted) return; + Navigator.of(context).pop(false); + }, + child: Text(context.l10n.cancel), + ), + Button.secondary( + onPressed: () async { + if (controller.text.isNotEmpty) { + if (!await File(controller.text).exists() && context.mounted) { + formKey.currentState?.fields["path"] + ?.invalidate(context.l10n.file_not_found); + return; + } + await KVStoreService.setYoutubeEnginePath( + engine, + controller.text, + ); + if (engine == YoutubeClientEngine.ytDlp) { + await YtDlp.instance.setBinaryLocation(controller.text); + } + } + if (!context.mounted) return; + Navigator.of(context).pop(true); + }, + child: Text(context.l10n.save), + ), + ], + ); + } +} diff --git a/lib/modules/stats/common/album_item.dart b/lib/modules/stats/common/album_item.dart index eec68717..2ac73b91 100644 --- a/lib/modules/stats/common/album_item.dart +++ b/lib/modules/stats/common/album_item.dart @@ -1,21 +1,21 @@ -import 'package:flutter/material.dart'; -import 'package:spotify/spotify.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/components/links/artist_link.dart'; +import 'package:spotube/components/ui/button_tile.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/album/album_card.dart'; import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/links/artist_link.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/album/album.dart'; -import 'package:spotube/utils/service_utils.dart'; class StatsAlbumItem extends StatelessWidget { - final AlbumSimple album; + final SpotubeSimpleAlbumObject album; final Widget info; const StatsAlbumItem({super.key, required this.album, required this.info}); @override Widget build(BuildContext context) { - return ListTile( - horizontalTitleGap: 8, + return ButtonTile( + style: ButtonVariance.ghost, leading: ClipRRect( borderRadius: BorderRadius.circular(4), child: UniversalImage( @@ -26,34 +26,24 @@ class StatsAlbumItem extends StatelessWidget { height: 40, ), ), - title: Text(album.name!), + title: Text(album.name), subtitle: Row( mainAxisSize: MainAxisSize.min, children: [ - Text("${album.albumType?.formatted} • "), + Text("${album.albumType.formatted} • "), Flexible( child: ArtistLink( - artists: album.artists ?? [], + artists: album.artists, mainAxisAlignment: WrapAlignment.start, - onOverflowArtistClick: () => ServiceUtils.pushNamed( - context, - AlbumPage.name, - pathParameters: { - "id": album.id!, - }, - ), + onOverflowArtistClick: () => + context.navigateTo(AlbumRoute(id: album.id, album: album)), ), ), ], ), trailing: info, - onTap: () { - ServiceUtils.pushNamed( - context, - AlbumPage.name, - pathParameters: {"id": album.id!}, - extra: album, - ); + onPressed: () { + context.navigateTo(AlbumRoute(id: album.id, album: album)); }, ); } diff --git a/lib/modules/stats/common/artist_item.dart b/lib/modules/stats/common/artist_item.dart index 7e7281da..92d3b915 100644 --- a/lib/modules/stats/common/artist_item.dart +++ b/lib/modules/stats/common/artist_item.dart @@ -1,12 +1,12 @@ -import 'package:flutter/material.dart'; -import 'package:spotify/spotify.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/artist/artist.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:spotube/components/ui/button_tile.dart'; +import 'package:spotube/models/metadata/metadata.dart'; class StatsArtistItem extends StatelessWidget { - final Artist artist; + final SpotubeSimpleArtistObject artist; final Widget info; const StatsArtistItem({ super.key, @@ -16,23 +16,20 @@ class StatsArtistItem extends StatelessWidget { @override Widget build(BuildContext context) { - return ListTile( - title: Text(artist.name!), - horizontalTitleGap: 8, - leading: CircleAvatar( - backgroundImage: UniversalImage.imageProvider( + return ButtonTile( + style: ButtonVariance.ghost, + title: Text(artist.name), + leading: Avatar( + initials: artist.name.substring(0, 1), + provider: UniversalImage.imageProvider( (artist.images).asUrlString( placeholder: ImagePlaceholder.artist, ), ), ), trailing: info, - onTap: () { - ServiceUtils.pushNamed( - context, - ArtistPage.name, - pathParameters: {"id": artist.id!}, - ); + onPressed: () { + context.navigateTo(ArtistRoute(artistId: artist.id)); }, ); } diff --git a/lib/modules/stats/common/playlist_item.dart b/lib/modules/stats/common/playlist_item.dart index 515c97b3..64abe7d5 100644 --- a/lib/modules/stats/common/playlist_item.dart +++ b/lib/modules/stats/common/playlist_item.dart @@ -1,21 +1,21 @@ -import 'package:flutter/material.dart'; -import 'package:spotify/spotify.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/image.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/string.dart'; -import 'package:spotube/pages/playlist/playlist.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:spotube/models/metadata/metadata.dart'; class StatsPlaylistItem extends StatelessWidget { - final PlaylistSimple playlist; + final SpotubeSimplePlaylistObject playlist; final Widget info; const StatsPlaylistItem( {super.key, required this.playlist, required this.info}); @override Widget build(BuildContext context) { - return ListTile( - horizontalTitleGap: 8, + return ButtonTile( + style: ButtonVariance.ghost, leading: ClipRRect( borderRadius: BorderRadius.circular(4), child: UniversalImage( @@ -26,20 +26,15 @@ class StatsPlaylistItem extends StatelessWidget { height: 40, ), ), - title: Text(playlist.name!), + title: Text(playlist.name), subtitle: Text( - playlist.description?.unescapeHtml() ?? '', + playlist.description.unescapeHtml(), maxLines: 1, overflow: TextOverflow.ellipsis, ), trailing: info, - onTap: () { - ServiceUtils.pushNamed( - context, - PlaylistPage.name, - pathParameters: {"id": playlist.id!}, - extra: playlist, - ); + onPressed: () { + context.navigateTo(PlaylistRoute(id: playlist.id, playlist: playlist)); }, ); } diff --git a/lib/modules/stats/common/track_item.dart b/lib/modules/stats/common/track_item.dart index 44e81340..eea3dd4b 100644 --- a/lib/modules/stats/common/track_item.dart +++ b/lib/modules/stats/common/track_item.dart @@ -1,13 +1,13 @@ -import 'package:flutter/material.dart'; -import 'package:spotify/spotify.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/links/artist_link.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/track/track.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:spotube/components/ui/button_tile.dart'; +import 'package:spotube/models/metadata/metadata.dart'; class StatsTrackItem extends StatelessWidget { - final Track track; + final SpotubeTrackObject track; final Widget info; const StatsTrackItem({ super.key, @@ -17,39 +17,29 @@ class StatsTrackItem extends StatelessWidget { @override Widget build(BuildContext context) { - return ListTile( - horizontalTitleGap: 8, + return ButtonTile( + style: ButtonVariance.ghost, leading: ClipRRect( borderRadius: BorderRadius.circular(4), child: UniversalImage( - path: (track.album?.images).asUrlString( + path: (track.album.images).asUrlString( placeholder: ImagePlaceholder.albumArt, ), width: 40, height: 40, ), ), - title: Text(track.name!), + title: Text(track.name), subtitle: ArtistLink( - artists: track.artists!, + artists: track.artists, mainAxisAlignment: WrapAlignment.start, - onOverflowArtistClick: () => ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": track.id!, - }, - ), + onOverflowArtistClick: () { + context.navigateTo(TrackRoute(trackId: track.id)); + }, ), trailing: info, - onTap: () { - ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": track.id!, - }, - ); + onPressed: () { + context.navigateTo(TrackRoute(trackId: track.id)); }, ); } diff --git a/lib/modules/stats/summary/summary.dart b/lib/modules/stats/summary/summary.dart index 46068fec..30e68b1f 100644 --- a/lib/modules/stats/summary/summary.dart +++ b/lib/modules/stats/summary/summary.dart @@ -1,19 +1,14 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/fake.dart'; import 'package:spotube/collections/formatters.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/modules/stats/summary/summary_card.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/stats/albums/albums.dart'; -import 'package:spotube/pages/stats/artists/artists.dart'; -import 'package:spotube/pages/stats/fees/fees.dart'; -import 'package:spotube/pages/stats/minutes/minutes.dart'; -import 'package:spotube/pages/stats/playlists/playlists.dart'; -import 'package:spotube/pages/stats/streams/streams.dart'; import 'package:spotube/provider/history/summary.dart'; -import 'package:spotube/utils/service_utils.dart'; class StatsPageSummarySection extends HookConsumerWidget { const StatsPageSummarySection({super.key}); @@ -48,18 +43,18 @@ class StatsPageSummarySection extends HookConsumerWidget { title: summaryData.duration.inMinutes.toDouble(), unit: context.l10n.summary_minutes, description: context.l10n.summary_listened_to_music, - color: Colors.purple, + color: Colors.indigo, onTap: () { - ServiceUtils.pushNamed(context, StatsMinutesPage.name); + context.navigateTo(const StatsMinutesRoute()); }, ), SummaryCard( title: summaryData.tracks.toDouble(), unit: context.l10n.summary_songs, description: context.l10n.summary_streamed_overall, - color: Colors.lightBlue, + color: Colors.blue, onTap: () { - ServiceUtils.pushNamed(context, StatsStreamsPage.name); + context.navigateTo(const StatsStreamsRoute()); }, ), SummaryCard.unformatted( @@ -68,7 +63,7 @@ class StatsPageSummarySection extends HookConsumerWidget { description: context.l10n.summary_owed_to_artists, color: Colors.green, onTap: () { - ServiceUtils.pushNamed(context, StatsStreamFeesPage.name); + context.navigateTo(const StatsStreamFeesRoute()); }, ), SummaryCard( @@ -77,7 +72,7 @@ class StatsPageSummarySection extends HookConsumerWidget { description: context.l10n.summary_music_reached_you, color: Colors.yellow, onTap: () { - ServiceUtils.pushNamed(context, StatsArtistsPage.name); + context.navigateTo(const StatsArtistsRoute()); }, ), SummaryCard( @@ -86,7 +81,7 @@ class StatsPageSummarySection extends HookConsumerWidget { description: context.l10n.summary_got_your_love, color: Colors.pink, onTap: () { - ServiceUtils.pushNamed(context, StatsAlbumsPage.name); + context.navigateTo(const StatsAlbumsRoute()); }, ), SummaryCard( @@ -95,7 +90,7 @@ class StatsPageSummarySection extends HookConsumerWidget { description: context.l10n.summary_were_on_repeat, color: Colors.teal, onTap: () { - ServiceUtils.pushNamed(context, StatsPlaylistsPage.name); + context.navigateTo(const StatsPlaylistsRoute()); }, ), ]), diff --git a/lib/modules/stats/summary/summary_card.dart b/lib/modules/stats/summary/summary_card.dart index 243c50e8..e78dd080 100644 --- a/lib/modules/stats/summary/summary_card.dart +++ b/lib/modules/stats/summary/summary_card.dart @@ -1,6 +1,7 @@ import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; +import 'package:flutter/foundation.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/formatters.dart'; class SummaryCard extends StatelessWidget { @@ -9,7 +10,7 @@ class SummaryCard extends StatelessWidget { final String description; final VoidCallback? onTap; - final MaterialColor color; + final ColorShades color; SummaryCard({ super.key, @@ -31,15 +32,18 @@ class SummaryCard extends StatelessWidget { @override Widget build(BuildContext context) { - final ThemeData(:textTheme, :brightness) = Theme.of(context); + final ThemeData(:typography, :brightness) = Theme.of(context); final descriptionNewLines = description.split("").where((s) => s == "\n"); return Card( - color: brightness == Brightness.dark ? color.shade100 : color.shade50, - child: InkWell( - borderRadius: BorderRadius.circular(16), - onTap: onTap, + fillColor: brightness == Brightness.dark ? color.shade100 : color.shade50, + filled: true, + borderColor: color, + padding: EdgeInsets.zero, + borderRadius: context.theme.borderRadiusLg, + child: Button.ghost( + onPressed: onTap, child: Padding( padding: const EdgeInsets.symmetric(vertical: 8.0, horizontal: 15), child: Column( @@ -52,13 +56,13 @@ class SummaryCard extends StatelessWidget { children: [ TextSpan( text: title, - style: textTheme.headlineLarge?.copyWith( + style: typography.h2.copyWith( color: color.shade900, ), ), TextSpan( text: " $unit", - style: textTheme.titleMedium?.copyWith( + style: typography.semiBold.copyWith( color: color.shade900, ), ), @@ -73,7 +77,7 @@ class SummaryCard extends StatelessWidget { ? descriptionNewLines.length + 1 : 1, minFontSize: 9, - style: textTheme.labelMedium!.copyWith( + style: typography.small.copyWith( color: color.shade900, ), ), diff --git a/lib/modules/stats/top/albums.dart b/lib/modules/stats/top/albums.dart index e401340e..e2a9042a 100644 --- a/lib/modules/stats/top/albums.dart +++ b/lib/modules/stats/top/albums.dart @@ -1,12 +1,14 @@ -import 'package:flutter/material.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/modules/stats/common/album_item.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/albums.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; class TopAlbums extends HookConsumerWidget { @@ -31,6 +33,24 @@ class TopAlbums extends HookConsumerWidget { isLoading: topAlbums.isLoading && !topAlbums.isLoadingNextPage, hasReachedMax: topAlbums.asData?.value.hasMore ?? true, itemCount: albumsData.length, + emptyBuilder: (context) => Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + const Gap(50), + Undraw( + illustration: UndrawIllustration.happyMusic, + color: context.theme.colorScheme.primary, + height: 200 * context.theme.scaling, + ), + Text( + context.l10n.no_tracks_listened_yet, + textAlign: TextAlign.center, + ).muted().small(), + ], + ), + ), itemBuilder: (context, index) { final album = albumsData[index]; return StatsAlbumItem( diff --git a/lib/modules/stats/top/artists.dart b/lib/modules/stats/top/artists.dart index 3e4e098d..5a8dc441 100644 --- a/lib/modules/stats/top/artists.dart +++ b/lib/modules/stats/top/artists.dart @@ -1,13 +1,15 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/modules/stats/common/artist_item.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; class TopArtists extends HookConsumerWidget { @@ -22,8 +24,8 @@ class TopArtists extends HookConsumerWidget { final topTracksNotifier = ref.watch(historyTopTracksProvider(historyDuration).notifier); - final artistsData = useMemoized( - () => topTracks.asData?.value.artists ?? [], [topTracks.asData?.value]); + final artistsData = + useMemoized(() => topTracksNotifier.artists, [topTracks.asData?.value]); return Skeletonizer.sliver( enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, @@ -35,6 +37,24 @@ class TopArtists extends HookConsumerWidget { isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, hasReachedMax: topTracks.asData?.value.hasMore ?? true, itemCount: artistsData.length, + emptyBuilder: (context) => Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + const Gap(50), + Undraw( + illustration: UndrawIllustration.happyMusic, + color: context.theme.colorScheme.primary, + height: 200 * context.theme.scaling, + ), + Text( + context.l10n.no_tracks_listened_yet, + textAlign: TextAlign.center, + ).muted().small(), + ], + ), + ), itemBuilder: (context, index) { final artist = artistsData[index]; return StatsArtistItem( diff --git a/lib/modules/stats/top/top.dart b/lib/modules/stats/top/top.dart index 643064aa..38f04ccb 100644 --- a/lib/modules/stats/top/top.dart +++ b/lib/modules/stats/top/top.dart @@ -1,7 +1,8 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/components/themed_button_tab_bar.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/modules/stats/top/albums.dart'; import 'package:spotube/modules/stats/top/artists.dart'; import 'package:spotube/modules/stats/top/tracks.dart'; @@ -14,94 +15,97 @@ class StatsPageTopSection extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final tabController = useTabController(initialLength: 3); + final selectedIndex = useState(0); final historyDuration = ref.watch(playbackHistoryTopDurationProvider); final historyDurationNotifier = ref.watch(playbackHistoryTopDurationProvider.notifier); - return SliverMainAxisGroup( - slivers: [ - SliverAppBar( - floating: true, - flexibleSpace: ThemedButtonsTabBar( - controller: tabController, - tabs: [ - Tab( - child: Padding( - padding: const EdgeInsets.all(5), - child: Text(context.l10n.top_tracks), - ), - ), - Tab( - child: Padding( - padding: const EdgeInsets.all(5), - child: Text(context.l10n.top_artists), - ), - ), - Tab( - child: Padding( - padding: const EdgeInsets.all(5), - child: Text(context.l10n.top_albums), - ), - ), - ], - ), - ), - SliverToBoxAdapter( - child: Align( - alignment: Alignment.centerRight, - child: DropdownButton( - style: Theme.of(context).textTheme.bodySmall!, - isDense: true, - padding: const EdgeInsets.all(4), - borderRadius: BorderRadius.circular(4), - underline: const SizedBox(), - value: historyDuration, - onChanged: (value) { - if (value == null) return; - historyDurationNotifier.update((_) => value); + final translations = { + HistoryDuration.days7: context.l10n.this_week, + HistoryDuration.days30: context.l10n.this_month, + HistoryDuration.months6: context.l10n.last_6_months, + HistoryDuration.year: context.l10n.this_year, + HistoryDuration.years2: context.l10n.last_2_years, + HistoryDuration.allTime: context.l10n.all_time, + }; + + final dropdown = Select( + popupConstraints: const BoxConstraints(maxWidth: 150), + popupWidthConstraint: PopoverConstraint.flexible, + padding: const EdgeInsets.all(4), + borderRadius: BorderRadius.circular(4), + value: historyDuration, + onChanged: (value) { + if (value == null) return; + historyDurationNotifier.update((_) => value); + }, + itemBuilder: (context, item) => Text(translations[item]!), + popup: (context) { + return SelectPopup( + items: SelectItemBuilder( + childCount: HistoryDuration.values.length, + builder: (context, index) { + final item = HistoryDuration.values[index]; + return SelectItemButton( + value: item, + child: Text(translations[item]!), + ); }, - icon: const Icon(Icons.arrow_drop_down), - items: [ - DropdownMenuItem( - value: HistoryDuration.days7, - child: Text(context.l10n.this_week), - ), - DropdownMenuItem( - value: HistoryDuration.days30, - child: Text(context.l10n.this_month), - ), - DropdownMenuItem( - value: HistoryDuration.months6, - child: Text(context.l10n.last_6_months), - ), - DropdownMenuItem( - value: HistoryDuration.year, - child: Text(context.l10n.this_year), - ), - DropdownMenuItem( - value: HistoryDuration.years2, - child: Text(context.l10n.last_2_years), - ), - DropdownMenuItem( - value: HistoryDuration.allTime, - child: Text(context.l10n.all_time), - ), - ], + ), + ); + }); + + return SliverLayoutBuilder(builder: (context, constraints) { + return SliverMainAxisGroup( + slivers: [ + SliverAppBar( + floating: true, + elevation: 0, + backgroundColor: context.theme.colorScheme.background, + automaticallyImplyLeading: false, + flexibleSpace: Padding( + padding: const EdgeInsets.all(8.0), + child: Row( + children: [ + TabList( + index: selectedIndex.value, + onChanged: (value) { + selectedIndex.value = value; + }, + children: [ + TabItem( + child: Text(context.l10n.top_tracks), + ), + TabItem( + child: Text(context.l10n.top_artists), + ), + TabItem( + child: Text(context.l10n.top_albums), + ), + ], + ), + if (constraints.mdAndUp) ...[ + const Spacer(), + dropdown, + ] + ], + ), ), ), - ), - ListenableBuilder( - listenable: tabController, - builder: (context, _) { - return switch (tabController.index) { - 1 => const TopArtists(), - 2 => const TopAlbums(), - _ => const TopTracks(), - }; + if (constraints.smAndDown) + SliverToBoxAdapter( + child: Align( + alignment: Alignment.centerRight, + child: dropdown, + ), + ), + switch (selectedIndex.value) { + 1 => const TopArtists(), + 2 => const TopAlbums(), + _ => const TopTracks(), }, - ), - ], - ); + ], + ); + }); } } diff --git a/lib/modules/stats/top/tracks.dart b/lib/modules/stats/top/tracks.dart index 7fba220d..08c742c4 100644 --- a/lib/modules/stats/top/tracks.dart +++ b/lib/modules/stats/top/tracks.dart @@ -1,12 +1,14 @@ -import 'package:flutter/material.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/modules/stats/common/track_item.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; class TopTracks extends HookConsumerWidget { @@ -33,6 +35,24 @@ class TopTracks extends HookConsumerWidget { isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, hasReachedMax: topTracks.asData?.value.hasMore ?? true, itemCount: tracksData.length, + emptyBuilder: (context) => Padding( + padding: const EdgeInsets.all(8.0), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + const Gap(50), + Undraw( + illustration: UndrawIllustration.happyMusic, + color: context.theme.colorScheme.primary, + height: 200 * context.theme.scaling, + ), + Text( + context.l10n.no_tracks_listened_yet, + textAlign: TextAlign.center, + ).muted().small(), + ], + ), + ), itemBuilder: (context, index) { final track = tracksData[index]; return StatsTrackItem( diff --git a/lib/pages/album/album.dart b/lib/pages/album/album.dart index 0c6cfd69..049d8023 100644 --- a/lib/pages/album/album.dart +++ b/lib/pages/album/album.dart @@ -1,65 +1,83 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' as material; +import 'package:auto_route/auto_route.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/components/tracks_view/track_view.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/track_presentation.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/library/albums.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/album.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +@RoutePage() class AlbumPage extends HookConsumerWidget { static const name = "album"; - final AlbumSimple album; + final SpotubeSimpleAlbumObject album; + final String id; const AlbumPage({ super.key, + @PathParam("id") required this.id, required this.album, }); @override Widget build(BuildContext context, ref) { - final tracks = ref.watch(albumTracksProvider(album)); - final tracksNotifier = ref.watch(albumTracksProvider(album).notifier); - final favoriteAlbumsNotifier = ref.watch(favoriteAlbumsProvider.notifier); - final isSavedAlbum = ref.watch(albumsIsSavedProvider(album.id!)); + final tracks = ref.watch(metadataPluginAlbumTracksProvider(album.id)); + final tracksNotifier = + ref.watch(metadataPluginAlbumTracksProvider(album.id).notifier); + final favoriteAlbumsNotifier = + ref.watch(metadataPluginSavedAlbumsProvider.notifier); + final isSavedAlbum = + ref.watch(metadataPluginIsSavedAlbumProvider(album.id)); - return InheritedTrackView( - collection: album, - image: album.images.asUrlString( - placeholder: ImagePlaceholder.albumArt, - ), - title: album.name!, - description: - "${context.l10n.released} • ${album.releaseDate} • ${album.artists!.first.name}", - tracks: tracks.asData?.value.items ?? [], - pagination: PaginationProps( - hasNextPage: tracks.asData?.value.hasMore ?? false, - isLoading: tracks.isLoadingNextPage, - onFetchMore: () async { - await tracksNotifier.fetchMore(); - }, - onFetchAll: () async { - return tracksNotifier.fetchAll(); - }, - onRefresh: () async { - ref.invalidate(albumTracksProvider(album)); - }, - ), - routePath: "/album/${album.id}", - shareUrl: album.externalUrls?.spotify ?? - "https://open.spotify.com/album/${album.id}", - isLiked: isSavedAlbum.asData?.value ?? false, - onHeart: isSavedAlbum.asData?.value == null - ? null - : () async { - if (isSavedAlbum.asData!.value) { - await favoriteAlbumsNotifier.removeFavorites([album.id!]); - } else { - await favoriteAlbumsNotifier.addFavorites([album.id!]); - } - return null; + return material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginAlbumTracksProvider(album.id)); + ref.invalidate(metadataPluginIsSavedAlbumProvider(album.id)); + ref.invalidate(metadataPluginSavedAlbumsProvider); + }, + child: TrackPresentation( + options: TrackPresentationOptions( + collection: album, + image: album.images.asUrlString( + placeholder: ImagePlaceholder.albumArt, + ), + title: album.name, + description: + "${context.l10n.released} • ${album.releaseDate} • ${album.artists.first.name}", + tracks: tracks.asData?.value.items ?? [], + error: tracks.error, + pagination: PaginationProps( + hasNextPage: tracks.asData?.value.hasMore ?? false, + isLoading: tracks.isLoading || tracks.isLoadingNextPage, + onFetchMore: () async { + await tracksNotifier.fetchMore(); }, - child: const TrackView(), + onFetchAll: () async { + return tracksNotifier.fetchAll(); + }, + onRefresh: () async { + ref.invalidate(metadataPluginAlbumTracksProvider(album.id)); + }, + ), + routePath: "/album/${album.id}", + shareUrl: album.externalUri, + isLiked: isSavedAlbum.asData?.value ?? false, + owner: album.artists.first.name, + onHeart: isSavedAlbum.asData?.value == null + ? null + : () async { + if (isSavedAlbum.asData!.value) { + await favoriteAlbumsNotifier.removeFavorite([album]); + } else { + await favoriteAlbumsNotifier.addFavorite([album]); + } + return null; + }, + ), + ), ); } } diff --git a/lib/pages/artist/artist.dart b/lib/pages/artist/artist.dart index 70ad72de..64bed283 100644 --- a/lib/pages/artist/artist.dart +++ b/lib/pages/artist/artist.dart @@ -1,84 +1,110 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' as material; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/components/button/back_button.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/modules/artist/artist_album_list.dart'; import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/artist/artist_album_list.dart'; import 'package:spotube/pages/artist/section/footer.dart'; import 'package:spotube/pages/artist/section/header.dart'; import 'package:spotube/pages/artist/section/related_artists.dart'; import 'package:spotube/pages/artist/section/top_tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/artist/albums.dart'; +import 'package:spotube/provider/metadata_plugin/artist/artist.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/provider/metadata_plugin/artist/related.dart'; +import 'package:spotube/provider/metadata_plugin/artist/top_tracks.dart'; +import 'package:spotube/provider/metadata_plugin/artist/wikipedia.dart'; +import 'package:spotube/provider/metadata_plugin/library/artists.dart'; +@RoutePage() class ArtistPage extends HookConsumerWidget { static const name = "artist"; final String artistId; - const ArtistPage(this.artistId, {super.key}); + const ArtistPage( + @PathParam("id") this.artistId, { + super.key, + }); @override Widget build(BuildContext context, ref) { final scrollController = useScrollController(); - final theme = Theme.of(context); - final artistQuery = ref.watch(artistProvider(artistId)); + final artistQuery = ref.watch(metadataPluginArtistProvider(artistId)); return SafeArea( bottom: false, child: Scaffold( - appBar: const PageWindowTitleBar( - leading: BackButton(), - backgroundColor: Colors.transparent, - ), - extendBodyBehindAppBar: true, - body: Builder(builder: (context) { - if (artistQuery.hasError && artistQuery.asData?.value == null) { - return Center(child: Text(artistQuery.error.toString())); - } - return Skeletonizer( - enabled: artistQuery.isLoading, - child: CustomScrollView( - controller: scrollController, - slivers: [ - SliverToBoxAdapter( - child: SafeArea( - bottom: false, - child: ArtistPageHeader(artistId: artistId), - ), - ), - const SliverGap(50), - ArtistPageTopTracks(artistId: artistId), - const SliverGap(50), - SliverToBoxAdapter(child: ArtistAlbumList(artistId)), - const SliverGap(20), - SliverPadding( - padding: const EdgeInsets.all(8.0), - sliver: SliverToBoxAdapter( - child: Text( - context.l10n.fans_also_like, - style: theme.textTheme.headlineSmall, + headers: const [ + TitleBar( + leading: [BackButton()], + backgroundColor: Colors.transparent, + ) + ], + floatingHeader: true, + child: material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginArtistProvider(artistId)); + ref.invalidate( + metadataPluginArtistRelatedArtistsProvider(artistId), + ); + ref.invalidate(metadataPluginArtistAlbumsProvider(artistId)); + ref.invalidate(metadataPluginIsSavedArtistProvider(artistId)); + ref.invalidate(metadataPluginArtistTopTracksProvider(artistId)); + if (artistQuery.hasValue) { + ref.invalidate( + artistWikipediaSummaryProvider(artistQuery.asData!.value), + ); + } + }, + child: Builder(builder: (context) { + if (artistQuery.hasError && artistQuery.asData?.value == null) { + return Center(child: Text(artistQuery.error.toString())); + } + return Skeletonizer( + enabled: artistQuery.isLoading, + child: CustomScrollView( + controller: scrollController, + slivers: [ + const SliverGap(material.kToolbarHeight), + SliverToBoxAdapter( + child: SafeArea( + bottom: false, + child: ArtistPageHeader(artistId: artistId), ), ), - ), - SliverSafeArea( - sliver: ArtistPageRelatedArtists(artistId: artistId), - ), - if (artistQuery.asData?.value != null) - SliverSafeArea( - top: false, + const SliverGap(20), + ArtistPageTopTracks(artistId: artistId), + const SliverGap(20), + SliverToBoxAdapter(child: ArtistAlbumList(artistId)), + SliverPadding( + padding: const EdgeInsets.all(8.0), sliver: SliverToBoxAdapter( + child: Text( + context.l10n.fans_also_like, + style: context.theme.typography.h4, + ), + ), + ), + ArtistPageRelatedArtists(artistId: artistId), + const SliverGap(20), + if (artistQuery.asData?.value != null) + SliverToBoxAdapter( child: ArtistPageFooter(artist: artistQuery.asData!.value), ), - ), - ], - ), - ); - }), + const SliverSafeArea(sliver: SliverGap(10)), + ], + ), + ); + }), + ), ), ); } diff --git a/lib/pages/artist/section/footer.dart b/lib/pages/artist/section/footer.dart index abe86410..938fb6fc 100644 --- a/lib/pages/artist/section/footer.dart +++ b/lib/pages/artist/section/footer.dart @@ -1,22 +1,20 @@ import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/artist/wikipedia.dart'; import 'package:url_launcher/url_launcher_string.dart'; class ArtistPageFooter extends ConsumerWidget { - final Artist artist; + final SpotubeFullArtistObject artist; const ArtistPageFooter({super.key, required this.artist}); @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); + final ThemeData(:typography) = Theme.of(context); final mediaQuery = MediaQuery.of(context); final artistImage = artist.images.asUrlString( @@ -26,7 +24,7 @@ class ArtistPageFooter extends ConsumerWidget { if (summary.asData?.value == null) return const SizedBox.shrink(); return Container( - margin: const EdgeInsets.all(16), + margin: const EdgeInsets.all(8), padding: mediaQuery.smAndDown ? const EdgeInsets.all(20) : const EdgeInsets.all(30), @@ -35,7 +33,7 @@ class ArtistPageFooter extends ConsumerWidget { borderRadius: BorderRadius.circular(10), image: DecorationImage( colorFilter: ColorFilter.mode( - Colors.black.withOpacity(0.5), + Colors.black.withValues(alpha: 0.5), BlendMode.darken, ), image: UniversalImage.imageProvider( @@ -50,7 +48,7 @@ class ArtistPageFooter extends ConsumerWidget { alignment: Alignment.center, child: RichText( text: TextSpan( - style: textTheme.bodyLarge?.copyWith( + style: typography.semiBold.copyWith( color: Colors.white, ), children: [ @@ -64,7 +62,7 @@ class ArtistPageFooter extends ConsumerWidget { ), TextSpan( text: " Wikipedia", - style: textTheme.titleLarge?.copyWith( + style: typography.large.copyWith( color: Colors.white, ), ), @@ -74,10 +72,10 @@ class ArtistPageFooter extends ConsumerWidget { ), TextSpan( text: '\n...read more at wikipedia', - style: textTheme.bodyLarge?.copyWith( - color: Colors.lightBlue[300], + style: typography.semiBold.copyWith( + color: Colors.sky[300], decoration: TextDecoration.underline, - decorationColor: Colors.lightBlue[300], + decorationColor: Colors.sky[300], ), recognizer: TapGestureRecognizer() ..onTap = () async { diff --git a/lib/pages/artist/section/header.dart b/lib/pages/artist/section/header.dart index 713e0d26..b8e7e5dc 100644 --- a/lib/pages/artist/section/header.dart +++ b/lib/pages/artist/section/header.dart @@ -1,19 +1,19 @@ -import 'package:flutter/material.dart'; +import 'package:auto_size_text/auto_size_text.dart'; import 'package:flutter/services.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide Consumer; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/fake.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/hooks/utils/use_breakpoint_value.dart'; import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/blacklist_provider.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/artist/artist.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/library/artists.dart'; import 'package:spotube/utils/primitive_utils.dart'; class ArtistPageHeader extends HookConsumerWidget { @@ -22,216 +22,205 @@ class ArtistPageHeader extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final artistQuery = ref.watch(artistProvider(artistId)); + final artistQuery = ref.watch(metadataPluginArtistProvider(artistId)); final artist = artistQuery.asData?.value ?? FakeData.artist; - final scaffoldMessenger = ScaffoldMessenger.of(context); - final mediaQuery = MediaQuery.of(context); final theme = Theme.of(context); - final ThemeData(:textTheme) = theme; + final ThemeData(:typography) = theme; - final chipTextVariant = useBreakpointValue( - xs: textTheme.bodySmall, - sm: textTheme.bodySmall, - md: textTheme.bodyMedium, - lg: textTheme.bodyLarge, - xl: textTheme.titleSmall, - xxl: textTheme.titleMedium, - ); - - final auth = ref.watch(authenticationProvider); + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); ref.watch(blacklistProvider); final blacklistNotifier = ref.watch(blacklistProvider.notifier); - final isBlackListed = blacklistNotifier.containsArtist(artist); + final isBlackListed = blacklistNotifier.containsArtist(artist.id); final image = artist.images.asUrlString( placeholder: ImagePlaceholder.artist, ); + final actions = Skeleton.keep( + child: Row( + mainAxisSize: MainAxisSize.min, + children: [ + if (authenticated.asData?.value == true) + Consumer( + builder: (context, ref, _) { + final isFollowingQuery = ref.watch( + metadataPluginIsSavedArtistProvider(artist.id), + ); + final followingArtistNotifier = + ref.watch(metadataPluginSavedArtistsProvider.notifier); + + return switch (isFollowingQuery) { + AsyncData(value: final following) => Builder( + builder: (context) { + if (following) { + return Button.outline( + onPressed: () async { + await followingArtistNotifier + .removeFavorite([artist]); + }, + child: Text(context.l10n.following), + ); + } + + return Button.primary( + onPressed: () async { + await followingArtistNotifier.addFavorite([artist]); + }, + child: Text(context.l10n.follow), + ); + }, + ), + AsyncError() => const SizedBox(), + _ => const SizedBox.square( + dimension: 20, + child: CircularProgressIndicator(), + ) + }; + }, + ), + const SizedBox(width: 5), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.add_artist_to_blacklist), + ).call, + child: IconButton( + icon: Icon( + SpotubeIcons.userRemove, + color: !isBlackListed ? Colors.red[400] : null, + ), + variance: isBlackListed + ? ButtonVariance.destructive + : ButtonVariance.ghost, + onPressed: () async { + if (isBlackListed) { + await ref.read(blacklistProvider.notifier).remove(artist.id); + } else { + await ref.read(blacklistProvider.notifier).add( + BlacklistTableCompanion.insert( + name: artist.name, + elementId: artist.id, + elementType: BlacklistedType.artist, + ), + ); + } + }, + ), + ), + IconButton.ghost( + icon: const Icon(SpotubeIcons.share), + onPressed: () async { + await Clipboard.setData( + ClipboardData( + text: artist.externalUri, + ), + ); + + if (!context.mounted) return; + + showToast( + context: context, + location: ToastLocation.topRight, + dismissible: true, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n.artist_url_copied, + textAlign: TextAlign.center, + ), + ); + }, + ); + }, + ) + ], + ), + ); + return LayoutBuilder( builder: (context, constrains) { - return Center( - child: Flex( - mainAxisAlignment: MainAxisAlignment.center, - crossAxisAlignment: constrains.smAndDown - ? CrossAxisAlignment.start - : CrossAxisAlignment.center, - direction: constrains.smAndDown ? Axis.vertical : Axis.horizontal, - children: [ - DecoratedBox( - decoration: BoxDecoration( - boxShadow: kElevationToShadow[2], - borderRadius: BorderRadius.circular(35), - ), - child: ClipRRect( - borderRadius: BorderRadius.circular(35), - child: UniversalImage( - path: image, - width: 250, - height: 250, - fit: BoxFit.cover, - ), - ), - ), - const Gap(20), - Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Row( - mainAxisSize: MainAxisSize.min, - children: [ - Container( - padding: const EdgeInsets.symmetric( - horizontal: 10, vertical: 5), - decoration: BoxDecoration( - color: Colors.blue, - borderRadius: BorderRadius.circular(50)), - child: Skeleton.keep( - child: Text( - artist.type!.toUpperCase(), - style: chipTextVariant.copyWith( - color: Colors.white, - ), - ), - ), - ), - if (isBlackListed) ...[ - const SizedBox(width: 5), - Container( - padding: const EdgeInsets.symmetric( - horizontal: 10, vertical: 5), - decoration: BoxDecoration( - color: Colors.red[400], - borderRadius: BorderRadius.circular(50)), - child: Text( - context.l10n.blacklisted, - style: chipTextVariant.copyWith( - color: Colors.white, - ), - ), - ), - ] - ], - ), - Text( - artist.name!, - style: mediaQuery.smAndDown - ? textTheme.headlineSmall - : textTheme.headlineMedium, - ), - Text( - context.l10n.followers( - PrimitiveUtils.toReadableNumber( - artist.followers!.total!.toDouble(), + return Padding( + padding: const EdgeInsets.symmetric(horizontal: 8.0), + child: Card( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + ClipRRect( + borderRadius: theme.borderRadiusXl, + child: UniversalImage( + path: image, + width: constrains.mdAndUp ? 200 : 120, + height: constrains.mdAndUp ? 200 : 120, + fit: BoxFit.cover, ), ), - style: textTheme.bodyMedium?.copyWith( - fontWeight: mediaQuery.mdAndUp ? FontWeight.bold : null, - ), - ), - const Gap(20), - Skeleton.keep( - child: Row( - mainAxisSize: MainAxisSize.min, - children: [ - if (auth.asData?.value != null) - Consumer( - builder: (context, ref, _) { - final isFollowingQuery = ref - .watch(artistIsFollowingProvider(artist.id!)); - final followingArtistNotifier = - ref.watch(followedArtistsProvider.notifier); - - return switch (isFollowingQuery) { - AsyncData(value: final following) => Builder( - builder: (context) { - if (following) { - return OutlinedButton( - onPressed: () async { - await followingArtistNotifier - .removeArtists([artist.id!]); - }, - child: Text(context.l10n.following), - ); - } - - return FilledButton( - onPressed: () async { - await followingArtistNotifier - .saveArtists([artist.id!]); - }, - child: Text(context.l10n.follow), - ); - }, - ), - AsyncError() => const SizedBox(), - _ => const SizedBox.square( - dimension: 20, - child: CircularProgressIndicator(), - ) - }; - }, - ), - const SizedBox(width: 5), - IconButton( - tooltip: context.l10n.add_artist_to_blacklist, - icon: Icon( - SpotubeIcons.userRemove, - color: - !isBlackListed ? Colors.red[400] : Colors.white, - ), - style: IconButton.styleFrom( - backgroundColor: - isBlackListed ? Colors.red[400] : null, - ), - onPressed: () async { - if (isBlackListed) { - await ref - .read(blacklistProvider.notifier) - .remove(artist.id!); - } else { - await ref.read(blacklistProvider.notifier).add( - BlacklistTableCompanion.insert( - name: artist.name!, - elementId: artist.id!, - elementType: BlacklistedType.artist, - ), - ); - } - }, - ), - IconButton( - icon: const Icon(SpotubeIcons.share), - onPressed: () async { - if (artist.externalUrls?.spotify != null) { - await Clipboard.setData( - ClipboardData( - text: artist.externalUrls!.spotify!, - ), - ); - } - - if (!context.mounted) return; - - scaffoldMessenger.showSnackBar( - SnackBar( - width: 300, - behavior: SnackBarBehavior.floating, - content: Text( - context.l10n.artist_url_copied, - textAlign: TextAlign.center, - ), + const Gap(20), + Flexible( + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + mainAxisSize: MainAxisSize.min, + children: [ + OutlineBadge( + child: + Text(context.l10n.artist).small().muted(), ), - ); - }, - ) - ], + if (isBlackListed) ...[ + const Gap(5), + DestructiveBadge( + child: Text(context.l10n.blacklisted).small(), + ), + ] + ], + ), + const Gap(10), + Flexible( + child: AutoSizeText( + artist.name, + style: constrains.smAndDown + ? typography.h4 + : typography.h3, + maxLines: 2, + overflow: TextOverflow.ellipsis, + minFontSize: 14, + ), + ), + const Gap(5), + Flexible( + child: AutoSizeText( + context.l10n.followers( + artist.followers == null + ? double.infinity + : PrimitiveUtils.toReadableNumber( + artist.followers!.toDouble(), + ), + ), + maxLines: 1, + overflow: TextOverflow.ellipsis, + minFontSize: 12, + ).muted(), + ), + if (constrains.mdAndUp) ...[ + const Gap(20), + actions, + ] + ], + ), ), - ) - ], - ), - ], + ], + ), + if (constrains.smAndDown) ...[ + const Gap(20), + actions, + ] + ], + ), ), ); }, diff --git a/lib/pages/artist/section/related_artists.dart b/lib/pages/artist/section/related_artists.dart index 066f73fd..ec17e240 100644 --- a/lib/pages/artist/section/related_artists.dart +++ b/lib/pages/artist/section/related_artists.dart @@ -1,7 +1,7 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:spotube/modules/artist/artist_card.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/artist/related.dart'; class ArtistPageRelatedArtists extends ConsumerWidget { final String artistId; @@ -12,13 +12,14 @@ class ArtistPageRelatedArtists extends ConsumerWidget { @override Widget build(BuildContext context, ref) { - final relatedArtists = ref.watch(relatedArtistsProvider(artistId)); + final relatedArtists = + ref.watch(metadataPluginArtistRelatedArtistsProvider(artistId)); return switch (relatedArtists) { AsyncData(value: final artists) => SliverPadding( padding: const EdgeInsets.symmetric(horizontal: 8.0), sliver: SliverGrid.builder( - itemCount: artists.length, + itemCount: artists.items.length, gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent( maxCrossAxisExtent: 200, mainAxisExtent: 250, @@ -27,8 +28,11 @@ class ArtistPageRelatedArtists extends ConsumerWidget { childAspectRatio: 0.8, ), itemBuilder: (context, index) { - final artist = artists.elementAt(index); - return ArtistCard(artist); + final artist = artists.items.elementAt(index); + return SizedBox( + width: 180, + child: ArtistCard(artist), + ); }, ), ), diff --git a/lib/pages/artist/section/top_tracks.dart b/lib/pages/artist/section/top_tracks.dart index d52ed470..30745a01 100644 --- a/lib/pages/artist/section/top_tracks.dart +++ b/lib/pages/artist/section/top_tracks.dart @@ -1,16 +1,18 @@ -import 'package:flutter/material.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/collections/fake.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/dialogs/select_device_dialog.dart'; import 'package:spotube/components/track_tile/track_tile.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/connect/connect.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/artist/top_tracks.dart'; class ArtistPageTopTracks extends HookConsumerWidget { final String artistId; @@ -19,14 +21,15 @@ class ArtistPageTopTracks extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { final theme = Theme.of(context); - final scaffoldMessenger = ScaffoldMessenger.of(context); + final isLoading = useState(false); final playlist = ref.watch(audioPlayerProvider); final playlistNotifier = ref.watch(audioPlayerProvider.notifier); - final topTracksQuery = ref.watch(artistTopTracksProvider(artistId)); + final topTracksQuery = + ref.watch(metadataPluginArtistTopTracksProvider(artistId)); final isPlaylistPlaying = playlist.containsTracks( - topTracksQuery.asData?.value ?? [], + topTracksQuery.asData?.value.items ?? [], ); if (topTracksQuery.hasError) { @@ -37,47 +40,57 @@ class ArtistPageTopTracks extends HookConsumerWidget { ); } - final topTracks = topTracksQuery.asData?.value ?? + final topTracks = topTracksQuery.asData?.value.items ?? List.generate(10, (index) => FakeData.track); - void playPlaylist(List tracks, {Track? currentTrack}) async { + void playPlaylist( + List tracks, { + SpotubeTrackObject? currentTrack, + }) async { + isLoading.value = true; + currentTrack ??= tracks.first; + try { + final isRemoteDevice = await showSelectDeviceDialog(context, ref); - final isRemoteDevice = await showSelectDeviceDialog(context, ref); - if (isRemoteDevice) { - final remotePlayback = ref.read(connectProvider.notifier); - final remotePlaylist = ref.read(queueProvider); + if (isRemoteDevice == null) return; - final isPlaylistPlaying = remotePlaylist.containsTracks(tracks); + if (isRemoteDevice) { + final remotePlayback = ref.read(connectProvider.notifier); + final remotePlaylist = ref.read(queueProvider); - if (!isPlaylistPlaying) { - await remotePlayback.load( - WebSocketLoadEventData.playlist( - tracks: tracks, - collection: null, + final isPlaylistPlaying = remotePlaylist.containsTracks(tracks); + + if (!isPlaylistPlaying) { + await remotePlayback.load( + WebSocketLoadEventData.playlist( + tracks: tracks, + collection: null, + initialIndex: + tracks.indexWhere((s) => s.id == currentTrack?.id), + ), + ); + } else if (isPlaylistPlaying && + currentTrack.id != remotePlaylist.activeTrack?.id) { + final index = playlist.tracks + .toList() + .indexWhere((s) => s.id == currentTrack!.id); + await remotePlayback.jumpTo(index); + } + } else { + if (!isPlaylistPlaying) { + playlistNotifier.load( + tracks, initialIndex: tracks.indexWhere((s) => s.id == currentTrack?.id), - ), - ); - } else if (isPlaylistPlaying && - currentTrack.id != null && - currentTrack.id != remotePlaylist.activeTrack?.id) { - final index = playlist.tracks - .toList() - .indexWhere((s) => s.id == currentTrack!.id); - await remotePlayback.jumpTo(index); - } - } else { - if (!isPlaylistPlaying) { - playlistNotifier.load( - tracks, - initialIndex: tracks.indexWhere((s) => s.id == currentTrack?.id), - autoPlay: true, - ); - } else if (isPlaylistPlaying && - currentTrack.id != null && - currentTrack.id != playlist.activeTrack?.id) { - await playlistNotifier.jumpToTrack(currentTrack); + autoPlay: true, + ); + } else if (isPlaylistPlaying && + currentTrack.id != playlist.activeTrack?.id) { + await playlistNotifier.jumpToTrack(currentTrack); + } } + } finally { + isLoading.value = false; } } @@ -90,46 +103,53 @@ class ArtistPageTopTracks extends HookConsumerWidget { padding: const EdgeInsets.all(8.0), child: Text( context.l10n.top_tracks, - style: theme.textTheme.headlineSmall, + style: theme.typography.h4, ), ), if (!isPlaylistPlaying) - IconButton( + IconButton.outline( icon: const Icon( SpotubeIcons.queueAdd, ), onPressed: () { playlistNotifier.addTracks(topTracks.toList()); - scaffoldMessenger.showSnackBar( - SnackBar( - width: 300, - behavior: SnackBarBehavior.floating, - content: Text( - context.l10n.added_to_queue( - topTracks.length, + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n.added_to_queue( + topTracks.length, + ), ), - textAlign: TextAlign.center, - ), - ), + ); + }, ); }, ), const SizedBox(width: 5), - IconButton( - icon: Skeleton.keep( - child: Icon( - isPlaylistPlaying ? SpotubeIcons.stop : SpotubeIcons.play, - color: Colors.white, - ), - ), - style: IconButton.styleFrom( - backgroundColor: theme.colorScheme.primary, - ), + IconButton.primary( + shape: ButtonShape.circle, + enabled: !isPlaylistPlaying && !isLoading.value, + icon: isLoading.value + ? CircularProgressIndicator( + size: 20 * context.theme.scaling, + color: theme.colorScheme.primaryForeground, + ) + : Skeleton.keep( + child: Icon( + isPlaylistPlaying + ? SpotubeIcons.pause + : SpotubeIcons.play, + ), + ), onPressed: () => playPlaylist(topTracks.toList()), ) ], ), ), + const SliverGap(10), SliverList.builder( itemCount: topTracks.length, itemBuilder: (context, index) { diff --git a/lib/pages/connect/connect.dart b/lib/pages/connect/connect.dart index d3b0d0cb..bb8bbfae 100644 --- a/lib/pages/connect/connect.dart +++ b/lib/pages/connect/connect.dart @@ -1,14 +1,15 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/modules/connect/local_devices.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/connect/control/control.dart'; import 'package:spotube/provider/connect/clients.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class ConnectPage extends HookConsumerWidget { static const name = "connect"; @@ -16,23 +17,18 @@ class ConnectPage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:colorScheme, :textTheme) = Theme.of(context); + final ThemeData(:colorScheme, :typography) = Theme.of(context); final connectClients = ref.watch(connectClientsProvider); final connectClientsNotifier = ref.read(connectClientsProvider.notifier); final discoveredDevices = connectClients.asData?.value.services; - return Scaffold( - appBar: PageWindowTitleBar( - automaticallyImplyLeading: true, - title: Text(context.l10n.devices), - titleSpacing: 0, - ), - body: ListTileTheme( - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(10), - ), - selectedTileColor: colorScheme.secondary.withOpacity(0.1), + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar(title: Text(context.l10n.devices)), + ], child: Padding( padding: const EdgeInsets.all(10.0), child: CustomScrollView( @@ -42,7 +38,7 @@ class ConnectPage extends HookConsumerWidget { sliver: SliverToBoxAdapter( child: Text( context.l10n.remote, - style: textTheme.titleMedium, + style: typography.bold, ), ), ), @@ -55,35 +51,31 @@ class ConnectPage extends HookConsumerWidget { final selected = connectClients.asData?.value.resolvedService?.name == device.name; - return Card( - child: ListTile( - leading: const Icon(SpotubeIcons.monitor), - title: Text(device.name), - subtitle: selected - ? Text( - "${connectClients.asData?.value.resolvedService?.host}" - ":${connectClients.asData?.value.resolvedService?.port}", - ) - : null, - selected: selected, - onTap: () { - if (selected) { - ServiceUtils.pushNamed( - context, - ConnectControlPage.name, - ); - } else { - connectClientsNotifier.resolveService(device); - } - }, - trailing: selected - ? IconButton( - icon: const Icon(SpotubeIcons.power), - onPressed: () => - connectClientsNotifier.clearResolvedService(), - ) - : null, - ), + return ButtonTile( + selected: selected, + leading: const Icon(SpotubeIcons.monitor), + title: Text(device.name), + subtitle: selected + ? Text( + "${connectClients.asData?.value.resolvedService?.host}" + ":${connectClients.asData?.value.resolvedService?.port}", + ) + : null, + trailing: selected + ? IconButton.outline( + icon: const Icon(SpotubeIcons.power), + size: ButtonSize.small, + onPressed: () => + connectClientsNotifier.clearResolvedService(), + ) + : null, + onPressed: () { + if (selected) { + context.navigateTo(const ConnectControlRoute()); + } else { + connectClientsNotifier.resolveService(device); + } + }, ); }, ), diff --git a/lib/pages/connect/control/control.dart b/lib/pages/connect/control/control.dart index cae0bd1b..164e5d43 100644 --- a/lib/pages/connect/control/control.dart +++ b/lib/pages/connect/control/control.dart @@ -1,8 +1,14 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; +import 'dart:convert'; + +import 'package:auto_route/auto_route.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/player/player_queue.dart'; import 'package:spotube/modules/player/volume_slider.dart'; import 'package:spotube/components/image/universal_image.dart'; @@ -12,12 +18,9 @@ import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/extensions/duration.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/track/track.dart'; import 'package:spotube/provider/connect/clients.dart'; import 'package:spotube/provider/connect/connect.dart'; import 'package:media_kit/media_kit.dart' hide Track; -import 'package:spotube/utils/service_utils.dart'; class RemotePlayerQueue extends ConsumerWidget { const RemotePlayerQueue({super.key}); @@ -46,6 +49,7 @@ class RemotePlayerQueue extends ConsumerWidget { } } +@RoutePage() class ConnectControlPage extends HookConsumerWidget { static const name = "connect_control"; @@ -53,46 +57,52 @@ class ConnectControlPage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :colorScheme) = Theme.of(context); + final ThemeData(:typography, :colorScheme) = Theme.of(context); final resolvedService = ref.watch(connectClientsProvider).asData?.value.resolvedService; + final connect = ref.watch(connectProvider); final connectNotifier = ref.read(connectProvider.notifier); final playlist = ref.watch(queueProvider); final playing = ref.watch(playingProvider); final shuffled = ref.watch(shuffleProvider); final loopMode = ref.watch(loopModeProvider); - final resumePauseStyle = IconButton.styleFrom( - backgroundColor: colorScheme.primary, - foregroundColor: colorScheme.onPrimary, - padding: const EdgeInsets.all(12), - iconSize: 24, - ); - final buttonStyle = IconButton.styleFrom( - backgroundColor: colorScheme.surface.withOpacity(0.4), - minimumSize: const Size(28, 28), - ); - - final activeButtonStyle = IconButton.styleFrom( - backgroundColor: colorScheme.primaryContainer, - foregroundColor: colorScheme.onPrimaryContainer, - minimumSize: const Size(28, 28), - ); - ref.listen(connectClientsProvider, (prev, next) { if (next.asData?.value.resolvedService == null) { - context.pop(); + context.back(); } }); + useEffect(() { + if (connect.asData?.value == null) return null; + + final subscription = connect.asData?.value?.stream.listen((message) { + final event = WebSocketEvent.fromJson( + jsonDecode(message), + (data) => data, + ); + event.onError((event) { + if (event.data != "Connection denied") return; + if (!context.mounted) return; + context.back(); + }); + }); + + return () { + subscription?.cancel(); + }; + }, [connect.asData?.value]); + return SafeArea( + bottom: false, child: Scaffold( - appBar: PageWindowTitleBar( - title: Text(resolvedService!.name), - automaticallyImplyLeading: true, - ), - body: LayoutBuilder(builder: (context, constrains) { + headers: [ + TitleBar( + title: Text(resolvedService?.name ?? ""), + ) + ], + child: LayoutBuilder(builder: (context, constrains) { return Row( children: [ Expanded( @@ -106,11 +116,11 @@ class ConnectControlPage extends HookConsumerWidget { vertical: 10, ).copyWith(top: 0), constraints: - const BoxConstraints(maxHeight: 400, maxWidth: 400), + const BoxConstraints(maxHeight: 350, maxWidth: 350), child: ClipRRect( borderRadius: BorderRadius.circular(20), child: UniversalImage( - path: (playlist.activeTrack?.album?.images) + path: (playlist.activeTrack?.album.images) .asUrlString( placeholder: ImagePlaceholder.albumArt, ), @@ -126,15 +136,11 @@ class ConnectControlPage extends HookConsumerWidget { SliverToBoxAdapter( child: AnchorButton( playlist.activeTrack?.name ?? "", - style: textTheme.titleLarge!, + style: typography.h4, onTap: () { if (playlist.activeTrack == null) return; - ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": playlist.activeTrack!.id!, - }, + context.navigateTo( + TrackRoute(trackId: playlist.activeTrack!.id), ); }, ), @@ -142,15 +148,10 @@ class ConnectControlPage extends HookConsumerWidget { SliverToBoxAdapter( child: ArtistLink( artists: playlist.activeTrack?.artists ?? [], - textStyle: textTheme.bodyMedium!, + textStyle: typography.normal, mainAxisAlignment: WrapAlignment.start, - onOverflowArtistClick: () => - ServiceUtils.pushNamed( - context, - TrackPage.name, - pathParameters: { - "id": playlist.activeTrack!.id!, - }, + onOverflowArtistClick: () => context.navigateTo( + TrackRoute(trackId: playlist.activeTrack!.id), ), ), ), @@ -164,19 +165,25 @@ class ConnectControlPage extends HookConsumerWidget { final position = ref.watch(positionProvider); final duration = ref.watch(durationProvider); + final progress = duration.inSeconds == 0 + ? 0 + : position.inSeconds / duration.inSeconds; + return Padding( padding: const EdgeInsets.symmetric(horizontal: 12), child: Column( children: [ Slider( - value: position > duration - ? 0 - : position.inSeconds.toDouble(), - min: 0, - max: duration.inSeconds.toDouble(), + value: + SliderValue.single(progress.toDouble()), onChanged: (value) { - connectNotifier - .seek(Duration(seconds: value.toInt())); + connectNotifier.seek( + Duration( + seconds: + (value.value * duration.inSeconds) + .toInt(), + ), + ); }, ), Row( @@ -196,94 +203,158 @@ class ConnectControlPage extends HookConsumerWidget { SliverToBoxAdapter( child: Row( mainAxisAlignment: MainAxisAlignment.center, + spacing: 20, children: [ - IconButton( - tooltip: shuffled - ? context.l10n.unshuffle_playlist - : context.l10n.shuffle_playlist, - icon: const Icon(SpotubeIcons.shuffle), - style: shuffled ? activeButtonStyle : buttonStyle, - onPressed: playlist.activeTrack == null - ? null - : () { - connectNotifier.setShuffle(!shuffled); - }, - ), - IconButton( - tooltip: context.l10n.previous_track, - icon: const Icon(SpotubeIcons.skipBack), - onPressed: playlist.activeTrack == null - ? null - : connectNotifier.previous, - ), - IconButton( - tooltip: playing - ? context.l10n.pause_playback - : context.l10n.resume_playback, - icon: playlist.activeTrack == null - ? SizedBox( - height: 20, - width: 20, - child: CircularProgressIndicator( - color: colorScheme.onPrimary, - ), - ) - : Icon( - playing - ? SpotubeIcons.pause - : SpotubeIcons.play, - ), - style: resumePauseStyle, - onPressed: playlist.activeTrack == null - ? null - : () { - if (playing) { - connectNotifier.pause(); - } else { - connectNotifier.resume(); - } - }, - ), - IconButton( - tooltip: context.l10n.next_track, - icon: const Icon(SpotubeIcons.skipForward), - onPressed: playlist.activeTrack == null - ? null - : connectNotifier.next, - ), - IconButton( - tooltip: loopMode == PlaylistMode.single - ? context.l10n.loop_track - : loopMode == PlaylistMode.loop - ? context.l10n.repeat_playlist - : null, - icon: Icon( - loopMode == PlaylistMode.single - ? SpotubeIcons.repeatOne - : SpotubeIcons.repeat, + Tooltip( + tooltip: TooltipContainer( + child: Text( + shuffled + ? context.l10n.unshuffle_playlist + : context.l10n.shuffle_playlist, + ), + ).call, + child: IconButton( + icon: const Icon(SpotubeIcons.shuffle), + variance: shuffled + ? ButtonVariance.secondary + : ButtonVariance.ghost, + onPressed: playlist.activeTrack == null + ? null + : () { + connectNotifier.setShuffle(!shuffled); + }, + ), + ), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.previous_track), + ).call, + child: IconButton.ghost( + icon: const Icon(SpotubeIcons.skipBack), + onPressed: playlist.activeTrack == null + ? null + : connectNotifier.previous, + ), + ), + Tooltip( + tooltip: TooltipContainer( + child: Text( + playing + ? context.l10n.pause_playback + : context.l10n.resume_playback, + ), + ).call, + child: IconButton.primary( + shape: ButtonShape.circle, + icon: playlist.activeTrack == null + ? const SizedBox( + height: 20, + width: 20, + child: CircularProgressIndicator( + onSurface: false), + ) + : Icon( + playing + ? SpotubeIcons.pause + : SpotubeIcons.play, + ), + onPressed: playlist.activeTrack == null + ? null + : () { + if (playing) { + connectNotifier.pause(); + } else { + connectNotifier.resume(); + } + }, + ), + ), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.next_track)) + .call, + child: IconButton.ghost( + icon: const Icon(SpotubeIcons.skipForward), + onPressed: playlist.activeTrack == null + ? null + : connectNotifier.next, + ), + ), + Tooltip( + tooltip: TooltipContainer( + child: Text( + loopMode == PlaylistMode.single + ? context.l10n.loop_track + : loopMode == PlaylistMode.loop + ? context.l10n.repeat_playlist + : context.l10n.no_loop, + ), + ).call, + child: IconButton( + icon: Icon( + loopMode == PlaylistMode.single + ? SpotubeIcons.repeatOne + : SpotubeIcons.repeat, + ), + variance: loopMode == PlaylistMode.single || + loopMode == PlaylistMode.loop + ? ButtonVariance.secondary + : ButtonVariance.ghost, + onPressed: playlist.activeTrack == null + ? null + : () async { + connectNotifier.setLoopMode( + switch (loopMode) { + PlaylistMode.loop => + PlaylistMode.single, + PlaylistMode.single => + PlaylistMode.none, + PlaylistMode.none => + PlaylistMode.loop, + }, + ); + }, ), - style: loopMode == PlaylistMode.single || - loopMode == PlaylistMode.loop - ? activeButtonStyle - : buttonStyle, - onPressed: playlist.activeTrack == null - ? null - : () async { - connectNotifier.setLoopMode( - switch (loopMode) { - PlaylistMode.loop => - PlaylistMode.single, - PlaylistMode.single => - PlaylistMode.none, - PlaylistMode.none => PlaylistMode.loop, - }, - ); - }, ) ], ), ), const SliverGap(30), + if (constrains.mdAndDown) + SliverPadding( + padding: const EdgeInsets.symmetric(horizontal: 20), + sliver: SliverToBoxAdapter( + child: Button.outline( + leading: const Icon(SpotubeIcons.queue), + child: Text(context.l10n.queue), + onPressed: () { + openDrawer( + context: context, + barrierDismissible: true, + draggable: true, + barrierColor: Colors.black.withAlpha(100), + borderRadius: BorderRadius.circular(10), + transformBackdrop: false, + position: OverlayPosition.bottom, + surfaceBlur: context.theme.surfaceBlur, + surfaceOpacity: 0.7, + expands: true, + builder: (context) { + return ConstrainedBox( + constraints: BoxConstraints( + maxHeight: + MediaQuery.sizeOf(context).height * + 0.8, + ), + child: const RemotePlayerQueue(), + ); + }, + ); + }, + ), + ), + ), + const SliverGap(30), SliverPadding( padding: const EdgeInsets.symmetric(horizontal: 20), sliver: SliverToBoxAdapter( @@ -300,25 +371,7 @@ class ConnectControlPage extends HookConsumerWidget { }), ), ), - const SliverGap(30), - if (constrains.mdAndDown) - SliverPadding( - padding: const EdgeInsets.symmetric(horizontal: 20), - sliver: SliverToBoxAdapter( - child: OutlinedButton.icon( - icon: const Icon(SpotubeIcons.queue), - label: Text(context.l10n.queue), - onPressed: () { - showModalBottomSheet( - context: context, - builder: (context) { - return const RemotePlayerQueue(); - }, - ); - }, - ), - ), - ) + const SliverSafeArea(sliver: SliverGap(10)), ], ), ), diff --git a/lib/pages/getting_started/getting_started.dart b/lib/pages/getting_started/getting_started.dart index 0159a77f..1662624c 100644 --- a/lib/pages/getting_started/getting_started.dart +++ b/lib/pages/getting_started/getting_started.dart @@ -1,6 +1,6 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/assets.gen.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/context.dart'; @@ -8,22 +8,16 @@ import 'package:spotube/pages/getting_started/sections/greeting.dart'; import 'package:spotube/pages/getting_started/sections/playback.dart'; import 'package:spotube/pages/getting_started/sections/region.dart'; import 'package:spotube/pages/getting_started/sections/support.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:spotube/themes/theme.dart'; +import 'package:auto_route/auto_route.dart'; -class GettingStarting extends HookConsumerWidget { +@RoutePage() +class GettingStartedPage extends HookConsumerWidget { static const name = "getting_started"; - const GettingStarting({super.key}); + const GettingStartedPage({super.key}); @override Widget build(BuildContext context, ref) { - final preferences = ref.watch(userPreferencesProvider); - final themeData = theme( - preferences.accentColorScheme, - Brightness.dark, - preferences.amoledDarkTheme, - ); final pageController = usePageController(); final onNext = useCallback(() { @@ -40,66 +34,59 @@ class GettingStarting extends HookConsumerWidget { ); }, [pageController]); - return Theme( - data: themeData, - child: Scaffold( - appBar: PageWindowTitleBar( - backgroundColor: Colors.transparent, - actions: [ - ListenableBuilder( - listenable: pageController, - builder: (context, _) { - return AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: pageController.hasClients && - (pageController.page == 0 || pageController.page == 3) - ? const SizedBox() - : TextButton( - onPressed: () { - pageController.animateToPage( - 3, - duration: const Duration(milliseconds: 300), - curve: Curves.easeInOut, - ); - }, - child: Text( - context.l10n.skip_this_nonsense, - style: TextStyle( - decoration: TextDecoration.underline, - decorationColor: themeData.colorScheme.primary, - ), + return Scaffold( + headers: [ + SafeArea( + child: TitleBar( + backgroundColor: Colors.transparent, + surfaceBlur: 0, + trailing: [ + ListenableBuilder( + listenable: pageController, + builder: (context, _) { + return AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + child: pageController.hasClients && + (pageController.page == 0 || + pageController.page == 3) + ? const SizedBox() + : Button.secondary( + onPressed: () { + pageController.animateToPage( + 3, + duration: const Duration(milliseconds: 300), + curve: Curves.easeInOut, + ); + }, + child: Text(context.l10n.skip_this_nonsense), ), - ), - ); - }, - ), - ], - ), - extendBodyBehindAppBar: true, - body: DecoratedBox( - decoration: BoxDecoration( - image: DecorationImage( - image: Assets.bengaliPatternsBg.provider(), - fit: BoxFit.cover, - colorFilter: const ColorFilter.mode( - Colors.black38, - BlendMode.srcOver, + ); + }, ), - ), - ), - child: PageView( - controller: pageController, - children: [ - GettingStartedPageGreetingSection(onNext: onNext), - GettingStartedPageLanguageRegionSection(onNext: onNext), - GettingStartedPagePlaybackSection( - onNext: onNext, - onPrevious: onPrevious, - ), - const GettingStartedScreenSupportSection(), ], ), ), + ], + floatingHeader: true, + child: DecoratedBox( + decoration: BoxDecoration( + image: DecorationImage( + image: Assets.images.bengaliPatternsBg.provider(), + fit: BoxFit.cover, + ), + ), + child: PageView( + controller: pageController, + children: [ + GettingStartedPageGreetingSection(onNext: onNext), + GettingStartedPageLanguageRegionSection(onNext: onNext), + GettingStartedPagePlaybackSection( + onNext: onNext, + onPrevious: onPrevious, + ), + const GettingStartedScreenSupportSection(), + ], + ), ), ); } diff --git a/lib/pages/getting_started/sections/greeting.dart b/lib/pages/getting_started/sections/greeting.dart index 6d649351..68903e07 100644 --- a/lib/pages/getting_started/sections/greeting.dart +++ b/lib/pages/getting_started/sections/greeting.dart @@ -1,10 +1,9 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/assets.gen.dart'; import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/getting_started/blur_card.dart'; import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/getting_started/blur_card.dart'; import 'package:spotube/utils/platform.dart'; class GettingStartedPageGreetingSection extends HookConsumerWidget { @@ -13,39 +12,26 @@ class GettingStartedPageGreetingSection extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); - return Center( child: BlurCard( child: Column( mainAxisSize: MainAxisSize.min, children: [ - Assets.spotubeLogoPng.image(height: 200), + Assets.branding.spotubeLogoPng.image(height: 200), const Gap(24), - Text( - "Spotube", - style: - textTheme.titleLarge?.copyWith(fontWeight: FontWeight.bold), - ), + const Text("Spotube").semiBold().h4(), const Gap(4), Text( kIsMobile ? context.l10n.freedom_of_music_palm : context.l10n.freedom_of_music, textAlign: TextAlign.center, - style: textTheme.titleMedium?.copyWith( - fontWeight: FontWeight.w300, - fontStyle: FontStyle.italic, - ), - ), + ).light().large().italic(), const Gap(84), - Directionality( - textDirection: TextDirection.rtl, - child: FilledButton.icon( - onPressed: onNext, - icon: const Icon(SpotubeIcons.angleRight), - label: Text(context.l10n.get_started), - ), + Button.primary( + onPressed: onNext, + trailing: const Icon(SpotubeIcons.angleRight), + child: Text(context.l10n.get_started), ), ], ), diff --git a/lib/pages/getting_started/sections/playback.dart b/lib/pages/getting_started/sections/playback.dart index dbf0bda2..699024b1 100644 --- a/lib/pages/getting_started/sections/playback.dart +++ b/lib/pages/getting_started/sections/playback.dart @@ -1,29 +1,11 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/assets.gen.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/models/database/database.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/modules/getting_started/blur_card.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/string.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -final audioSourceToIconMap = { - AudioSource.youtube: const Icon( - SpotubeIcons.youtube, - color: Colors.red, - size: 30, - ), - AudioSource.piped: const Icon(SpotubeIcons.piped, size: 30), - AudioSource.invidious: ClipRRect( - borderRadius: BorderRadius.circular(48), - child: Assets.invidious.image(width: 48, height: 48), - ), - AudioSource.jiosaavn: Assets.jiosaavn.image(width: 48, height: 48), -}; - class GettingStartedPagePlaybackSection extends HookConsumerWidget { final VoidCallback onNext; final VoidCallback onPrevious; @@ -36,22 +18,22 @@ class GettingStartedPagePlaybackSection extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :colorScheme, :dividerColor) = - Theme.of(context); final preferences = ref.watch(userPreferencesProvider); final preferencesNotifier = ref.read(userPreferencesProvider.notifier); - final audioSourceToDescription = useMemoized( - () => { - AudioSource.youtube: "${context.l10n.youtube_source_description}\n" - "${context.l10n.highest_quality("148kbps mp4, 128kbps opus")}", - AudioSource.piped: context.l10n.piped_source_description, - AudioSource.jiosaavn: - "${context.l10n.jiosaavn_source_description}\n" - "${context.l10n.highest_quality("320kbps mp")}", - AudioSource.invidious: context.l10n.invidious_source_description, - }, - []); + // final audioSourceToDescription = useMemoized( + // () => { + // AudioSource.youtube: "${context.l10n.youtube_source_description}\n" + // "${context.l10n.highest_quality("148kbps mp4, 128kbps opus")}", + // AudioSource.piped: context.l10n.piped_source_description, + // AudioSource.jiosaavn: + // "${context.l10n.jiosaavn_source_description}\n" + // "${context.l10n.highest_quality("320kbps mp4")}", + // AudioSource.invidious: context.l10n.invidious_source_description, + // AudioSource.dabMusic: "${context.l10n.dab_music_source_description}\n" + // "${context.l10n.highest_quality("320kbps mp3, HI-RES 24bit 44.1kHz-96kHz flac")}", + // }, + // []); return Center( child: BlurCard( @@ -62,76 +44,55 @@ class GettingStartedPagePlaybackSection extends HookConsumerWidget { children: [ const Icon(SpotubeIcons.album, size: 16), const Gap(8), - Text(context.l10n.playback, style: textTheme.titleMedium), + Text(context.l10n.playback).semiBold().large(), ], ), const Gap(16), - ListTile( - title: Text( - context.l10n.select_audio_source, - style: textTheme.titleMedium, - ), - ), + // Align( + // alignment: Alignment.centerLeft, + // child: Text(context.l10n.select_audio_source).semiBold().large(), + // ), + // const Gap(16), + // RadioGroup( + // value: preferences.audioSource, + // onChanged: (value) { + // preferencesNotifier.setAudioSource(value); + // }, + // child: Wrap( + // spacing: 6, + // runSpacing: 6, + // children: [ + // for (final source in AudioSource.values) + // Badge( + // isLabelVisible: source == AudioSource.dabMusic, + // label: const Text("NEW"), + // backgroundColor: Colors.lime[300], + // textColor: Colors.black, + // child: RadioCard( + // value: source, + // child: Column( + // mainAxisSize: MainAxisSize.min, + // children: [ + // audioSourceToIconMap[source]!, + // Text(source.label), + // ], + // ), + // ), + // ), + // ], + // ), + // ), + // const Gap(16), + // Text( + // audioSourceToDescription[preferences.audioSource]!, + // ).small().muted(), const Gap(16), - ToggleButtons( - isSelected: [ - for (final source in AudioSource.values) - preferences.audioSource == source, - ], - onPressed: (index) { - preferencesNotifier.setAudioSource(AudioSource.values[index]); - }, - borderRadius: BorderRadius.circular(8), - children: [ - for (final source in AudioSource.values) - SizedBox.square( - dimension: 84, - child: Column( - mainAxisSize: MainAxisSize.min, - mainAxisAlignment: MainAxisAlignment.center, - children: [ - audioSourceToIconMap[source]!, - const Gap(8), - Text( - source.name.capitalize(), - style: textTheme.bodySmall!.copyWith( - color: preferences.audioSource == source - ? colorScheme.primary - : null, - ), - ), - ], - ), - ), - ], - ), - ListTile( - title: Align( - alignment: switch (preferences.audioSource) { - AudioSource.youtube => Alignment.centerLeft, - AudioSource.piped || - AudioSource.invidious => - Alignment.center, - AudioSource.jiosaavn => Alignment.centerRight, - }, - child: Text( - audioSourceToDescription[preferences.audioSource]!, - style: textTheme.bodySmall?.copyWith( - color: dividerColor, - ), - ), - ), - ), - const Gap(16), - ListTile( + ButtonTile( title: Text(context.l10n.endless_playback), subtitle: Text( context.l10n.endless_playback_description, - style: textTheme.bodySmall?.copyWith( - color: dividerColor, - ), - ), - onTap: () { + ).small().muted(), + onPressed: () { preferencesNotifier .setEndlessPlayback(!preferences.endlessPlayback); }, @@ -146,17 +107,17 @@ class GettingStartedPagePlaybackSection extends HookConsumerWidget { Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ - FilledButton.icon( - icon: const Icon(SpotubeIcons.angleLeft), - label: Text(context.l10n.previous), + Button.secondary( + leading: const Icon(SpotubeIcons.angleLeft), onPressed: onPrevious, + child: Text(context.l10n.previous), ), Directionality( textDirection: TextDirection.rtl, - child: FilledButton.icon( - icon: const Icon(SpotubeIcons.angleRight), - label: Text(context.l10n.next), + child: Button.primary( + leading: const Icon(SpotubeIcons.angleRight), onPressed: onNext, + child: Text(context.l10n.next), ), ), ], diff --git a/lib/pages/getting_started/sections/region.dart b/lib/pages/getting_started/sections/region.dart index 9e31a273..917cc41e 100644 --- a/lib/pages/getting_started/sections/region.dart +++ b/lib/pages/getting_started/sections/region.dart @@ -1,8 +1,7 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/language_codes.dart'; -import 'package:spotube/collections/spotify_markets.dart'; +import 'package:spotube/collections/markets.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/getting_started/blur_card.dart'; import 'package:spotube/extensions/context.dart'; @@ -14,9 +13,24 @@ class GettingStartedPageLanguageRegionSection extends HookConsumerWidget { const GettingStartedPageLanguageRegionSection( {super.key, required this.onNext}); + bool filterMarkets(dynamic item, String query) { + final market = + marketsMap.firstWhere((element) => element.$1 == item).$2.toLowerCase(); + + return market.contains(query.toLowerCase()); + } + + bool filterLocale(Locale locale, String query) { + final language = LanguageLocals.getDisplayLanguage( + locale.languageCode, + locale.countryCode, + ).toString(); + + return language.toLowerCase().contains(query.toLowerCase()); + } + @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :dividerColor) = Theme.of(context); final preferences = ref.watch(userPreferencesProvider); return SafeArea( @@ -32,92 +46,143 @@ class GettingStartedPageLanguageRegionSection extends HookConsumerWidget { size: 16, ), const SizedBox(width: 8), - Text( - context.l10n.language_region, - style: textTheme.titleMedium, - ), + Text(context.l10n.language_region).semiBold(), ], ), - const Gap(48), + const Gap(30), Column( crossAxisAlignment: CrossAxisAlignment.start, mainAxisSize: MainAxisSize.min, children: [ - Text( - context.l10n.choose_your_region, - style: textTheme.titleSmall, - ), + Text(context.l10n.choose_your_region).semiBold(), Text( context.l10n.choose_your_region_description, - style: textTheme.bodySmall?.copyWith( - color: dividerColor, - ), - ), + ).small().muted(), const Gap(16), - DropdownMenu( - initialSelection: preferences.market, - onSelected: (value) { - if (value == null) return; - ref - .read(userPreferencesProvider.notifier) - .setRecommendationMarket(value); - }, - hintText: preferences.market.name, - label: Text(context.l10n.market_place_region), - inputDecorationTheme: - const InputDecorationTheme(isDense: true), - dropdownMenuEntries: [ - for (final market in spotifyMarkets) - DropdownMenuEntry( - value: market.$1, - label: market.$2, - ), - ], + Text(context.l10n.market_place_region).small(), + const Gap(8), + SizedBox( + width: double.infinity, + child: Select( + value: preferences.market, + onChanged: (value) { + if (value == null) return; + ref + .read(userPreferencesProvider.notifier) + .setRecommendationMarket(value); + }, + placeholder: Text(preferences.market.name), + itemBuilder: (context, value) => Text( + marketsMap + .firstWhere((element) => element.$1 == value) + .$2, + ), + popup: SelectPopup.builder( + searchPlaceholder: Text(context.l10n.search), + builder: (context, searchQuery) { + final filteredMarkets = searchQuery == null || + searchQuery.isEmpty + ? marketsMap + : marketsMap + .where( + (element) => + filterMarkets(element.$1, searchQuery), + ) + .toList(); + return SelectItemBuilder( + childCount: filteredMarkets.length, + builder: (context, index) { + final market = filteredMarkets[index]; + return SelectItemButton( + value: market.$1, + child: Text(market.$2), + ); + }, + ); + }, + ).call, + ), ), const Gap(36), Text( context.l10n.choose_your_language, - style: textTheme.titleSmall, - ), + ).semiBold(), const Gap(16), - DropdownMenu( - initialSelection: preferences.locale, - onSelected: (locale) { - if (locale == null) return; - ref - .read(userPreferencesProvider.notifier) - .setLocale(locale); - }, - hintText: context.l10n.system_default, - label: Text(context.l10n.language), - inputDecorationTheme: - const InputDecorationTheme(isDense: true), - dropdownMenuEntries: [ - DropdownMenuEntry( - value: const Locale("system", "system"), - label: context.l10n.system_default, - ), - for (final locale in L10n.all) - DropdownMenuEntry( - value: locale, - label: LanguageLocals.getDisplayLanguage( - locale.languageCode) - .toString(), - ), - ], + Text(context.l10n.language).small(), + const Gap(8), + SizedBox( + width: double.infinity, + child: Select( + value: preferences.locale, + onChanged: (locale) { + if (locale == null) return; + ref + .read(userPreferencesProvider.notifier) + .setLocale(locale); + }, + placeholder: Text(context.l10n.system_default), + itemBuilder: (context, value) => + value.languageCode == "system" + ? Text(context.l10n.system_default) + : Text( + LanguageLocals.getDisplayLanguage( + value.languageCode, + value.countryCode, + ).toString(), + ), + popup: SelectPopup.builder( + searchPlaceholder: Text(context.l10n.search), + builder: (context, searchQuery) { + final hasNotQueried = + searchQuery == null || searchQuery.trim().isEmpty; + final filteredLocale = hasNotQueried + ? [ + const Locale("system", "system"), + ...L10n.all, + ] + : L10n.all + .where( + (element) => filterLocale( + element, + searchQuery.trim(), + ), + ) + .toList(); + + return SelectItemBuilder( + childCount: filteredLocale.length, + builder: (context, index) { + final locale = filteredLocale[index]; + if (locale == const Locale("system", "system")) { + return SelectItemButton( + value: locale, + child: Text(context.l10n.system_default), + ); + } + return SelectItemButton( + value: locale, + child: Text( + LanguageLocals.getDisplayLanguage( + locale.languageCode, + locale.countryCode, + ).toString(), + ), + ); + }, + ); + }, + ).call, + ), ), ], ), const Gap(48), Align( alignment: Alignment.centerRight, - child: Directionality( - textDirection: TextDirection.rtl, - child: FilledButton.icon( - icon: const Icon(SpotubeIcons.angleRight), - label: Text(context.l10n.next), - onPressed: onNext, - ), + child: Button.primary( + trailing: const Icon(SpotubeIcons.angleRight), + onPressed: onNext, + child: Text(context.l10n.next), ), ), ], diff --git a/lib/pages/getting_started/sections/support.dart b/lib/pages/getting_started/sections/support.dart index f09a585d..ef549296 100644 --- a/lib/pages/getting_started/sections/support.dart +++ b/lib/pages/getting_started/sections/support.dart @@ -1,13 +1,11 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/env.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/getting_started/blur_card.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/home/home.dart'; -import 'package:spotube/pages/mobile_login/hooks/login_callback.dart'; import 'package:spotube/services/kv_store/kv_store.dart'; import 'package:url_launcher/url_launcher_string.dart'; @@ -16,9 +14,6 @@ class GettingStartedScreenSupportSection extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :colorScheme) = Theme.of(context); - final onLogin = useLoginCallback(ref); - return Center( child: Column( mainAxisSize: MainAxisSize.min, @@ -34,9 +29,8 @@ class GettingStartedScreenSupportSection extends HookConsumerWidget { const SizedBox(width: 8), Text( context.l10n.help_project_grow, - style: - textTheme.titleMedium?.copyWith(color: Colors.pink), - ), + style: const TextStyle(color: Colors.pink), + ).semiBold(), ], ), const Gap(16), @@ -46,38 +40,57 @@ class GettingStartedScreenSupportSection extends HookConsumerWidget { mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.stretch, children: [ - FilledButton.icon( - icon: const Icon(SpotubeIcons.github), - label: Text(context.l10n.contribute_on_github), - style: FilledButton.styleFrom( - backgroundColor: Colors.black, - foregroundColor: Colors.white, - shape: RoundedRectangleBorder( + Button( + leading: const Icon(SpotubeIcons.github), + style: ButtonVariance.primary.copyWith( + decoration: (context, states, value) { + if (states.isNotEmpty) { + return ButtonVariance.primary + .decoration(context, states); + } + + return BoxDecoration( + color: Colors.black, borderRadius: BorderRadius.circular(8), - ), - ), + ); + }), onPressed: () async { await launchUrlString( "https://github.com/KRTirtho/spotube", mode: LaunchMode.externalApplication, ); }, + child: Text( + context.l10n.contribute_on_github, + style: const TextStyle(color: Colors.white), + ), ), if (!Env.hideDonations) ...[ const Gap(16), - FilledButton.icon( - icon: const Icon(SpotubeIcons.openCollective), - label: Text(context.l10n.donate_on_open_collective), - style: FilledButton.styleFrom( - backgroundColor: const Color(0xff4cb7f6), - foregroundColor: Colors.white, - ), + Button( + leading: const Icon(SpotubeIcons.openCollective), + style: ButtonVariance.primary.copyWith( + decoration: (context, states, value) { + if (states.isNotEmpty) { + return ButtonVariance.primary + .decoration(context, states); + } + + return BoxDecoration( + color: const Color(0xff4cb7f6), + borderRadius: BorderRadius.circular(8), + ); + }), onPressed: () async { await launchUrlString( "https://opencollective.com/spotube", mode: LaunchMode.externalApplication, ); }, + child: Text( + context.l10n.donate_on_open_collective, + style: const TextStyle(color: Colors.white), + ), ), ] ], @@ -91,42 +104,15 @@ class GettingStartedScreenSupportSection extends HookConsumerWidget { child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ - DecoratedBox( - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(8), - gradient: LinearGradient( - colors: [ - colorScheme.primary, - colorScheme.secondary, - ], - ), - ), - child: TextButton.icon( - icon: const Icon(SpotubeIcons.anonymous), - label: Text(context.l10n.browse_anonymously), - style: TextButton.styleFrom( - foregroundColor: Colors.white, - ), - onPressed: () async { - await KVStoreService.setDoneGettingStarted(true); - if (context.mounted) { - context.goNamed(HomePage.name); - } - }, - ), - ), - const Gap(16), - FilledButton.icon( - icon: const Icon(SpotubeIcons.spotify), - label: Text(context.l10n.connect_with_spotify), - style: FilledButton.styleFrom( - backgroundColor: const Color(0xff1db954), - foregroundColor: Colors.white, - ), + Button.primary( + leading: const Icon(SpotubeIcons.extensions), onPressed: () async { await KVStoreService.setDoneGettingStarted(true); - await onLogin(); + if (context.mounted) { + context.pushRoute(const SettingsMetadataProviderRoute()); + } }, + child: Text(context.l10n.install_a_metadata_provider), ), ], ), diff --git a/lib/pages/home/feed/feed_section.dart b/lib/pages/home/feed/feed_section.dart deleted file mode 100644 index bcfc0b81..00000000 --- a/lib/pages/home/feed/feed_section.dart +++ /dev/null @@ -1,69 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/modules/album/album_card.dart'; -import 'package:spotube/modules/artist/artist_card.dart'; -import 'package:spotube/modules/playlist/playlist_card.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/provider/spotify/views/home_section.dart'; - -class HomeFeedSectionPage extends HookConsumerWidget { - static const name = "home_feed_section"; - - final String sectionUri; - const HomeFeedSectionPage({super.key, required this.sectionUri}); - - @override - Widget build(BuildContext context, ref) { - final homeFeedSection = ref.watch(homeSectionViewProvider(sectionUri)); - final section = homeFeedSection.asData?.value ?? FakeData.feedSection; - - return Skeletonizer( - enabled: homeFeedSection.isLoading, - child: Scaffold( - appBar: PageWindowTitleBar( - title: Text(section.title ?? ""), - centerTitle: false, - automaticallyImplyLeading: true, - titleSpacing: 0, - ), - body: CustomScrollView( - slivers: [ - SliverLayoutBuilder( - builder: (context, constrains) { - return SliverGrid.builder( - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: 200, - mainAxisExtent: constrains.smAndDown ? 225 : 250, - crossAxisSpacing: 8, - mainAxisSpacing: 8, - ), - itemCount: section.items.length, - itemBuilder: (context, index) { - final item = section.items[index]; - - if (item.album != null) { - return AlbumCard(item.album!.asAlbum); - } else if (item.artist != null) { - return ArtistCard(item.artist!.asArtist); - } else if (item.playlist != null) { - return PlaylistCard(item.playlist!.asPlaylist); - } - return const SizedBox(); - }, - ); - }, - ), - const SliverToBoxAdapter( - child: SafeArea( - child: SizedBox(), - ), - ), - ], - ), - ), - ); - } -} diff --git a/lib/pages/home/genres/genre_playlists.dart b/lib/pages/home/genres/genre_playlists.dart deleted file mode 100644 index 04658965..00000000 --- a/lib/pages/home/genres/genre_playlists.dart +++ /dev/null @@ -1,163 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotify/spotify.dart' hide Offset; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/hooks/utils/use_custom_status_bar_color.dart'; -import 'package:spotube/modules/playlist/playlist_card.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/components/waypoint.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; -import 'package:collection/collection.dart'; -import 'package:spotube/utils/platform.dart'; - -class GenrePlaylistsPage extends HookConsumerWidget { - static const name = "genre_playlists"; - - final Category category; - const GenrePlaylistsPage({super.key, required this.category}); - - @override - Widget build(BuildContext context, ref) { - final mediaQuery = MediaQuery.of(context); - final playlists = ref.watch(categoryPlaylistsProvider(category.id!)); - final playlistsNotifier = - ref.read(categoryPlaylistsProvider(category.id!).notifier); - final scrollController = useScrollController(); - final routeName = GoRouterState.of(context).name; - - useCustomStatusBarColor( - Colors.black, - routeName == GenrePlaylistsPage.name, - noSetBGColor: true, - automaticSystemUiAdjustment: false, - ); - - return Scaffold( - appBar: kIsDesktop - ? const PageWindowTitleBar( - leading: BackButton(color: Colors.white), - backgroundColor: Colors.transparent, - foregroundColor: Colors.white, - ) - : null, - extendBodyBehindAppBar: true, - body: DecoratedBox( - decoration: BoxDecoration( - image: DecorationImage( - image: UniversalImage.imageProvider(category.icons!.first.url!), - alignment: Alignment.topCenter, - fit: BoxFit.cover, - colorFilter: ColorFilter.mode( - Colors.black.withOpacity(0.5), - BlendMode.darken, - ), - repeat: ImageRepeat.noRepeat, - matchTextDirection: true, - ), - ), - child: CustomScrollView( - controller: scrollController, - slivers: [ - SliverAppBar( - automaticallyImplyLeading: kIsMobile, - expandedHeight: mediaQuery.mdAndDown ? 200 : 150, - title: const Text(""), - backgroundColor: Colors.transparent, - flexibleSpace: FlexibleSpaceBar( - centerTitle: kIsDesktop, - title: Text( - category.name!, - style: Theme.of(context).textTheme.headlineMedium?.copyWith( - color: Colors.white, - letterSpacing: 3, - shadows: [ - const Shadow( - offset: Offset(-1.5, -1.5), - color: Colors.black54, - ), - const Shadow( - offset: Offset(1.5, -1.5), - color: Colors.black54, - ), - const Shadow( - offset: Offset(1.5, 1.5), - color: Colors.black54, - ), - const Shadow( - offset: Offset(-1.5, 1.5), - color: Colors.black54, - ), - ], - ), - ), - collapseMode: CollapseMode.parallax, - ), - ), - const SliverGap(20), - SliverSafeArea( - top: false, - sliver: SliverPadding( - padding: EdgeInsets.symmetric( - horizontal: mediaQuery.mdAndDown ? 12 : 24, - ), - sliver: playlists.asData?.value.items.isNotEmpty != true - ? Skeletonizer.sliver( - child: SliverToBoxAdapter( - child: Wrap( - spacing: 12, - runSpacing: 12, - children: List.generate( - 6, - (index) => PlaylistCard(FakeData.playlist), - ), - ), - ), - ) - : SliverGrid.builder( - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: 190, - mainAxisExtent: mediaQuery.mdAndDown ? 225 : 250, - crossAxisSpacing: 12, - mainAxisSpacing: 12, - ), - itemCount: - (playlists.asData?.value.items.length ?? 0) + 1, - itemBuilder: (context, index) { - final playlist = playlists.asData?.value.items - .elementAtOrNull(index); - - if (playlist == null) { - if (playlists.asData?.value.hasMore == false) { - return const SizedBox.shrink(); - } - return Skeletonizer( - enabled: true, - child: Waypoint( - controller: scrollController, - isGrid: true, - onTouchEdge: playlistsNotifier.fetchMore, - child: PlaylistCard(FakeData.playlist), - ), - ); - } - - return Skeleton.keep( - child: PlaylistCard(playlist), - ); - }, - ), - ), - ), - const SliverGap(20), - ], - ), - ), - ); - } -} diff --git a/lib/pages/home/genres/genres.dart b/lib/pages/home/genres/genres.dart deleted file mode 100644 index 4846d633..00000000 --- a/lib/pages/home/genres/genres.dart +++ /dev/null @@ -1,98 +0,0 @@ -import 'dart:math'; - -import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/gradients.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/pages/home/genres/genre_playlists.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class GenrePage extends HookConsumerWidget { - static const name = "genre"; - const GenrePage({super.key}); - - @override - Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); - final scrollController = useScrollController(); - final categories = ref.watch(categoriesProvider); - - final mediaQuery = MediaQuery.of(context); - - return Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.explore_genres), - automaticallyImplyLeading: true, - titleSpacing: 0, - ), - body: SafeArea( - top: false, - child: GridView.builder( - padding: const EdgeInsets.all(12), - controller: scrollController, - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - childAspectRatio: 9 / 18, - maxCrossAxisExtent: mediaQuery.smAndDown ? 200 : 300, - mainAxisExtent: 200, - crossAxisSpacing: 12, - mainAxisSpacing: 12, - ), - itemCount: categories.asData!.value.length, - itemBuilder: (context, index) { - final category = categories.asData!.value[index]; - final gradient = gradients[Random().nextInt(gradients.length)]; - return InkWell( - borderRadius: BorderRadius.circular(8), - onTap: () { - context.pushNamed( - GenrePlaylistsPage.name, - pathParameters: { - "categoryId": category.id!, - }, - extra: category, - ); - }, - child: Ink( - padding: const EdgeInsets.all(8), - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(8), - image: DecorationImage( - image: NetworkImage(category.icons!.first.url!), - fit: BoxFit.cover, - ), - gradient: gradient, - ), - child: Align( - alignment: Alignment.bottomCenter, - child: AutoSizeText( - category.name!, - style: textTheme.titleLarge?.copyWith( - color: Colors.white, - shadows: [ - // stroke shadow - const Shadow( - color: Colors.black, - offset: Offset(1, 1), - blurRadius: 2, - ), - ], - ), - maxLines: 1, - textAlign: TextAlign.center, - maxFontSize: textTheme.titleLarge!.fontSize!, - minFontSize: textTheme.titleMedium!.fontSize!, - ), - ), - ), - ); - }, - ), - ), - ); - } -} diff --git a/lib/pages/home/home.dart b/lib/pages/home/home.dart index efdca4f7..a92c776e 100644 --- a/lib/pages/home/home.dart +++ b/lib/pages/home/home.dart @@ -1,31 +1,29 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/assets.gen.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/models/database/database.dart'; import 'package:spotube/modules/connect/connect_device.dart'; import 'package:spotube/modules/home/sections/featured.dart'; -import 'package:spotube/modules/home/sections/feed.dart'; -import 'package:spotube/modules/home/sections/friends.dart'; -import 'package:spotube/modules/home/sections/genres.dart'; -import 'package:spotube/modules/home/sections/made_for_user.dart'; +import 'package:spotube/modules/home/sections/sections.dart'; import 'package:spotube/modules/home/sections/new_releases.dart'; import 'package:spotube/modules/home/sections/recent.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/pages/settings/settings.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/utils/platform.dart'; -import 'package:spotube/utils/service_utils.dart'; +@RoutePage() class HomePage extends HookConsumerWidget { static const name = "home"; const HomePage({super.key}); @override Widget build(BuildContext context, ref) { + final theme = Theme.of(context); final controller = useScrollController(); final mediaQuery = MediaQuery.of(context); final layoutMode = @@ -34,21 +32,33 @@ class HomePage extends HookConsumerWidget { return SafeArea( bottom: false, child: Scaffold( - appBar: kIsMobile || kIsMacOS ? null : const PageWindowTitleBar(), - body: CustomScrollView( + headers: [ + if (kTitlebarVisible) const TitleBar(height: 30), + ], + child: CustomScrollView( controller: controller, slivers: [ if (mediaQuery.smAndDown || layoutMode == LayoutMode.compact) SliverAppBar( floating: true, - title: Assets.spotubeLogoPng.image(height: 45), + title: DefaultTextStyle( + style: TextStyle( + fontFamily: "Cookie", + fontSize: 30, + letterSpacing: 1.8, + color: theme.colorScheme.foreground, + ), + child: const Text("Spotube"), + ), + backgroundColor: theme.colorScheme.background, + foregroundColor: theme.colorScheme.foreground, actions: [ const ConnectDeviceButton(), const Gap(10), - IconButton( + IconButton.ghost( icon: const Icon(SpotubeIcons.settings, size: 20), onPressed: () { - ServiceUtils.pushNamed(context, SettingsPage.name); + context.navigateTo(const SettingsRoute()); }, ), const Gap(10), @@ -56,14 +66,20 @@ class HomePage extends HookConsumerWidget { ) else if (kIsMacOS) const SliverGap(10), - const HomeGenresSection(), const SliverGap(10), - const SliverToBoxAdapter(child: HomeRecentlyPlayedSection()), - const SliverToBoxAdapter(child: HomeFeaturedSection()), - const HomePageFriendsSection(), - const SliverToBoxAdapter(child: HomeNewReleasesSection()), - const HomePageFeedSection(), - const SliverSafeArea(sliver: HomeMadeForUserSection()), + SliverList.builder( + itemCount: 3, + itemBuilder: (context, index) { + return switch (index) { + // 0 => const HomeGenresSection(), + 0 => const HomeRecentlyPlayedSection(), + 1 => const HomeFeaturedSection(), + // 3 => const HomePageFriendsSection(), + _ => const HomeNewReleasesSection() + }; + }, + ), + const SliverSafeArea(sliver: HomePageBrowseSection()), ], ), )); diff --git a/lib/pages/home/sections/section_items.dart b/lib/pages/home/sections/section_items.dart new file mode 100644 index 00000000..89666d26 --- /dev/null +++ b/lib/pages/home/sections/section_items.dart @@ -0,0 +1,122 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/components/playbutton_view/playbutton_card.dart'; +import 'package:spotube/components/waypoint.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/album/album_card.dart'; +import 'package:spotube/modules/artist/artist_card.dart'; +import 'package:spotube/modules/playlist/playlist_card.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/provider/metadata_plugin/browse/section_items.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; + +const _dummyPlaybuttonCard = PlaybuttonCard( + imageUrl: 'https://placehold.co/150x150.png', + isLoading: false, + isPlaying: false, + title: "Playbutton", + description: "A really cool playbutton", + isOwner: false, +); + +@RoutePage() +class HomeBrowseSectionItemsPage extends HookConsumerWidget { + static const name = "home_browse_section_items"; + + final String sectionId; + final SpotubeBrowseSectionObject section; + const HomeBrowseSectionItemsPage({ + super.key, + @PathParam("sectionId") required this.sectionId, + required this.section, + }); + + @override + Widget build(BuildContext context, ref) { + final scale = context.theme.scaling; + + final sectionItems = + ref.watch(metadataPluginBrowseSectionItemsProvider(sectionId)); + final sectionItemsNotifier = + ref.watch(metadataPluginBrowseSectionItemsProvider(sectionId).notifier); + final items = sectionItems.asData?.value.items ?? []; + final controller = useScrollController(); + + final isLoading = sectionItems.isLoading || sectionItems.isLoadingNextPage; + final itemCount = items.length; + final hasMore = sectionItems.asData?.value.hasMore ?? false; + + return SafeArea( + bottom: false, + child: Skeletonizer( + enabled: sectionItems.isLoading, + child: Scaffold( + headers: [ + TitleBar( + title: Text(section.title), + ) + ], + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 8.0), + child: CustomScrollView( + controller: controller, + slivers: [ + SliverGrid.builder( + itemCount: isLoading ? 6 : itemCount + 1, + gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( + maxCrossAxisExtent: 150 * scale, + mainAxisExtent: 225 * scale, + crossAxisSpacing: 12 * scale, + mainAxisSpacing: 12 * scale, + ), + itemBuilder: (context, index) { + if (isLoading) { + return const Skeletonizer( + enabled: true, + child: _dummyPlaybuttonCard, + ); + } + + if (index == itemCount) { + if (!hasMore) return const SizedBox.shrink(); + return Waypoint( + controller: controller, + isGrid: true, + onTouchEdge: () async { + await sectionItemsNotifier.fetchMore(); + }, + child: const Skeletonizer( + enabled: true, + child: _dummyPlaybuttonCard, + ), + ); + } + + final item = items[index]; + return switch (item) { + SpotubeFullArtistObject() => ArtistCard(item), + SpotubeSimplePlaylistObject() => PlaylistCard(item), + SpotubeSimpleAlbumObject() => AlbumCard(item), + _ => throw Exception( + "Unsupported item type: ${item.runtimeType}", + ), + }; + }, + ), + const SliverToBoxAdapter( + child: SafeArea( + child: SizedBox(), + ), + ), + ], + ), + ), + ), + ), + ); + } +} diff --git a/lib/pages/lastfm_login/lastfm_login.dart b/lib/pages/lastfm_login/lastfm_login.dart index 8107e627..164b9b0d 100644 --- a/lib/pages/lastfm_login/lastfm_login.dart +++ b/lib/pages/lastfm_login/lastfm_login.dart @@ -1,140 +1,156 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:form_validator/form_validator.dart'; -import 'package:go_router/go_router.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/button/back_button.dart'; import 'package:spotube/components/dialogs/prompt_dialog.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/scrobbler/scrobbler.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class LastFMLoginPage extends HookConsumerWidget { static const name = "lastfm_login"; const LastFMLoginPage({super.key}); @override Widget build(BuildContext context, ref) { - final theme = Theme.of(context); - final router = GoRouter.of(context); final scrobblerNotifier = ref.read(scrobblerProvider.notifier); - final formKey = useMemoized(() => GlobalKey(), []); - final username = useTextEditingController(); - final password = useTextEditingController(); + final usernameKey = + useMemoized(() => const FormKey("username"), []); + final passwordKey = + useMemoized(() => const FormKey("password"), []); + final passwordVisible = useState(false); final isLoading = useState(false); return Scaffold( - appBar: const PageWindowTitleBar(leading: BackButton()), - body: Center( - child: ConstrainedBox( - constraints: const BoxConstraints(maxWidth: 400), - child: Card( - margin: const EdgeInsets.all(8.0), - child: Padding( - padding: const EdgeInsets.all(16.0).copyWith(top: 8), - child: Form( - key: formKey, - autovalidateMode: AutovalidateMode.onUserInteraction, - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - Container( - decoration: BoxDecoration( - borderRadius: BorderRadius.circular(30), - color: const Color.fromARGB(255, 186, 0, 0), + headers: const [ + SafeArea( + bottom: false, + child: TitleBar( + leading: [BackButton()], + ), + ), + ], + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Flexible( + child: Container( + constraints: const BoxConstraints(maxWidth: 400), + alignment: Alignment.center, + padding: const EdgeInsets.all(16), + child: Card( + padding: const EdgeInsets.all(16.0), + child: Form( + onSubmit: (context, values) async { + try { + isLoading.value = true; + await scrobblerNotifier.login( + values[usernameKey].trim(), + values[passwordKey], + ); + if (context.mounted) { + context.back(); + } + } catch (e) { + if (context.mounted) { + showPromptDialog( + context: context, + title: context.l10n.error("Authentication failed"), + message: e.toString(), + cancelText: null, + ); + } + } finally { + isLoading.value = false; + } + }, + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Container( + decoration: BoxDecoration( + borderRadius: BorderRadius.circular(30), + color: const Color.fromARGB(255, 186, 0, 0), + ), + padding: const EdgeInsets.all(12), + child: const Icon( + SpotubeIcons.lastFm, + color: Colors.white, + size: 60, + ), ), - padding: const EdgeInsets.all(12), - child: const Icon( - SpotubeIcons.lastFm, - color: Colors.white, - size: 60, - ), - ), - Text( - "last.fm", - style: theme.textTheme.titleLarge, - ), - const SizedBox(height: 10), - Text(context.l10n.login_with_your_lastfm), - const SizedBox(height: 10), - AutofillGroup( - child: Column( - children: [ - TextFormField( - autofillHints: const [ - AutofillHints.username, - AutofillHints.email, - ], - controller: username, - validator: ValidationBuilder().required().build(), - decoration: InputDecoration( - labelText: context.l10n.username, - ), - ), - const SizedBox(height: 10), - TextFormField( - autofillHints: const [ - AutofillHints.password, - ], - controller: password, - validator: ValidationBuilder().required().build(), - obscureText: !passwordVisible.value, - decoration: InputDecoration( - labelText: context.l10n.password, - suffixIcon: IconButton( - icon: Icon( - passwordVisible.value - ? SpotubeIcons.eye - : SpotubeIcons.noEye, - ), - onPressed: () => passwordVisible.value = - !passwordVisible.value, + const Text("last.fm").h3(), + Text(context.l10n.login_with_your_lastfm), + AutofillGroup( + child: Column( + spacing: 10, + children: [ + FormField( + label: Text(context.l10n.username), + key: usernameKey, + validator: const NotEmptyValidator( + message: "Username is required", + ), + child: TextField( + autofillHints: const [ + AutofillHints.username, + AutofillHints.email, + ], + placeholder: Text(context.l10n.username), ), ), - ), - ], + FormField( + key: passwordKey, + validator: const NotEmptyValidator( + message: "Password is required", + ), + label: Text(context.l10n.password), + child: TextField( + autofillHints: const [ + AutofillHints.password, + ], + obscureText: !passwordVisible.value, + placeholder: Text(context.l10n.password), + features: [ + InputFeature.trailing( + IconButton.ghost( + icon: Icon( + passwordVisible.value + ? SpotubeIcons.eye + : SpotubeIcons.noEye, + ), + onPressed: () => passwordVisible.value = + !passwordVisible.value, + ), + ), + ], + ), + ), + ], + ), ), - ), - const SizedBox(height: 10), - FilledButton( - onPressed: isLoading.value - ? null - : () async { - try { - isLoading.value = true; - if (formKey.currentState?.validate() != true) { - return; - } - await scrobblerNotifier.login( - username.text.trim(), - password.text, - ); - router.pop(); - } catch (e) { - if (context.mounted) { - showPromptDialog( - context: context, - title: context.l10n - .error("Authentication failed"), - message: e.toString(), - cancelText: null, - ); - } - } finally { - isLoading.value = false; - } - }, - child: Text(context.l10n.login), - ), - ], + FormErrorBuilder(builder: (context, errors, child) { + return Button.primary( + onPressed: () => context.submitForm(), + enabled: errors.isEmpty && !isLoading.value, + child: Text(context.l10n.login), + ); + }), + ], + ), ), ), ), ), - ), + ], ), ); } diff --git a/lib/pages/library/library.dart b/lib/pages/library/library.dart index a0bc1bb7..de438451 100644 --- a/lib/pages/library/library.dart +++ b/lib/pages/library/library.dart @@ -1,58 +1,92 @@ -import 'package:flutter/material.dart' hide Image; +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/material.dart' show Badge; +import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; - -import 'package:spotube/modules/library/user_local_tracks.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/collections/side_bar_tiles.dart'; +import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/modules/library/user_albums.dart'; -import 'package:spotube/modules/library/user_artists.dart'; -import 'package:spotube/modules/library/user_downloads.dart'; -import 'package:spotube/modules/library/user_playlists.dart'; -import 'package:spotube/components/themed_button_tab_bar.dart'; +import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/download_manager_provider.dart'; +@RoutePage() class LibraryPage extends HookConsumerWidget { - static const name = "library"; - const LibraryPage({super.key}); + @override Widget build(BuildContext context, ref) { - final downloadingCount = ref.watch(downloadManagerProvider).$downloadCount; + final downloadingCount = ref + .watch(downloadManagerProvider) + .where((e) => + e.status == DownloadStatus.downloading || + e.status == DownloadStatus.queued) + .length; + final router = context.watchRouter; + final sidebarLibraryTileList = useMemoized( + () => [ + ...getSidebarLibraryTileList(context.l10n), + SideBarTiles( + id: "downloads", + pathPrefix: "library/downloads", + title: context.l10n.downloads, + route: const UserDownloadsRoute(), + icon: SpotubeIcons.download, + ), + ], + [context.l10n], + ); + final index = sidebarLibraryTileList.indexWhere( + (e) => router.currentPath.startsWith(e.pathPrefix), + ); - return DefaultTabController( - length: 5, + return PopScope( + canPop: false, + onPopInvokedWithResult: (didPop, result) { + context.navigateTo(const HomeRoute()); + }, child: SafeArea( bottom: false, - child: Scaffold( - appBar: PageWindowTitleBar( - centerTitle: true, - leading: ThemedButtonsTabBar( - tabs: [ - Tab(text: " ${context.l10n.playlists} "), - Tab(text: " ${context.l10n.local_tab} "), - Tab( - child: Badge( - isLabelVisible: downloadingCount > 0, - label: Text(downloadingCount.toString()), - child: Text(" ${context.l10n.downloads} "), + child: LayoutBuilder(builder: (context, constraints) { + return Scaffold( + headers: [ + if (constraints.smAndDown) + TitleBar( + automaticallyImplyLeading: false, + child: SingleChildScrollView( + scrollDirection: Axis.horizontal, + child: TabList( + index: index, + onChanged: (index) { + context.navigateTo(sidebarLibraryTileList[index].route); + }, + children: [ + for (final tile in sidebarLibraryTileList) + TabItem( + child: Badge( + isLabelVisible: tile.id == 'downloads' && + downloadingCount > 0, + label: Text(downloadingCount.toString()), + child: Text(tile.title), + ), + ), + ], + ), ), + ) + else + const TitleBar( + automaticallyImplyLeading: false, + backgroundColor: Colors.transparent, + surfaceBlur: 0, + height: 32, ), - Tab(text: " ${context.l10n.artists} "), - Tab(text: " ${context.l10n.albums} "), - ], - ), - leadingWidth: double.infinity, - ), - body: const TabBarView( - children: [ - UserPlaylists(), - UserLocalTracks(), - UserDownloads(), - UserArtists(), - UserAlbums(), + const Gap(10), ], - ), - ), + child: const AutoRouter(), + ); + }), ), ); } diff --git a/lib/pages/library/local_folder.dart b/lib/pages/library/local_folder.dart deleted file mode 100644 index c2848b24..00000000 --- a/lib/pages/library/local_folder.dart +++ /dev/null @@ -1,375 +0,0 @@ -import 'dart:io'; -import 'dart:math'; - -import 'package:collection/collection.dart'; -import 'package:file_picker/file_picker.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:skeletonizer/skeletonizer.dart'; -import 'package:spotube/collections/fake.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/extensions/string.dart'; -import 'package:spotube/modules/library/local_folder/cache_export_dialog.dart'; -import 'package:spotube/modules/library/user_local_tracks.dart'; -import 'package:spotube/components/expandable_search/expandable_search.dart'; -import 'package:spotube/components/fallbacks/not_found.dart'; -import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/components/sort_tracks_dropdown.dart'; -import 'package:spotube/components/track_tile/track_tile.dart'; -import 'package:spotube/extensions/artist_simple.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/models/local_track.dart'; -import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:spotube/utils/service_utils.dart'; - -class LocalLibraryPage extends HookConsumerWidget { - static const name = "local_library_page"; - - final String location; - final bool isDownloads; - final bool isCache; - const LocalLibraryPage( - this.location, { - super.key, - this.isDownloads = false, - this.isCache = false, - }); - - Future playLocalTracks( - WidgetRef ref, - List tracks, { - LocalTrack? currentTrack, - }) async { - final playlist = ref.read(audioPlayerProvider); - final playback = ref.read(audioPlayerProvider.notifier); - currentTrack ??= tracks.first; - final isPlaylistPlaying = playlist.containsTracks(tracks); - if (!isPlaylistPlaying) { - var indexWhere = tracks.indexWhere((s) => s.id == currentTrack?.id); - await playback.load( - tracks, - initialIndex: indexWhere, - autoPlay: true, - ); - } else if (isPlaylistPlaying && - currentTrack.id != null && - currentTrack.id != playlist.activeTrack?.id) { - await playback.jumpToTrack(currentTrack); - } - } - - @override - Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); - - final sortBy = useState(SortBy.none); - final playlist = ref.watch(audioPlayerProvider); - final trackSnapshot = ref.watch(localTracksProvider); - final isPlaylistPlaying = playlist.containsTracks( - trackSnapshot.asData?.value.values.flattened.toList() ?? []); - - final searchController = useTextEditingController(); - useValueListenable(searchController); - final searchFocus = useFocusNode(); - final isFiltering = useState(false); - - final controller = useScrollController(); - - final directorySize = useMemoized(() async { - final dir = Directory(location); - final files = await dir.list(recursive: true).toList(); - - final filesLength = - await Future.wait(files.whereType().map((e) => e.length())); - - return (filesLength.sum.toInt() / pow(10, 9)).toStringAsFixed(2); - }, [location]); - - return SafeArea( - bottom: false, - child: Scaffold( - appBar: PageWindowTitleBar( - leading: const BackButton(), - centerTitle: true, - title: Column( - mainAxisSize: MainAxisSize.min, - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text( - isDownloads - ? context.l10n.downloads - : isCache - ? context.l10n.cache_folder.capitalize() - : location, - style: textTheme.titleLarge, - ), - FutureBuilder( - future: directorySize, - builder: (context, snapshot) { - return Text( - "${(snapshot.data ?? 0)} GB", - style: textTheme.labelSmall, - ); - }, - ) - ], - ), - backgroundColor: Colors.transparent, - actions: [ - if (isCache) ...[ - IconButton( - iconSize: 16, - icon: Column( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon(SpotubeIcons.delete), - Text( - context.l10n.clear_cache, - style: textTheme.labelSmall, - ) - ], - ), - onPressed: () async { - final accepted = await showDialog( - context: context, - builder: (context) => AlertDialog.adaptive( - title: Text(context.l10n.clear_cache_confirmation), - actions: [ - TextButton( - onPressed: () { - Navigator.of(context).pop(false); - }, - child: Text(context.l10n.decline), - ), - TextButton( - onPressed: () async { - Navigator.of(context).pop(true); - }, - child: Text(context.l10n.accept), - ), - ], - ), - ); - - if (accepted ?? false) return; - - final cacheDir = Directory( - await UserPreferencesNotifier.getMusicCacheDir(), - ); - - if (cacheDir.existsSync()) { - await cacheDir.delete(recursive: true); - } - }, - ), - IconButton( - iconSize: 16, - icon: Column( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon(SpotubeIcons.export), - Text( - context.l10n.export, - style: textTheme.labelSmall, - ) - ], - ), - onPressed: () async { - final exportPath = - await FilePicker.platform.getDirectoryPath(); - - if (exportPath == null) return; - final exportDirectory = Directory(exportPath); - - if (!exportDirectory.existsSync()) { - await exportDirectory.create(recursive: true); - } - - final cacheDir = Directory( - await UserPreferencesNotifier.getMusicCacheDir()); - - if (!context.mounted) return; - await showDialog( - context: context, - builder: (context) { - return LocalFolderCacheExportDialog( - cacheDir: cacheDir, - exportDir: exportDirectory, - ); - }, - ); - }, - ), - ] - ], - ), - body: Column( - children: [ - Padding( - padding: const EdgeInsets.all(8.0), - child: Row( - children: [ - const SizedBox(width: 5), - FilledButton( - onPressed: trackSnapshot.asData?.value != null - ? () async { - if (trackSnapshot.asData?.value.isNotEmpty == - true) { - if (!isPlaylistPlaying) { - await playLocalTracks( - ref, - trackSnapshot.asData!.value[location] ?? [], - ); - } - } - } - : null, - child: Row( - children: [ - Text(context.l10n.play), - Icon( - isPlaylistPlaying - ? SpotubeIcons.stop - : SpotubeIcons.play, - ) - ], - ), - ), - const Spacer(), - ExpandableSearchButton( - isFiltering: isFiltering.value, - onPressed: (value) => isFiltering.value = value, - searchFocus: searchFocus, - ), - const SizedBox(width: 10), - SortTracksDropdown( - value: sortBy.value, - onChanged: (value) { - sortBy.value = value; - }, - ), - const SizedBox(width: 5), - FilledButton( - child: const Icon(SpotubeIcons.refresh), - onPressed: () { - ref.invalidate(localTracksProvider); - }, - ) - ], - ), - ), - ExpandableSearchField( - searchController: searchController, - searchFocus: searchFocus, - isFiltering: isFiltering.value, - onChangeFiltering: (value) => isFiltering.value = value, - ), - trackSnapshot.when( - data: (tracks) { - final sortedTracks = useMemoized(() { - return ServiceUtils.sortTracks( - tracks[location] ?? [], sortBy.value); - }, [sortBy.value, tracks]); - - final filteredTracks = useMemoized(() { - if (searchController.text.isEmpty) { - return sortedTracks; - } - return sortedTracks - .map((e) => ( - weightedRatio( - "${e.name} - ${e.artists?.asString() ?? ""}", - searchController.text, - ), - e, - )) - .toList() - .sorted( - (a, b) => b.$1.compareTo(a.$1), - ) - .where((e) => e.$1 > 50) - .map((e) => e.$2) - .toList() - .toList(); - }, [searchController.text, sortedTracks]); - - if (!trackSnapshot.isLoading && filteredTracks.isEmpty) { - return const Expanded( - child: Row( - mainAxisAlignment: MainAxisAlignment.center, - children: [NotFound()], - ), - ); - } - - return Expanded( - child: RefreshIndicator( - onRefresh: () async { - ref.invalidate(localTracksProvider); - }, - child: InterScrollbar( - controller: controller, - child: Skeletonizer( - enabled: trackSnapshot.isLoading, - child: ListView.builder( - controller: controller, - physics: const AlwaysScrollableScrollPhysics(), - itemCount: trackSnapshot.isLoading - ? 5 - : filteredTracks.length, - itemBuilder: (context, index) { - if (trackSnapshot.isLoading) { - return TrackTile( - playlist: playlist, - track: FakeData.track, - index: index, - ); - } - - final track = filteredTracks[index]; - return TrackTile( - index: index, - playlist: playlist, - track: track, - userPlaylist: false, - onTap: () async { - await playLocalTracks( - ref, - sortedTracks, - currentTrack: track, - ); - }, - ); - }, - ), - ), - ), - ), - ); - }, - loading: () => Expanded( - child: Skeletonizer( - enabled: true, - child: ListView.builder( - itemCount: 5, - itemBuilder: (context, index) => TrackTile( - track: FakeData.track, - index: index, - playlist: playlist, - ), - ), - ), - ), - error: (error, stackTrace) => - Text(error.toString() + stackTrace.toString()), - ) - ], - )), - ); - } -} diff --git a/lib/pages/library/playlist_generate/playlist_generate.dart b/lib/pages/library/playlist_generate/playlist_generate.dart deleted file mode 100644 index b62013c5..00000000 --- a/lib/pages/library/playlist_generate/playlist_generate.dart +++ /dev/null @@ -1,659 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/spotify_markets.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/library/playlist_generate/multi_select_field.dart'; -import 'package:spotube/modules/library/playlist_generate/recommendation_attribute_dials.dart'; -import 'package:spotube/modules/library/playlist_generate/recommendation_attribute_fields.dart'; -import 'package:spotube/modules/library/playlist_generate/seeds_multi_autocomplete.dart'; -import 'package:spotube/modules/library/playlist_generate/simple_track_tile.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/extensions/constrains.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/models/spotify/recommendation_seeds.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; -import 'package:spotube/provider/spotify_provider.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -const RecommendationAttribute zeroValues = (min: 0, target: 0, max: 0); - -class PlaylistGeneratorPage extends HookConsumerWidget { - static const name = "playlist_generator"; - - const PlaylistGeneratorPage({super.key}); - - @override - Widget build(BuildContext context, ref) { - final spotify = ref.watch(spotifyProvider); - - final theme = Theme.of(context); - final textTheme = theme.textTheme; - final preferences = ref.watch(userPreferencesProvider); - - final genresCollection = ref.watch(categoryGenresProvider); - - final limit = useValueNotifier(10); - final market = useValueNotifier(preferences.market); - - final genres = useState>([]); - final artists = useState>([]); - final tracks = useState>([]); - - final enabled = - genres.value.length + artists.value.length + tracks.value.length < 5; - - final leftSeedCount = - 5 - genres.value.length - artists.value.length - tracks.value.length; - - // Dial (int 0-1) attributes - final min = useState(RecommendationSeeds()); - final max = useState(RecommendationSeeds()); - final target = useState(RecommendationSeeds()); - - final artistAutoComplete = SeedsMultiAutocomplete( - seeds: artists, - enabled: enabled, - inputDecoration: InputDecoration( - labelText: context.l10n.artists, - labelStyle: textTheme.titleMedium, - helperText: context.l10n.select_up_to_count_type( - leftSeedCount, - context.l10n.artists, - ), - ), - fetchSeeds: (textEditingValue) => spotify.search - .get( - textEditingValue.text, - types: [SearchType.artist], - ) - .first(6) - .then( - (v) => List.castFrom( - v.expand((e) => e.items ?? []).toList(), - ) - .where( - (element) => - artists.value.none((artist) => element.id == artist.id), - ) - .toList(), - ), - autocompleteOptionBuilder: (option, onSelected) => ListTile( - leading: CircleAvatar( - backgroundImage: UniversalImage.imageProvider( - option.images.asUrlString( - placeholder: ImagePlaceholder.artist, - ), - ), - ), - horizontalTitleGap: 20, - title: Text(option.name!), - subtitle: option.genres?.isNotEmpty != true - ? null - : Wrap( - spacing: 4, - runSpacing: 4, - children: option.genres!.mapIndexed( - (index, genre) { - return Chip( - label: Text(genre), - labelStyle: textTheme.bodySmall?.copyWith( - color: theme.colorScheme.secondary, - fontWeight: FontWeight.w600, - ), - side: BorderSide.none, - backgroundColor: theme.colorScheme.secondaryContainer, - ); - }, - ).toList(), - ), - onTap: () => onSelected(option), - ), - displayStringForOption: (option) => option.name!, - selectedSeedBuilder: (artist) => Chip( - avatar: CircleAvatar( - backgroundImage: UniversalImage.imageProvider( - artist.images.asUrlString( - placeholder: ImagePlaceholder.artist, - ), - ), - ), - label: Text(artist.name!), - onDeleted: () { - artists.value = [ - ...artists.value..removeWhere((element) => element.id == artist.id) - ]; - }, - ), - ); - - final tracksAutocomplete = SeedsMultiAutocomplete( - seeds: tracks, - enabled: enabled, - selectedItemDisplayType: SelectedItemDisplayType.list, - inputDecoration: InputDecoration( - labelText: context.l10n.tracks, - labelStyle: textTheme.titleMedium, - helperText: context.l10n.select_up_to_count_type( - leftSeedCount, - context.l10n.tracks, - ), - ), - fetchSeeds: (textEditingValue) => spotify.search - .get( - textEditingValue.text, - types: [SearchType.track], - ) - .first(6) - .then( - (v) => List.castFrom( - v.expand((e) => e.items ?? []).toList(), - ) - .where( - (element) => - tracks.value.none((track) => element.id == track.id), - ) - .toList(), - ), - autocompleteOptionBuilder: (option, onSelected) => ListTile( - leading: CircleAvatar( - backgroundImage: UniversalImage.imageProvider( - (option.album?.images).asUrlString( - placeholder: ImagePlaceholder.artist, - ), - ), - ), - horizontalTitleGap: 20, - title: Text(option.name!), - subtitle: Text( - option.artists?.map((e) => e.name).join(", ") ?? - option.album?.name ?? - "", - ), - onTap: () => onSelected(option), - ), - displayStringForOption: (option) => option.name!, - selectedSeedBuilder: (option) => SimpleTrackTile( - track: option, - onDelete: () { - tracks.value = [ - ...tracks.value..removeWhere((element) => element.id == option.id) - ]; - }, - ), - ); - - final genreSelector = MultiSelectField( - options: genresCollection.asData?.value ?? [], - selectedOptions: genres.value, - getValueForOption: (option) => option, - onSelected: (value) { - genres.value = value; - }, - dialogTitle: Text(context.l10n.select_genres), - label: Text(context.l10n.add_genres), - helperText: context.l10n.select_up_to_count_type( - leftSeedCount, - context.l10n.genre, - ), - enabled: enabled, - ); - final countrySelector = ValueListenableBuilder( - valueListenable: market, - builder: (context, value, _) { - return DropdownButtonFormField( - decoration: InputDecoration( - labelText: context.l10n.country, - labelStyle: textTheme.titleMedium, - ), - isExpanded: true, - items: spotifyMarkets - .map( - (country) => DropdownMenuItem( - value: country.$1, - child: Text(country.$2), - ), - ) - .toList(), - value: market.value, - onChanged: (value) { - market.value = value!; - }, - ); - }, - ); - - final controller = useScrollController(); - - return Scaffold( - appBar: PageWindowTitleBar( - leading: const BackButton(), - title: Text(context.l10n.generate_playlist), - centerTitle: true, - ), - body: Scrollbar( - controller: controller, - child: Center( - child: ConstrainedBox( - constraints: BoxConstraints(maxWidth: Breakpoints.lg), - child: SliderTheme( - data: const SliderThemeData( - overlayShape: RoundSliderOverlayShape(), - ), - child: SafeArea( - child: LayoutBuilder(builder: (context, constrains) { - return ScrollConfiguration( - behavior: ScrollConfiguration.of(context) - .copyWith(scrollbars: false), - child: ListView( - controller: controller, - padding: const EdgeInsets.all(16), - children: [ - ValueListenableBuilder( - valueListenable: limit, - builder: (context, value, child) { - return Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text( - context.l10n.number_of_tracks_generate, - style: textTheme.titleMedium, - ), - Row( - children: [ - Container( - width: 40, - height: 40, - alignment: Alignment.center, - decoration: BoxDecoration( - color: theme.colorScheme.primary, - shape: BoxShape.circle, - ), - child: Text( - value.round().toString(), - style: textTheme.bodyLarge?.copyWith( - color: theme - .colorScheme.primaryContainer, - ), - ), - ), - Expanded( - child: Slider( - value: value.toDouble(), - min: 10, - max: 100, - divisions: 9, - label: value.round().toString(), - onChanged: (value) { - limit.value = value.round(); - }, - ), - ) - ], - ) - ], - ); - }, - ), - const SizedBox(height: 16), - if (constrains.mdAndUp) - Row( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Expanded( - child: countrySelector, - ), - const SizedBox(width: 16), - Expanded( - child: genreSelector, - ), - ], - ) - else ...[ - countrySelector, - const SizedBox(height: 16), - genreSelector, - ], - const SizedBox(height: 16), - if (constrains.mdAndUp) - Row( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Expanded( - child: artistAutoComplete, - ), - const SizedBox(width: 16), - Expanded( - child: tracksAutocomplete, - ), - ], - ) - else ...[ - artistAutoComplete, - const SizedBox(height: 16), - tracksAutocomplete, - ], - const SizedBox(height: 16), - RecommendationAttributeDials( - title: Text(context.l10n.acousticness), - values: ( - target: target.value.acousticness?.toDouble() ?? 0, - min: min.value.acousticness?.toDouble() ?? 0, - max: max.value.acousticness?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - acousticness: value.target, - ); - min.value = min.value.copyWith( - acousticness: value.min, - ); - max.value = max.value.copyWith( - acousticness: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.danceability), - values: ( - target: target.value.danceability?.toDouble() ?? 0, - min: min.value.danceability?.toDouble() ?? 0, - max: max.value.danceability?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - danceability: value.target, - ); - min.value = min.value.copyWith( - danceability: value.min, - ); - max.value = max.value.copyWith( - danceability: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.energy), - values: ( - target: target.value.energy?.toDouble() ?? 0, - min: min.value.energy?.toDouble() ?? 0, - max: max.value.energy?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - energy: value.target, - ); - min.value = min.value.copyWith( - energy: value.min, - ); - max.value = max.value.copyWith( - energy: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.instrumentalness), - values: ( - target: - target.value.instrumentalness?.toDouble() ?? 0, - min: min.value.instrumentalness?.toDouble() ?? 0, - max: max.value.instrumentalness?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - instrumentalness: value.target, - ); - min.value = min.value.copyWith( - instrumentalness: value.min, - ); - max.value = max.value.copyWith( - instrumentalness: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.liveness), - values: ( - target: target.value.liveness?.toDouble() ?? 0, - min: min.value.liveness?.toDouble() ?? 0, - max: max.value.liveness?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - liveness: value.target, - ); - min.value = min.value.copyWith( - liveness: value.min, - ); - max.value = max.value.copyWith( - liveness: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.loudness), - values: ( - target: target.value.loudness?.toDouble() ?? 0, - min: min.value.loudness?.toDouble() ?? 0, - max: max.value.loudness?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - loudness: value.target, - ); - min.value = min.value.copyWith( - loudness: value.min, - ); - max.value = max.value.copyWith( - loudness: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.speechiness), - values: ( - target: target.value.speechiness?.toDouble() ?? 0, - min: min.value.speechiness?.toDouble() ?? 0, - max: max.value.speechiness?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - speechiness: value.target, - ); - min.value = min.value.copyWith( - speechiness: value.min, - ); - max.value = max.value.copyWith( - speechiness: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.valence), - values: ( - target: target.value.valence?.toDouble() ?? 0, - min: min.value.valence?.toDouble() ?? 0, - max: max.value.valence?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - valence: value.target, - ); - min.value = min.value.copyWith( - valence: value.min, - ); - max.value = max.value.copyWith( - valence: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.popularity), - base: 100, - values: ( - target: target.value.popularity?.toDouble() ?? 0, - min: min.value.popularity?.toDouble() ?? 0, - max: max.value.popularity?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - popularity: value.target, - ); - min.value = min.value.copyWith( - popularity: value.min, - ); - max.value = max.value.copyWith( - popularity: value.max, - ); - }, - ), - RecommendationAttributeDials( - title: Text(context.l10n.key), - base: 11, - values: ( - target: target.value.key?.toDouble() ?? 0, - min: min.value.key?.toDouble() ?? 0, - max: max.value.key?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - key: value.target, - ); - min.value = min.value.copyWith( - key: value.min, - ); - max.value = max.value.copyWith( - key: value.max, - ); - }, - ), - RecommendationAttributeFields( - title: Text(context.l10n.duration), - values: ( - max: (max.value.durationMs ?? 0) / 1000, - target: (target.value.durationMs ?? 0) / 1000, - min: (min.value.durationMs ?? 0) / 1000, - ), - onChanged: (value) { - target.value = target.value.copyWith( - durationMs: (value.target * 1000).toInt(), - ); - min.value = min.value.copyWith( - durationMs: (value.min * 1000).toInt(), - ); - max.value = max.value.copyWith( - durationMs: (value.max * 1000).toInt(), - ); - }, - presets: { - context.l10n.short: (min: 50, target: 90, max: 120), - context.l10n.medium: ( - min: 120, - target: 180, - max: 200 - ), - context.l10n.long: (min: 480, target: 560, max: 640) - }, - ), - RecommendationAttributeFields( - title: Text(context.l10n.tempo), - values: ( - max: max.value.tempo?.toDouble() ?? 0, - target: target.value.tempo?.toDouble() ?? 0, - min: min.value.tempo?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - tempo: value.target, - ); - min.value = min.value.copyWith( - tempo: value.min, - ); - max.value = max.value.copyWith( - tempo: value.max, - ); - }, - ), - RecommendationAttributeFields( - title: Text(context.l10n.mode), - values: ( - max: max.value.mode?.toDouble() ?? 0, - target: target.value.mode?.toDouble() ?? 0, - min: min.value.mode?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - mode: value.target, - ); - min.value = min.value.copyWith( - mode: value.min, - ); - max.value = max.value.copyWith( - mode: value.max, - ); - }, - ), - RecommendationAttributeFields( - title: Text(context.l10n.time_signature), - values: ( - max: max.value.timeSignature?.toDouble() ?? 0, - target: target.value.timeSignature?.toDouble() ?? 0, - min: min.value.timeSignature?.toDouble() ?? 0, - ), - onChanged: (value) { - target.value = target.value.copyWith( - timeSignature: value.target, - ); - min.value = min.value.copyWith( - timeSignature: value.min, - ); - max.value = max.value.copyWith( - timeSignature: value.max, - ); - }, - ), - const SizedBox(height: 20), - FilledButton.icon( - icon: const Icon(SpotubeIcons.magic), - label: Text(context.l10n.generate_playlist), - onPressed: artists.value.isEmpty && - tracks.value.isEmpty && - genres.value.isEmpty - ? null - : () { - final routeState = - GeneratePlaylistProviderInput( - seedArtists: artists.value - .map((a) => a.id!) - .toList(), - seedTracks: - tracks.value.map((t) => t.id!).toList(), - seedGenres: genres.value, - limit: limit.value, - max: max.value, - min: min.value, - target: target.value, - ); - GoRouter.of(context).push( - "/library/generate/result", - extra: routeState, - ); - }, - ), - ], - ), - ); - }), - ), - ), - ), - ), - ), - ); - } -} diff --git a/lib/pages/library/playlist_generate/playlist_generate_result.dart b/lib/pages/library/playlist_generate/playlist_generate_result.dart deleted file mode 100644 index 3bdc3b52..00000000 --- a/lib/pages/library/playlist_generate/playlist_generate_result.dart +++ /dev/null @@ -1,243 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/modules/library/playlist_generate/simple_track_tile.dart'; -import 'package:spotube/modules/playlist/playlist_create_dialog.dart'; -import 'package:spotube/components/dialogs/playlist_add_track_dialog.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/models/spotify/recommendation_seeds.dart'; -import 'package:spotube/pages/playlist/playlist.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class PlaylistGenerateResultPage extends HookConsumerWidget { - static const name = "playlist_generate_result"; - - final GeneratePlaylistProviderInput state; - - const PlaylistGenerateResultPage({ - super.key, - required this.state, - }); - - @override - Widget build(BuildContext context, ref) { - final router = GoRouter.of(context); - final scaffoldMessenger = ScaffoldMessenger.of(context); - final playlistNotifier = ref.watch(audioPlayerProvider.notifier); - - final generatedPlaylist = ref.watch(generatePlaylistProvider(state)); - - final selectedTracks = useState>( - generatedPlaylist.asData?.value.map((e) => e.id!).toList() ?? [], - ); - - useEffect(() { - if (generatedPlaylist.asData?.value != null) { - selectedTracks.value = - generatedPlaylist.asData!.value.map((e) => e.id!).toList(); - } - return null; - }, [generatedPlaylist.asData?.value]); - - final isAllTrackSelected = selectedTracks.value.length == - (generatedPlaylist.asData?.value.length ?? 0); - - return Scaffold( - appBar: const PageWindowTitleBar(leading: BackButton()), - body: generatedPlaylist.isLoading - ? Center( - child: Column( - mainAxisAlignment: MainAxisAlignment.center, - crossAxisAlignment: CrossAxisAlignment.center, - children: [ - const CircularProgressIndicator(), - Text(context.l10n.generating_playlist), - ], - ), - ) - : Padding( - padding: const EdgeInsets.all(8.0), - child: ListView( - children: [ - GridView( - gridDelegate: - const SliverGridDelegateWithFixedCrossAxisCount( - crossAxisCount: 2, - crossAxisSpacing: 8, - mainAxisSpacing: 8, - mainAxisExtent: 32, - ), - shrinkWrap: true, - children: [ - FilledButton.tonalIcon( - icon: const Icon(SpotubeIcons.play), - label: Text(context.l10n.play), - onPressed: selectedTracks.value.isEmpty - ? null - : () async { - await playlistNotifier.load( - generatedPlaylist.asData!.value - .where( - (e) => selectedTracks.value - .contains(e.id!), - ) - .toList(), - autoPlay: true, - ); - }, - ), - FilledButton.tonalIcon( - icon: const Icon(SpotubeIcons.queueAdd), - label: Text(context.l10n.add_to_queue), - onPressed: selectedTracks.value.isEmpty - ? null - : () async { - await playlistNotifier.addTracks( - generatedPlaylist.asData!.value.where( - (e) => selectedTracks.value.contains(e.id!), - ), - ); - if (context.mounted) { - scaffoldMessenger.showSnackBar( - SnackBar( - content: Text( - context.l10n.add_count_to_queue( - selectedTracks.value.length, - ), - ), - ), - ); - } - }, - ), - FilledButton.tonalIcon( - icon: const Icon(SpotubeIcons.addFilled), - label: Text(context.l10n.create_a_playlist), - onPressed: selectedTracks.value.isEmpty - ? null - : () async { - final playlist = await showDialog( - context: context, - builder: (context) => PlaylistCreateDialog( - trackIds: selectedTracks.value, - ), - ); - - if (playlist != null) { - router.goNamed( - PlaylistPage.name, - pathParameters: { - "id": playlist.id!, - }, - extra: playlist, - ); - } - }, - ), - FilledButton.tonalIcon( - icon: const Icon(SpotubeIcons.playlistAdd), - label: Text(context.l10n.add_to_playlist), - onPressed: selectedTracks.value.isEmpty - ? null - : () async { - final hasAdded = await showDialog( - context: context, - builder: (context) => PlaylistAddTrackDialog( - openFromPlaylist: null, - tracks: selectedTracks.value - .map( - (e) => generatedPlaylist.asData!.value - .firstWhere( - (element) => element.id == e, - ), - ) - .toList(), - ), - ); - - if (context.mounted && hasAdded == true) { - scaffoldMessenger.showSnackBar( - SnackBar( - content: Text( - context.l10n.add_count_to_playlist( - selectedTracks.value.length, - ), - ), - ), - ); - } - }, - ) - ], - ), - const SizedBox(height: 16), - if (generatedPlaylist.asData?.value != null) - Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text( - context.l10n.selected_count_tracks( - selectedTracks.value.length, - ), - ), - ElevatedButton.icon( - onPressed: () { - if (isAllTrackSelected) { - selectedTracks.value = []; - } else { - selectedTracks.value = generatedPlaylist - .asData?.value - .map((e) => e.id!) - .toList() ?? - []; - } - }, - icon: const Icon(SpotubeIcons.selectionCheck), - label: Text( - isAllTrackSelected - ? context.l10n.deselect_all - : context.l10n.select_all, - ), - ), - ], - ), - const SizedBox(height: 8), - Card( - margin: const EdgeInsets.all(0), - child: SafeArea( - child: Column( - mainAxisSize: MainAxisSize.min, - children: [ - for (final track - in generatedPlaylist.asData?.value ?? []) - CheckboxListTile( - value: selectedTracks.value.contains(track.id), - onChanged: (value) { - if (value == true) { - selectedTracks.value.add(track.id!); - } else { - selectedTracks.value.remove(track.id); - } - selectedTracks.value = - selectedTracks.value.toList(); - }, - controlAffinity: ListTileControlAffinity.leading, - contentPadding: EdgeInsets.zero, - dense: true, - title: SimpleTrackTile(track: track), - ) - ], - ), - ), - ), - ], - ), - ), - ); - } -} diff --git a/lib/pages/library/user_albums.dart b/lib/pages/library/user_albums.dart new file mode 100644 index 00000000..2d989138 --- /dev/null +++ b/lib/pages/library/user_albums.dart @@ -0,0 +1,155 @@ +import 'package:flutter/material.dart' as material; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide Image; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:collection/collection.dart'; +import 'package:fuzzywuzzy/fuzzywuzzy.dart'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; + +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/fallbacks/no_default_metadata_plugin.dart'; +import 'package:spotube/components/playbutton_view/playbutton_view.dart'; +import 'package:spotube/modules/album/album_card.dart'; +import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; +import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/library/albums.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +@RoutePage() +class UserAlbumsPage extends HookConsumerWidget { + static const name = 'user_albums'; + const UserAlbumsPage({super.key}); + + @override + Widget build(BuildContext context, ref) { + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); + final albumsQuery = ref.watch(metadataPluginSavedAlbumsProvider); + final albumsQueryNotifier = + ref.watch(metadataPluginSavedAlbumsProvider.notifier); + + final controller = useScrollController(); + + final searchText = useState(''); + + final albums = useMemoized(() { + if (searchText.value.isEmpty) { + return albumsQuery.asData?.value.items ?? []; + } + return albumsQuery.asData?.value.items + .map((e) => ( + weightedRatio(e.name, searchText.value), + e, + )) + .sorted((a, b) => b.$1.compareTo(a.$1)) + .where((e) => e.$1 > 50) + .map((e) => e.$2) + .toList() ?? + []; + }, [albumsQuery.asData?.value, searchText.value]); + + if (albumsQuery.error + case MetadataPluginException( + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + message: _, + )) { + return const Center(child: NoDefaultMetadataPlugin()); + } + + if (authenticated.asData?.value != true) { + return const AnonymousFallback(); + } + + if (albumsQuery.hasError) { + return ErrorBox( + error: albumsQuery.error!, + onRetry: () { + ref.invalidate(metadataPluginSavedAlbumsProvider); + }, + ); + } + + return SafeArea( + bottom: false, + child: Scaffold( + child: material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginSavedAlbumsProvider); + }, + child: InterScrollbar( + controller: controller, + child: CustomScrollView( + controller: controller, + slivers: [ + SliverAppBar( + automaticallyImplyLeading: false, + backgroundColor: Theme.of(context).colorScheme.background, + floating: true, + flexibleSpace: Padding( + padding: const EdgeInsets.symmetric(horizontal: 8.0), + child: SizedBox( + height: 48, + child: TextField( + onChanged: (value) => searchText.value = value, + features: const [ + InputFeature.leading(Icon(SpotubeIcons.filter)) + ], + placeholder: Text(context.l10n.filter_albums), + ), + ), + ), + ), + const SliverGap(10), + if (albums.isEmpty && + !albumsQuery.isLoading && + searchText.value.isEmpty) + SliverPadding( + padding: const EdgeInsets.symmetric(horizontal: 8), + sliver: SliverToBoxAdapter( + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.followMeDrone, + color: Theme.of(context).colorScheme.primary, + ), + Text( + context.l10n.no_favorite_albums_yet, + textAlign: TextAlign.center, + ).muted().small() + ], + ), + ), + ) + else + SliverPadding( + padding: const EdgeInsets.symmetric(horizontal: 8), + sliver: PlaybuttonView( + controller: controller, + itemCount: albums.length, + hasMore: albumsQuery.asData?.value.hasMore == true, + isLoading: albumsQuery.isLoading, + onRequestMore: albumsQueryNotifier.fetchMore, + gridItemBuilder: (context, index) => AlbumCard( + albums[index], + ), + listItemBuilder: (context, index) => + AlbumCard.tile(albums[index]), + ), + ), + const SliverSafeArea(sliver: SliverGap(10)), + ], + ), + ), + ), + ), + ); + } +} diff --git a/lib/pages/library/user_artists.dart b/lib/pages/library/user_artists.dart new file mode 100644 index 00000000..750cb50b --- /dev/null +++ b/lib/pages/library/user_artists.dart @@ -0,0 +1,197 @@ +import 'package:flutter/material.dart' as material; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:collection/collection.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:fuzzywuzzy/fuzzywuzzy.dart'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/fake.dart'; + +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/fallbacks/no_default_metadata_plugin.dart'; +import 'package:spotube/modules/artist/artist_card.dart'; +import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; +import 'package:spotube/components/waypoint.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/library/artists.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +@RoutePage() +class UserArtistsPage extends HookConsumerWidget { + static const name = 'user_artists'; + const UserArtistsPage({super.key}); + + @override + Widget build(BuildContext context, ref) { + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); + + final artistQuery = ref.watch(metadataPluginSavedArtistsProvider); + final artistQueryNotifier = + ref.watch(metadataPluginSavedArtistsProvider.notifier); + + final searchText = useState(''); + + final filteredArtists = useMemoized(() { + final artists = artistQuery.asData?.value.items ?? []; + + if (searchText.value.isEmpty) { + return artists.toList(); + } + return artists + .map((e) => ( + weightedRatio(e.name, searchText.value), + e, + )) + .sorted((a, b) => b.$1.compareTo(a.$1)) + .where((e) => e.$1 > 50) + .map((e) => e.$2) + .toList(); + }, [artistQuery.asData?.value.items, searchText.value]); + + final controller = useScrollController(); + + if (artistQuery.error + case MetadataPluginException( + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + message: _, + )) { + return const Center(child: NoDefaultMetadataPlugin()); + } + + if (authenticated.asData?.value != true) { + return const AnonymousFallback(); + } + + if (artistQuery.hasError) { + return ErrorBox( + error: artistQuery.error!, + onRetry: () { + ref.invalidate(metadataPluginSavedArtistsProvider); + }, + ); + } + + return SafeArea( + bottom: false, + child: Scaffold( + child: material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginSavedArtistsProvider); + }, + child: InterScrollbar( + controller: controller, + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 8.0), + child: CustomScrollView( + controller: controller, + slivers: [ + SliverAppBar( + automaticallyImplyLeading: false, + backgroundColor: Theme.of(context).colorScheme.background, + floating: true, + flexibleSpace: SizedBox( + height: 48, + child: TextField( + onChanged: (value) => searchText.value = value, + features: const [ + InputFeature.leading(Icon(SpotubeIcons.filter)), + ], + placeholder: Text(context.l10n.filter_artist), + ), + ), + ), + const SliverGap(10), + if (filteredArtists.isNotEmpty || artistQuery.isLoading) + SliverLayoutBuilder(builder: (context, constrains) { + return SliverGrid.builder( + itemCount: filteredArtists.length + 1, + gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( + maxCrossAxisExtent: 200, + mainAxisExtent: constrains.smAndDown ? 225 : 250, + crossAxisSpacing: 8, + mainAxisSpacing: 8, + ), + itemBuilder: (context, index) { + if (filteredArtists.isNotEmpty && + index == filteredArtists.length) { + if (artistQuery.asData?.value.hasMore != true) { + return const SizedBox.shrink(); + } + + return Waypoint( + controller: controller, + isGrid: true, + onTouchEdge: artistQueryNotifier.fetchMore, + child: Skeletonizer( + enabled: true, + child: ArtistCard(FakeData.artist), + ), + ); + } + + return Skeletonizer( + enabled: artistQuery.isLoading, + child: ArtistCard( + filteredArtists.elementAtOrNull(index) ?? + FakeData.artist, + ), + ); + }, + ); + }) + else if (filteredArtists.isEmpty && + searchText.value.isEmpty && + !artistQuery.isLoading) + SliverToBoxAdapter( + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.followMeDrone, + color: Theme.of(context).colorScheme.primary, + ), + Text( + context.l10n.not_following_artists, + textAlign: TextAlign.center, + ).muted().small() + ], + ), + ) + else + SliverToBoxAdapter( + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.taken, + color: Theme.of(context).colorScheme.primary, + ), + Text( + context.l10n.nothing_found, + textAlign: TextAlign.center, + ).muted().small() + ], + ), + ), + const SliverSafeArea(sliver: SliverGap(10)), + ], + ), + ), + ), + ), + ), + ); + } +} diff --git a/lib/modules/library/user_downloads.dart b/lib/pages/library/user_downloads.dart similarity index 55% rename from lib/modules/library/user_downloads.dart rename to lib/pages/library/user_downloads.dart index 7fe9800c..f6a130bb 100644 --- a/lib/modules/library/user_downloads.dart +++ b/lib/pages/library/user_downloads.dart @@ -1,22 +1,21 @@ import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/modules/library/user_downloads/download_item.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/download_manager_provider.dart'; +import 'package:auto_route/auto_route.dart'; -class UserDownloads extends HookConsumerWidget { - const UserDownloads({super.key}); +@RoutePage() +class UserDownloadsPage extends HookConsumerWidget { + static const name = 'user_downloads'; + const UserDownloadsPage({super.key}); @override Widget build(BuildContext context, ref) { - final downloadManager = ref.watch(downloadManagerProvider); - - final history = [ - ...downloadManager.$history, - ...downloadManager.$backHistory, - ]; + final downloadQueue = ref.watch(downloadManagerProvider); + final downloadManagerNotifier = ref.watch(downloadManagerProvider.notifier); return Column( crossAxisAlignment: CrossAxisAlignment.start, @@ -28,21 +27,15 @@ class UserDownloads extends HookConsumerWidget { children: [ Expanded( child: AutoSizeText( - context.l10n - .currently_downloading(downloadManager.$downloadCount), + context.l10n.currently_downloading(downloadQueue.length), maxLines: 1, - style: Theme.of(context).textTheme.titleMedium, - ), + ).semiBold(), ), const SizedBox(width: 10), - FilledButton( - style: FilledButton.styleFrom( - backgroundColor: Colors.red[50], - foregroundColor: Colors.red[400], - ), - onPressed: downloadManager.$downloadCount == 0 + Button.destructive( + onPressed: downloadQueue.isEmpty ? null - : downloadManager.cancelAll, + : downloadManagerNotifier.clearAll, child: Text(context.l10n.cancel_all), ), ], @@ -51,9 +44,12 @@ class UserDownloads extends HookConsumerWidget { Expanded( child: SafeArea( child: ListView.builder( - itemCount: history.length, + itemCount: downloadQueue.length, + padding: const EdgeInsets.only(bottom: 200), itemBuilder: (context, index) { - return DownloadItem(track: history[index]); + return DownloadItem( + task: downloadQueue.elementAt(index), + ); }, ), ), diff --git a/lib/pages/library/user_local_tracks/local_folder.dart b/lib/pages/library/user_local_tracks/local_folder.dart new file mode 100644 index 00000000..523097e1 --- /dev/null +++ b/lib/pages/library/user_local_tracks/local_folder.dart @@ -0,0 +1,504 @@ +import 'dart:io'; +import 'dart:math'; + +import 'package:flutter/material.dart' as material; +import 'package:collection/collection.dart'; +import 'package:file_picker/file_picker.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:fuzzywuzzy/fuzzywuzzy.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/fake.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/button/back_button.dart'; +import 'package:spotube/components/track_presentation/presentation_actions.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/string.dart'; +import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/library/local_folder/cache_export_dialog.dart'; +import 'package:spotube/pages/library/user_local_tracks/user_local_tracks.dart'; +import 'package:spotube/components/expandable_search/expandable_search.dart'; +import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/components/track_presentation/sort_tracks_dropdown.dart'; +import 'package:spotube/components/track_tile/track_tile.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +import 'package:spotube/utils/service_utils.dart'; +import 'package:auto_route/auto_route.dart'; + +@RoutePage() +class LocalLibraryPage extends HookConsumerWidget { + static const name = "local_library_page"; + + final String location; + final bool isDownloads; + final bool isCache; + const LocalLibraryPage( + this.location, { + super.key, + this.isDownloads = false, + this.isCache = false, + }); + + Future playLocalTracks( + WidgetRef ref, + List tracks, { + SpotubeLocalTrackObject? currentTrack, + }) async { + final playlist = ref.read(audioPlayerProvider); + final playback = ref.read(audioPlayerProvider.notifier); + currentTrack ??= tracks.first; + final isPlaylistPlaying = playlist.containsTracks(tracks); + if (!isPlaylistPlaying) { + var indexWhere = tracks.indexWhere((s) => s.id == currentTrack?.id); + await playback.load( + tracks, + initialIndex: indexWhere, + autoPlay: true, + ); + } else if (isPlaylistPlaying && + currentTrack.id != playlist.activeTrack?.id) { + await playback.jumpToTrack(currentTrack); + } + } + + Future shufflePlayLocalTracks( + WidgetRef ref, + List tracks, + ) async { + final playlist = ref.read(audioPlayerProvider); + final playback = ref.read(audioPlayerProvider.notifier); + final isPlaylistPlaying = playlist.containsTracks(tracks); + final shuffledTracks = tracks.shuffled(); + if (isPlaylistPlaying) return; + + await playback.load( + shuffledTracks, + initialIndex: 0, + autoPlay: true, + ); + } + + Future addToQueueLocalTracks( + BuildContext context, + WidgetRef ref, + List tracks, + ) async { + final playlist = ref.read(audioPlayerProvider); + final playback = ref.read(audioPlayerProvider.notifier); + final isPlaylistPlaying = playlist.containsTracks(tracks); + if (isPlaylistPlaying) return; + await playback.addTracks(tracks); + if (!context.mounted) return; + showToastForAction(context, "add-to-queue", tracks.length); + } + + @override + Widget build(BuildContext context, ref) { + final scale = context.theme.scaling; + + final sortBy = useState(SortBy.none); + final playlist = ref.watch(audioPlayerProvider); + final trackSnapshot = ref.watch(localTracksProvider); + final isPlaylistPlaying = useMemoized( + () => playlist.containsTracks( + trackSnapshot.asData?.value[location] ?? [], + ), + [playlist, trackSnapshot, location], + ); + + final searchController = useShadcnTextEditingController(); + useValueListenable(searchController); + final searchFocus = useFocusNode(); + final isFiltering = useState(false); + + final controller = useScrollController(); + + final directorySize = useMemoized(() async { + final dir = Directory(location); + final files = await dir.list(recursive: true).toList(); + + final filesLength = + await Future.wait(files.whereType().map((e) => e.length())); + + return (filesLength.sum.toInt() / pow(10, 9)).toStringAsFixed(2); + }, [location]); + + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + padding: const EdgeInsets.symmetric( + horizontal: 10, + vertical: 0, + ), + surfaceBlur: 0, + leading: const [BackButton()], + title: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + isDownloads + ? context.l10n.downloads + : isCache + ? context.l10n.cache_folder.capitalize() + : location, + ), + FutureBuilder( + future: directorySize, + builder: (context, snapshot) { + return Text( + "${(snapshot.data ?? 0)} GB", + ).xSmall().muted(); + }, + ) + ], + ), + backgroundColor: Colors.transparent, + trailingGap: 10, + trailing: [ + if (isCache) ...[ + IconButton.outline( + size: ButtonSize.small, + icon: Column( + mainAxisSize: MainAxisSize.min, + children: [ + const Icon(SpotubeIcons.delete), + Text(context.l10n.clear_cache) + ], + ).xSmall().iconSmall(), + onPressed: () async { + final accepted = await showDialog( + context: context, + builder: (context) => AlertDialog( + title: Text(context.l10n.clear_cache_confirmation), + actions: [ + Button.outline( + onPressed: () { + Navigator.of(context).pop(false); + }, + child: Text(context.l10n.decline), + ), + Button.destructive( + onPressed: () async { + Navigator.of(context).pop(true); + }, + child: Text(context.l10n.accept), + ), + ], + ), + ); + + if (accepted != true) return; + + final cacheDir = Directory( + await UserPreferencesNotifier.getMusicCacheDir(), + ); + + if (cacheDir.existsSync()) { + await cacheDir.delete(recursive: true); + } + + ref.invalidate(localTracksProvider); + }, + ), + IconButton.outline( + size: ButtonSize.small, + icon: Column( + mainAxisSize: MainAxisSize.min, + children: [ + const Icon(SpotubeIcons.export), + Text( + context.l10n.export, + ) + ], + ).xSmall().iconSmall(), + onPressed: () async { + final exportPath = + await FilePicker.platform.getDirectoryPath(); + + if (exportPath == null) return; + final exportDirectory = Directory(exportPath); + + if (!exportDirectory.existsSync()) { + await exportDirectory.create(recursive: true); + } + + final cacheDir = Directory( + await UserPreferencesNotifier.getMusicCacheDir()); + + if (!context.mounted) return; + await showDialog( + context: context, + builder: (context) { + return LocalFolderCacheExportDialog( + cacheDir: cacheDir, + exportDir: exportDirectory, + ); + }, + ); + }, + ), + ] + ], + ), + ], + child: LayoutBuilder( + builder: (context, constraints) => Column( + children: [ + Padding( + padding: const EdgeInsets.all(8.0), + child: Row( + children: [ + const Gap(5), + Tooltip( + tooltip: + TooltipContainer(child: Text(context.l10n.play)).call, + child: IconButton.primary( + onPressed: trackSnapshot.asData?.value != null + ? () async { + if (trackSnapshot.asData?.value.isNotEmpty == + true) { + if (!isPlaylistPlaying) { + await playLocalTracks( + ref, + trackSnapshot.asData!.value[location] ?? + [], + ); + } + } + } + : null, + icon: Icon( + isPlaylistPlaying + ? SpotubeIcons.stop + : SpotubeIcons.play, + ), + ), + ), + const Gap(5), + Tooltip( + tooltip: + TooltipContainer(child: Text(context.l10n.shuffle)) + .call, + child: IconButton.outline( + onPressed: trackSnapshot.asData?.value != null + ? () async { + if (trackSnapshot.asData?.value.isNotEmpty == + true) { + if (!isPlaylistPlaying) { + await shufflePlayLocalTracks( + ref, + trackSnapshot.asData!.value[location] ?? + [], + ); + } + } + } + : null, + enabled: !isPlaylistPlaying, + icon: const Icon(SpotubeIcons.shuffle), + ), + ), + const Gap(5), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.add_to_queue)) + .call, + child: IconButton.outline( + onPressed: trackSnapshot.asData?.value != null + ? () async { + if (trackSnapshot.asData?.value.isNotEmpty == + true) { + if (!isPlaylistPlaying) { + await addToQueueLocalTracks( + context, + ref, + trackSnapshot.asData!.value[location] ?? + [], + ); + } + } + } + : null, + enabled: !isPlaylistPlaying, + icon: const Icon(SpotubeIcons.queueAdd), + ), + ), + const Spacer(), + if (constraints.smAndDown) + ExpandableSearchButton( + isFiltering: isFiltering.value, + onPressed: (value) => isFiltering.value = value, + searchFocus: searchFocus, + ) + else + ConstrainedBox( + constraints: BoxConstraints( + maxWidth: 300 * scale, + maxHeight: 38 * scale, + ), + child: ExpandableSearchField( + isFiltering: true, + onChangeFiltering: (value) {}, + searchController: searchController, + searchFocus: searchFocus, + ), + ), + const Gap(5), + SortTracksDropdown( + value: sortBy.value, + onChanged: (value) { + sortBy.value = value; + }, + ), + const Gap(5), + IconButton.outline( + icon: const Icon(SpotubeIcons.refresh), + onPressed: () { + ref.invalidate(localTracksProvider); + }, + ) + ], + ), + ), + ExpandableSearchField( + searchController: searchController, + searchFocus: searchFocus, + isFiltering: isFiltering.value, + onChangeFiltering: (value) => isFiltering.value = value, + ), + HookBuilder(builder: (context) { + return trackSnapshot.when( + data: (tracks) { + final sortedTracks = useMemoized(() { + return ServiceUtils.sortTracks( + tracks[location] ?? [], + sortBy.value); + }, [sortBy.value, tracks]); + + final filteredTracks = useMemoized(() { + if (searchController.text.isEmpty) { + return sortedTracks; + } + return sortedTracks + .map((e) => ( + weightedRatio( + "${e.name} - ${e.artists.asString()}", + searchController.text, + ), + e, + )) + .toList() + .sorted( + (a, b) => b.$1.compareTo(a.$1), + ) + .where((e) => e.$1 > 50) + .map((e) => e.$2) + .toList() + .toList(); + }, [searchController.text, sortedTracks]); + + if (!trackSnapshot.isLoading && filteredTracks.isEmpty) { + return Expanded( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Undraw( + illustration: UndrawIllustration.empty, + height: 200 * scale, + color: context.theme.colorScheme.primary, + ), + const Gap(10), + Text( + context.l10n.nothing_found, + textAlign: TextAlign.center, + ).muted().small() + ], + ), + ); + } + + return Expanded( + child: material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(localTracksProvider); + }, + child: InterScrollbar( + controller: controller, + child: Skeletonizer( + enabled: trackSnapshot.isLoading, + child: CustomScrollView( + controller: controller, + physics: const AlwaysScrollableScrollPhysics(), + slivers: [ + SliverList.builder( + itemCount: trackSnapshot.isLoading + ? 5 + : filteredTracks.length, + itemBuilder: (context, index) { + if (trackSnapshot.isLoading) { + return TrackTile( + playlist: playlist, + track: FakeData.track, + index: index, + ); + } + + final track = filteredTracks[index]; + return TrackTile( + index: index, + playlist: playlist, + track: track, + userPlaylist: false, + onTap: () async { + await playLocalTracks( + ref, + sortedTracks, + currentTrack: track, + ); + }, + ); + }, + ), + const SliverGap(200), + ], + ), + ), + ), + ), + ); + }, + loading: () => Expanded( + child: Skeletonizer( + enabled: true, + child: ListView.builder( + itemCount: 5, + itemBuilder: (context, index) => TrackTile( + track: FakeData.track, + index: index, + playlist: playlist, + ), + ), + ), + ), + error: (error, stackTrace) => + Text(error.toString() + stackTrace.toString()), + ); + }), + ], + ), + ), + ), + ); + } +} diff --git a/lib/modules/library/user_local_tracks.dart b/lib/pages/library/user_local_tracks/user_local_tracks.dart similarity index 50% rename from lib/modules/library/user_local_tracks.dart rename to lib/pages/library/user_local_tracks/user_local_tracks.dart index 23fb3be0..5f7502e6 100644 --- a/lib/modules/library/user_local_tracks.dart +++ b/lib/pages/library/user_local_tracks/user_local_tracks.dart @@ -1,9 +1,10 @@ +import 'package:auto_route/auto_route.dart'; import 'package:file_picker/file_picker.dart'; import 'package:file_selector/file_selector.dart'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/library/local_folder/local_folder_item.dart'; @@ -12,7 +13,6 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/utils/platform.dart'; -// ignore: depend_on_referenced_packages enum SortBy { none, @@ -25,8 +25,10 @@ enum SortBy { album, } -class UserLocalTracks extends HookConsumerWidget { - const UserLocalTracks({super.key}); +@RoutePage() +class UserLocalLibraryPage extends HookConsumerWidget { + static const name = 'user_local_library'; + const UserLocalLibraryPage({super.key}); @override Widget build(BuildContext context, ref) { @@ -58,49 +60,48 @@ class UserLocalTracks extends HookConsumerWidget { // For now, this gets all of them. ref.watch(localTracksProvider); - return LayoutBuilder(builder: (context, constrains) { - return Padding( - padding: const EdgeInsets.symmetric(horizontal: 12.0), - child: Column( - children: [ - Align( - alignment: Alignment.centerRight, - child: TextButton.icon( - icon: const Icon(SpotubeIcons.folderAdd), - label: Text(context.l10n.add_library_location), - onPressed: addLocalLibraryLocation, + final locations = [ + preferences.downloadLocation, + if (cacheDir.hasData) cacheDir.data!, + ...preferences.localLibraryLocation, + ]; + + return LayoutBuilder( + builder: (context, constrains) => Padding( + padding: const EdgeInsets.symmetric(horizontal: 12.0), + child: Column( + children: [ + Align( + alignment: Alignment.centerRight, + child: Button.secondary( + leading: const Icon(SpotubeIcons.folderAdd), + onPressed: addLocalLibraryLocation, + child: Text(context.l10n.add_library_location), + ), + ), + const Gap(8), + Expanded( + child: GridView.builder( + gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( + maxCrossAxisExtent: 200, + mainAxisExtent: constrains.isXs + ? 230 * context.theme.scaling + : constrains.mdAndDown + ? 280 * context.theme.scaling + : 250 * context.theme.scaling, + crossAxisSpacing: 10, + mainAxisSpacing: 10, + ), + itemCount: locations.length, + itemBuilder: (context, index) { + return LocalFolderItem( + folder: locations[index], + ); + }, + ), + ), + ], ), - ), - const Gap(8), - Expanded( - child: GridView.builder( - gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( - maxCrossAxisExtent: 200, - mainAxisExtent: constrains.isXs - ? 210 - : constrains.mdAndDown - ? 280 - : 250, - crossAxisSpacing: 10, - mainAxisSpacing: 10, - ), - itemCount: preferences.localLibraryLocation.length + - 1 + - (cacheDir.hasData ? 1 : 0), - itemBuilder: (context, index) { - return LocalFolderItem( - folder: index == 0 - ? preferences.downloadLocation - : index == 1 && cacheDir.hasData - ? cacheDir.data! - : preferences.localLibraryLocation[index - 1], - ); - }, - ), - ), - ], - ), - ); - }); + )); } } diff --git a/lib/pages/library/user_playlists.dart b/lib/pages/library/user_playlists.dart new file mode 100644 index 00000000..740bc947 --- /dev/null +++ b/lib/pages/library/user_playlists.dart @@ -0,0 +1,172 @@ +import 'package:flutter/material.dart' as material; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:fuzzywuzzy/fuzzywuzzy.dart'; +import 'package:collection/collection.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide Image; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/assets.gen.dart'; + +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/fallbacks/no_default_metadata_plugin.dart'; +import 'package:spotube/components/playbutton_view/playbutton_view.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/playlist/playlist_create_dialog.dart'; +import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; +import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; +import 'package:spotube/modules/playlist/playlist_card.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +@RoutePage() +class UserPlaylistsPage extends HookConsumerWidget { + static const name = 'user_playlists'; + const UserPlaylistsPage({super.key}); + + @override + Widget build(BuildContext context, ref) { + final searchText = useState(''); + + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); + + final me = ref.watch(metadataPluginUserProvider); + final playlistsQuery = ref.watch(metadataPluginSavedPlaylistsProvider); + final playlistsQueryNotifier = + ref.watch(metadataPluginSavedPlaylistsProvider.notifier); + + final likedTracksPlaylist = useMemoized( + () => me.asData?.value == null + ? null + : SpotubeSimplePlaylistObject( + id: "user-liked-tracks", + name: context.l10n.liked_tracks, + description: context.l10n.liked_tracks_description, + externalUri: "", + owner: me.asData!.value!, + images: [ + SpotubeImageObject( + url: Assets.images.likedTracks.path, + width: 300, + height: 300, + ) + ]), + [context.l10n, me.asData?.value], + ); + + final playlists = useMemoized( + () { + if (searchText.value.isEmpty) { + return [ + if (likedTracksPlaylist != null) likedTracksPlaylist, + ...?playlistsQuery.asData?.value.items, + ]; + } + return [ + if (likedTracksPlaylist != null) likedTracksPlaylist, + ...?playlistsQuery.asData?.value.items, + ] + .map((e) => (weightedRatio(e.name, searchText.value), e)) + .sorted((a, b) => b.$1.compareTo(a.$1)) + .where((e) => e.$1 > 50) + .map((e) => e.$2) + .toList(); + }, + [playlistsQuery, searchText.value], + ); + + final controller = useScrollController(); + + if (playlistsQuery.error + case MetadataPluginException( + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + message: _, + )) { + return const Center(child: NoDefaultMetadataPlugin()); + } + + if (authenticated.asData?.value != true) { + return const AnonymousFallback(); + } + + if (playlistsQuery.hasError) { + return ErrorBox( + error: playlistsQuery.error!, + onRetry: () { + ref.invalidate(metadataPluginSavedPlaylistsProvider); + }, + ); + } + + return material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginSavedPlaylistsProvider); + }, + child: SafeArea( + bottom: false, + child: InterScrollbar( + controller: controller, + child: CustomScrollView( + controller: controller, + slivers: [ + SliverAppBar( + automaticallyImplyLeading: false, + floating: true, + backgroundColor: context.theme.colorScheme.background, + flexibleSpace: Container( + padding: const EdgeInsets.symmetric(horizontal: 8), + height: 48, + child: TextField( + onChanged: (value) => searchText.value = value, + placeholder: Text(context.l10n.filter_playlists), + features: const [ + InputFeature.leading(Icon(SpotubeIcons.filter)), + ], + ), + ), + ), + const SliverGap(10), + SliverPadding( + padding: const EdgeInsets.symmetric(horizontal: 8), + sliver: PlaybuttonView( + leading: const Expanded( + child: Row( + children: [ + PlaylistCreateDialogButton(), + // const Gap(10), + // Button.primary( + // leading: const Icon(SpotubeIcons.magic), + // child: Text(context.l10n.generate), + // onPressed: () { + // context.navigateTo(const PlaylistGeneratorRoute()); + // }, + // ), + // const Gap(10), + ], + ), + ), + controller: controller, + hasMore: playlistsQuery.asData?.value.hasMore == true, + isLoading: playlistsQuery.isLoading, + onRequestMore: playlistsQueryNotifier.fetchMore, + itemCount: playlists.length, + gridItemBuilder: (context, index) { + return PlaylistCard(playlists[index]); + }, + listItemBuilder: (context, index) { + return PlaylistCard.tile(playlists[index]); + }, + ), + ), + const SliverSafeArea(sliver: SliverGap(10)), + ], + ), + ), + ), + ); + } +} diff --git a/lib/pages/lyrics/lyrics.dart b/lib/pages/lyrics/lyrics.dart index 0f4f9473..b55dc02e 100644 --- a/lib/pages/lyrics/lyrics.dart +++ b/lib/pages/lyrics/lyrics.dart @@ -1,179 +1,114 @@ -import 'dart:ui'; - -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide Consumer; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; -import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/components/themed_button_tab_bar.dart'; -import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/hooks/utils/use_custom_status_bar_color.dart'; import 'package:spotube/hooks/utils/use_palette_color.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/pages/lyrics/plain_lyrics.dart'; import 'package:spotube/pages/lyrics/synced_lyrics.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/lyrics/synced.dart'; import 'package:spotube/utils/platform.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class LyricsPage extends HookConsumerWidget { static const name = "lyrics"; - final bool isModal; - const LyricsPage({super.key, this.isModal = false}); + const LyricsPage({super.key}); @override Widget build(BuildContext context, ref) { final playlist = ref.watch(audioPlayerProvider); String albumArt = useMemoized( - () => (playlist.activeTrack?.album?.images).asUrlString( - index: (playlist.activeTrack?.album?.images?.length ?? 1) - 1, + () => (playlist.activeTrack?.album.images).asUrlString( + index: (playlist.activeTrack?.album.images.length ?? 1) - 1, placeholder: ImagePlaceholder.albumArt, ), - [playlist.activeTrack?.album?.images], + [playlist.activeTrack?.album.images], ); final palette = usePaletteColor(albumArt, ref); - final mediaQuery = MediaQuery.of(context); - final route = ModalRoute.of(context); + final selectedIndex = useState(0); - final resetStatusBar = useCustomStatusBarColor( - palette.color, - route?.isCurrent ?? false, - noSetBGColor: true, - ); - - PreferredSizeWidget tabbar = ThemedButtonsTabBar( - tabs: [ - Tab(text: " ${context.l10n.synced} "), - Tab(text: " ${context.l10n.plain} "), - ], - ); - - tabbar = PreferredSize( - preferredSize: tabbar.preferredSize, - child: Row( + Widget tabbar = Padding( + padding: const EdgeInsets.all(10), + child: Tabs( + index: selectedIndex.value, + onChanged: (index) => selectedIndex.value = index, children: [ - tabbar, - const Spacer(), - Consumer( - builder: (context, ref, child) { - final playback = ref.watch(audioPlayerProvider); - final lyric = - ref.watch(syncedLyricsProvider(playback.activeTrack)); - final providerName = lyric.asData?.value.provider; - - if (providerName == null) { - return const SizedBox.shrink(); - } - - return Align( - alignment: Alignment.bottomRight, - child: Text(context.l10n.powered_by_provider(providerName)), - ); - }, - ), - const Gap(5), + TabItem(child: Text(context.l10n.synced)), + TabItem(child: Text(context.l10n.plain)), ], ), ); - if (isModal) { - return PopScope( - canPop: true, - onPopInvokedWithResult: (_, __) => resetStatusBar(), - child: DefaultTabController( - length: 2, - child: SafeArea( - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 15, sigmaY: 15), - child: Container( - clipBehavior: Clip.hardEdge, - decoration: BoxDecoration( - color: Theme.of(context).colorScheme.surface.withOpacity(.4), - borderRadius: const BorderRadius.only( - topLeft: Radius.circular(10), - topRight: Radius.circular(10), - ), - ), - child: Column( - children: [ - const SizedBox(height: 5), - Container( - height: 7, - width: 150, - decoration: BoxDecoration( - color: palette.titleTextColor, - borderRadius: BorderRadius.circular(10), - ), - ), - AppBar( - leadingWidth: double.infinity, - leading: tabbar, - backgroundColor: Colors.transparent, - automaticallyImplyLeading: false, - actions: [ - IconButton( - icon: const Icon(SpotubeIcons.minimize), - onPressed: () => Navigator.of(context).pop(), - ), - const SizedBox(width: 5), - ], - ), - Expanded( - child: TabBarView( - children: [ - SyncedLyrics(palette: palette, isModal: isModal), - PlainLyrics(palette: palette, isModal: isModal), - ], - ), - ), - ], - ), - ), - ), - ), + tabbar = Row( + children: [ + tabbar, + const Spacer(), + Consumer( + builder: (context, ref, child) { + final playback = ref.watch(audioPlayerProvider); + final lyric = ref.watch(syncedLyricsProvider(playback.activeTrack)); + final providerName = lyric.asData?.value.provider; + + if (providerName == null) { + return const SizedBox.shrink(); + } + + return Align( + alignment: Alignment.bottomRight, + child: Text(context.l10n.powered_by_provider(providerName)), + ); + }, ), - ); - } - return DefaultTabController( - length: 2, - child: SafeArea( - bottom: mediaQuery.mdAndUp, - child: Scaffold( - extendBodyBehindAppBar: true, - appBar: !kIsMacOS - ? PageWindowTitleBar( + const Gap(5), + ], + ); + + return SafeArea( + bottom: false, + child: Scaffold( + floatingHeader: true, + headers: [ + !kIsMacOS + ? TitleBar( backgroundColor: Colors.transparent, title: tabbar, + height: 58 * context.theme.scaling, + surfaceBlur: 0, + automaticallyImplyLeading: false, ) - : tabbar, - body: Container( - clipBehavior: Clip.hardEdge, - decoration: BoxDecoration( - image: DecorationImage( - image: UniversalImage.imageProvider(albumArt), - fit: BoxFit.cover, - ), - borderRadius: const BorderRadius.only( - bottomLeft: Radius.circular(10), - ), + : tabbar + ], + child: Container( + clipBehavior: Clip.hardEdge, + decoration: BoxDecoration( + image: DecorationImage( + image: UniversalImage.imageProvider(albumArt), + fit: BoxFit.cover, ), - margin: const EdgeInsets.only(bottom: 10), - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 5, sigmaY: 5), - child: ColoredBox( - color: palette.color.withOpacity(.7), - child: SafeArea( - child: TabBarView( - children: [ - SyncedLyrics(palette: palette, isModal: isModal), - PlainLyrics(palette: palette, isModal: isModal), - ], - ), + ), + margin: const EdgeInsets.only(bottom: 10), + child: SurfaceCard( + surfaceBlur: context.theme.surfaceBlur, + surfaceOpacity: context.theme.surfaceOpacity, + padding: EdgeInsets.zero, + borderRadius: BorderRadius.zero, + borderWidth: 0, + child: ColoredBox( + color: palette.color.withValues(alpha: .7), + child: SafeArea( + child: IndexedStack( + index: selectedIndex.value, + children: [ + SyncedLyrics(palette: palette, isModal: false), + PlainLyrics(palette: palette, isModal: false), + ], ), ), ), diff --git a/lib/pages/lyrics/mini_lyrics.dart b/lib/pages/lyrics/mini_lyrics.dart index 8f6ec1fc..4c28eddd 100644 --- a/lib/pages/lyrics/mini_lyrics.dart +++ b/lib/pages/lyrics/mini_lyrics.dart @@ -1,13 +1,13 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:palette_generator/palette_generator.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide Consumer; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/player/player_controls.dart'; import 'package:spotube/modules/player/player_queue.dart'; -import 'package:spotube/modules/root/sidebar.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/hooks/utils/use_force_update.dart'; import 'package:spotube/pages/lyrics/plain_lyrics.dart'; @@ -15,7 +15,9 @@ import 'package:spotube/pages/lyrics/synced_lyrics.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/utils/platform.dart'; import 'package:window_manager/window_manager.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class MiniLyricsPage extends HookConsumerWidget { static const name = "mini_lyrics"; @@ -30,6 +32,8 @@ class MiniLyricsPage extends HookConsumerWidget { final playlistQueue = ref.watch(audioPlayerProvider); + final index = useState(0); + final areaActive = useState(false); final hoverMode = useState(true); final showLyrics = useState(true); @@ -43,8 +47,6 @@ class MiniLyricsPage extends HookConsumerWidget { return null; }, []); - - return MouseRegion( onEnter: !hoverMode.value ? null @@ -56,12 +58,11 @@ class MiniLyricsPage extends HookConsumerWidget { : (event) { areaActive.value = false; }, - child: DefaultTabController( - length: 2, - child: Scaffold( - backgroundColor: theme.colorScheme.surface.withOpacity(0.4), - appBar: PreferredSize( - preferredSize: const Size.fromHeight(60), + child: Scaffold( + backgroundColor: theme.colorScheme.background.withValues(alpha: 0.4), + headers: [ + Padding( + padding: const EdgeInsets.all(8.0), child: AnimatedCrossFade( duration: const Duration(milliseconds: 200), crossFadeState: areaActive.value @@ -70,91 +71,91 @@ class MiniLyricsPage extends HookConsumerWidget { secondChild: const SizedBox(), firstChild: DragToMoveArea( child: Row( + spacing: 2, children: [ const Gap(10), - if (!kIsMacOS) - SizedBox( - height: 30, - width: 30, - child: Sidebar.brandLogo(), - ), - const Spacer(), + if (kIsMacOS) const SizedBox(width: 65), if (showLyrics.value) - SizedBox( - height: 30, - child: TabBar( - tabs: [ - Tab(text: context.l10n.synced), - Tab(text: context.l10n.plain), - ], - isScrollable: true, - ), + Tabs( + index: index.value, + onChanged: (i) { + index.value = i; + }, + children: [ + TabItem(child: Text(context.l10n.synced)), + TabItem(child: Text(context.l10n.plain)), + ], ), const Spacer(), - IconButton( - tooltip: context.l10n.lyrics, - icon: showLyrics.value - ? const Icon(SpotubeIcons.lyrics) - : const Icon(SpotubeIcons.lyricsOff), - style: ButtonStyle( - foregroundColor: showLyrics.value - ? WidgetStateProperty.all(theme.colorScheme.primary) - : null, - ), - onPressed: () async { - showLyrics.value = !showLyrics.value; - areaActive.value = true; - hoverMode.value = false; + Tooltip( + tooltip: + TooltipContainer(child: Text(context.l10n.lyrics)) + .call, + child: IconButton( + variance: showLyrics.value + ? ButtonVariance.secondary + : ButtonVariance.ghost, + icon: showLyrics.value + ? const Icon(SpotubeIcons.lyrics) + : const Icon(SpotubeIcons.lyricsOff), + onPressed: () async { + showLyrics.value = !showLyrics.value; + areaActive.value = true; + hoverMode.value = false; - if (kIsDesktop) { - await windowManager.setSize( - showLyrics.value - ? const Size(400, 500) - : const Size(400, 150), - ); - } - }, - ), - IconButton( - tooltip: context.l10n.show_hide_ui_on_hover, - icon: hoverMode.value - ? const Icon(SpotubeIcons.hoverOn) - : const Icon(SpotubeIcons.hoverOff), - style: ButtonStyle( - foregroundColor: hoverMode.value - ? WidgetStateProperty.all(theme.colorScheme.primary) - : null, + if (kIsDesktop) { + await windowManager.setSize( + showLyrics.value + ? const Size(400, 500) + : const Size(400, 150), + ); + } + }, + ), + ), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.show_hide_ui_on_hover), + ).call, + child: IconButton( + variance: hoverMode.value + ? ButtonVariance.secondary + : ButtonVariance.ghost, + icon: hoverMode.value + ? const Icon(SpotubeIcons.hoverOn) + : const Icon(SpotubeIcons.hoverOff), + onPressed: () async { + areaActive.value = true; + hoverMode.value = !hoverMode.value; + }, ), - onPressed: () async { - areaActive.value = true; - hoverMode.value = !hoverMode.value; - }, ), if (kIsDesktop) FutureBuilder( future: windowManager.isAlwaysOnTop(), builder: (context, snapshot) { - return IconButton( - tooltip: context.l10n.always_on_top, - icon: Icon( - snapshot.data == true - ? SpotubeIcons.pinOn - : SpotubeIcons.pinOff, + return Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.always_on_top), + ).call, + child: IconButton( + variance: snapshot.data == true + ? ButtonVariance.secondary + : ButtonVariance.ghost, + icon: Icon( + snapshot.data == true + ? SpotubeIcons.pinOn + : SpotubeIcons.pinOff, + ), + onPressed: snapshot.data == null + ? null + : () async { + await windowManager.setAlwaysOnTop( + snapshot.data == true ? false : true, + ); + update(); + }, ), - style: ButtonStyle( - foregroundColor: snapshot.data == true - ? WidgetStateProperty.all( - theme.colorScheme.primary) - : null, - ), - onPressed: snapshot.data == null - ? null - : () async { - await windowManager.setAlwaysOnTop( - snapshot.data == true ? false : true, - ); - update(); - }, ); }, ), @@ -163,79 +164,91 @@ class MiniLyricsPage extends HookConsumerWidget { ), ), ), - body: Column( - children: [ - if (playlistQueue.activeTrack != null) - Text( - playlistQueue.activeTrack!.name!, - style: theme.textTheme.titleMedium, - ), - if (showLyrics.value) - Expanded( - child: TabBarView( - children: [ - SyncedLyrics( - palette: PaletteColor(theme.colorScheme.surface, 0), - isModal: true, - defaultTextZoom: 65, - ), - PlainLyrics( - palette: PaletteColor(theme.colorScheme.surface, 0), - isModal: true, - defaultTextZoom: 65, - ), - ], - ), - ) - else - const Gap(20), - AnimatedCrossFade( - crossFadeState: areaActive.value - ? CrossFadeState.showFirst - : CrossFadeState.showSecond, - duration: const Duration(milliseconds: 200), - secondChild: const SizedBox(), - firstChild: Row( + ], + child: Column( + children: [ + if (playlistQueue.activeTrack != null) + Text(playlistQueue.activeTrack!.name!).semiBold(), + if (showLyrics.value) + Expanded( + child: IndexedStack( + index: index.value, children: [ - IconButton( + SyncedLyrics( + palette: PaletteColor(theme.colorScheme.background, 0), + isModal: true, + defaultTextZoom: 65, + ), + PlainLyrics( + palette: PaletteColor(theme.colorScheme.background, 0), + isModal: true, + defaultTextZoom: 65, + ), + ], + ), + ) + else + const Gap(20), + AnimatedCrossFade( + crossFadeState: areaActive.value + ? CrossFadeState.showFirst + : CrossFadeState.showSecond, + duration: const Duration(milliseconds: 200), + secondChild: const SizedBox(), + firstChild: Row( + children: [ + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.queue), + ).call, + child: IconButton.ghost( icon: const Icon(SpotubeIcons.queue), - tooltip: context.l10n.queue, onPressed: playlistQueue.activeTrack != null ? () { - showModalBottomSheet( + openDrawer( context: context, - isDismissible: true, - enableDrag: true, - isScrollControlled: true, - backgroundColor: Colors.black12, - barrierColor: Colors.black12, - shape: RoundedRectangleBorder( - borderRadius: BorderRadius.circular(10), - ), - constraints: BoxConstraints( - maxHeight: - MediaQuery.of(context).size.height * .7, - ), - builder: (context) { - return Consumer(builder: (context, ref, _) { - final playlist = - ref.watch(audioPlayerProvider); - - return PlayerQueue.fromAudioPlayerNotifier( - floating: true, - playlist: playlist, - notifier: ref - .read(audioPlayerProvider.notifier), + barrierDismissible: true, + draggable: true, + barrierColor: Colors.black.withAlpha(100), + borderRadius: BorderRadius.circular(10), + transformBackdrop: false, + position: OverlayPosition.bottom, + surfaceBlur: context.theme.surfaceBlur, + surfaceOpacity: 0.7, + expands: true, + builder: (context) => Consumer( + builder: (context, ref, _) { + final playlist = ref.watch( + audioPlayerProvider, ); - }); - }, + final playlistNotifier = + ref.read(audioPlayerProvider.notifier); + return ConstrainedBox( + constraints: BoxConstraints( + maxHeight: + MediaQuery.of(context).size.height * + 0.8, + ), + child: + PlayerQueue.fromAudioPlayerNotifier( + floating: false, + playlist: playlist, + notifier: playlistNotifier, + ), + ); + }, + ), ); } : null, ), - const Flexible(child: PlayerControls(compact: true)), - IconButton( - tooltip: context.l10n.exit_mini_player, + ), + const Flexible(child: PlayerControls(compact: true)), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.exit_mini_player)) + .call, + child: IconButton.ghost( icon: const Icon(SpotubeIcons.maximize), onPressed: () async { if (!kIsDesktop) return; @@ -257,16 +270,16 @@ class MiniLyricsPage extends HookConsumerWidget { const Duration(milliseconds: 200)); } finally { if (context.mounted) { - GoRouter.of(context).go('/lyrics'); + context.navigateTo(const LyricsRoute()); } } }, ), - ], - ), - ) - ], - ), + ), + ], + ), + ) + ], ), ), ); diff --git a/lib/pages/lyrics/plain_lyrics.dart b/lib/pages/lyrics/plain_lyrics.dart index 7c571d5f..3f0d7d1b 100644 --- a/lib/pages/lyrics/plain_lyrics.dart +++ b/lib/pages/lyrics/plain_lyrics.dart @@ -1,18 +1,18 @@ import 'package:collection/collection.dart'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:palette_generator/palette_generator.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/lyrics/zoom_controls.dart'; import 'package:spotube/components/shimmers/shimmer_lyrics.dart'; -import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/lyrics/synced.dart'; class PlainLyrics extends HookConsumerWidget { final PaletteColor palette; @@ -30,7 +30,7 @@ class PlainLyrics extends HookConsumerWidget { final playlist = ref.watch(audioPlayerProvider); final lyricsQuery = ref.watch(syncedLyricsProvider(playlist.activeTrack)); final mediaQuery = MediaQuery.of(context); - final textTheme = Theme.of(context).textTheme; + final typography = Theme.of(context).typography; final textZoomLevel = useState(defaultTextZoom); @@ -44,20 +44,19 @@ class PlainLyrics extends HookConsumerWidget { child: Text( playlist.activeTrack?.name ?? "", style: mediaQuery.mdAndUp - ? textTheme.displaySmall - : textTheme.headlineMedium?.copyWith( - fontSize: 25, + ? typography.h3 + : typography.h4.copyWith( color: palette.titleTextColor, ), ), ), Center( child: Text( - playlist.activeTrack?.artists?.asString() ?? "", - style: (mediaQuery.mdAndUp - ? textTheme.headlineSmall - : textTheme.titleLarge) - ?.copyWith(color: palette.bodyTextColor), + playlist.activeTrack?.artists.asString() ?? "", + style: (mediaQuery.mdAndUp ? typography.h4 : typography.large) + .copyWith( + color: palette.bodyTextColor, + ), ), ) ], @@ -79,7 +78,7 @@ class PlainLyrics extends HookConsumerWidget { children: [ Text( context.l10n.no_lyrics_available, - style: textTheme.bodyLarge?.copyWith( + style: typography.large.copyWith( color: palette.bodyTextColor, ), textAlign: TextAlign.center, @@ -107,7 +106,9 @@ class PlainLyrics extends HookConsumerWidget { return AnimatedDefaultTextStyle( duration: const Duration(milliseconds: 200), style: TextStyle( - color: palette.bodyTextColor, + color: isModal == true + ? context.theme.colorScheme.foreground + : palette.bodyTextColor, fontSize: 24 * textZoomLevel.value / 100, height: textZoomLevel.value < 70 ? 1.5 @@ -117,7 +118,7 @@ class PlainLyrics extends HookConsumerWidget { ), child: SelectableText( lyrics == null && playlist.activeTrack == null - ? "No Track being played currently" + ? context.l10n.no_tracks_playing : lyrics ?? "", textAlign: TextAlign.center, ), diff --git a/lib/pages/lyrics/synced_lyrics.dart b/lib/pages/lyrics/synced_lyrics.dart index 59bd863a..cb331724 100644 --- a/lib/pages/lyrics/synced_lyrics.dart +++ b/lib/pages/lyrics/synced_lyrics.dart @@ -1,26 +1,23 @@ import 'dart:async'; -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:palette_generator/palette_generator.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/modules/lyrics/zoom_controls.dart'; import 'package:spotube/components/shimmers/shimmer_lyrics.dart'; -import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/hooks/controllers/use_auto_scroll_controller.dart'; import 'package:spotube/modules/lyrics/use_synced_lyrics.dart'; import 'package:scroll_to_index/scroll_to_index.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/lyrics/synced.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/logger/logger.dart'; -import 'package:stroke_text/stroke_text.dart'; - class SyncedLyrics extends HookConsumerWidget { final PaletteColor palette; final bool? isModal; @@ -35,9 +32,11 @@ class SyncedLyrics extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { + final mediaQuery = MediaQuery.sizeOf(context); + final theme = Theme.of(context); + final playlist = ref.watch(audioPlayerProvider); - final mediaQuery = MediaQuery.of(context); final controller = useAutoScrollController(); final delay = ref.watch(syncedLyricsDelayProvider); @@ -54,7 +53,7 @@ class SyncedLyrics extends HookConsumerWidget { useSyncedLyrics(ref, lyricsState.asData?.value.lyricsMap ?? {}, delay); final textZoomLevel = useState(defaultTextZoom); - final textTheme = Theme.of(context).textTheme; + final typography = Theme.of(context).typography; ref.listen( audioPlayerProvider.select((s) => s.activeTrack), @@ -69,11 +68,13 @@ class SyncedLyrics extends HookConsumerWidget { ); final headlineTextStyle = (mediaQuery.mdAndUp - ? textTheme.displaySmall - : textTheme.headlineMedium?.copyWith(fontSize: 25)) - ?.copyWith(color: palette.titleTextColor); + ? typography.h3 + : typography.h4.copyWith(fontSize: 25)) + .copyWith( + color: palette.titleTextColor, + ); - final bodyTextTheme = textTheme.bodyLarge?.copyWith( + final bodyTextTheme = typography.large.copyWith( color: palette.bodyTextColor, ); @@ -114,10 +115,9 @@ class SyncedLyrics extends HookConsumerWidget { bottom: PreferredSize( preferredSize: const Size.fromHeight(40), child: Text( - playlist.activeTrack?.artists?.asString() ?? "", - style: mediaQuery.mdAndUp - ? textTheme.headlineSmall - : textTheme.titleLarge, + playlist.activeTrack?.artists.asString() ?? "", + style: + mediaQuery.mdAndUp ? typography.h4 : typography.x2Large, ), ), ), @@ -144,7 +144,7 @@ class SyncedLyrics extends HookConsumerWidget { ? Container( padding: index == lyricValue.lyrics.length - 1 ? EdgeInsets.only( - bottom: mediaQuery.size.height / 2, + bottom: mediaQuery.height / 2, ) : null, ) @@ -158,6 +158,9 @@ class SyncedLyrics extends HookConsumerWidget { child: AnimatedDefaultTextStyle( duration: const Duration(milliseconds: 250), style: TextStyle( + color: isActive + ? theme.colorScheme.foreground + : theme.colorScheme.mutedForeground, fontWeight: isActive ? FontWeight.w500 : FontWeight.normal, @@ -165,31 +168,22 @@ class SyncedLyrics extends HookConsumerWidget { (textZoomLevel.value / 100), ), textAlign: TextAlign.center, - child: InkWell( - onTap: () async { - final time = Duration( - seconds: - lyricSlice.time.inSeconds - delay, - ); - if (time > audioPlayer.duration || - time.isNegative) { - return; - } - audioPlayer.seek(time); - }, - child: Builder(builder: (context) { - return StrokeText( - text: lyricSlice.text, - textStyle: - DefaultTextStyle.of(context).style, - textColor: isActive - ? Colors.white - : palette.bodyTextColor, - strokeColor: isActive - ? Colors.black - : Colors.transparent, - ); - }), + child: MouseRegion( + cursor: SystemMouseCursors.click, + child: GestureDetector( + onTap: () async { + final time = Duration( + seconds: + lyricSlice.time.inSeconds - delay, + ); + if (time > audioPlayer.duration || + time.isNegative) { + return; + } + audioPlayer.seek(time); + }, + child: Text(lyricSlice.text), + ), ), ), ), @@ -225,18 +219,17 @@ class SyncedLyrics extends HookConsumerWidget { text: TextSpan( style: bodyTextTheme, children: [ - const TextSpan( - text: - "Synced lyrics are not available for this song. Please use the", + TextSpan( + text: context.l10n.synced_lyrics_not_available, ), TextSpan( - text: " Plain Lyrics ", - style: textTheme.bodyLarge?.copyWith( + text: " ${context.l10n.plain_lyrics} ", + style: typography.large.copyWith( color: palette.bodyTextColor, fontWeight: FontWeight.bold, ), ), - const TextSpan(text: "tab instead."), + TextSpan(text: context.l10n.tab_instead), ], ), ), diff --git a/lib/pages/mobile_login/hooks/login_callback.dart b/lib/pages/mobile_login/hooks/login_callback.dart deleted file mode 100644 index 07c0210a..00000000 --- a/lib/pages/mobile_login/hooks/login_callback.dart +++ /dev/null @@ -1,79 +0,0 @@ -import 'dart:io'; - -import 'package:desktop_webview_window/desktop_webview_window.dart'; -import 'package:flutter/material.dart'; -import 'package:flutter/services.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:path/path.dart'; -import 'package:path_provider/path_provider.dart'; -import 'package:spotube/pages/mobile_login/mobile_login.dart'; -import 'package:spotube/pages/mobile_login/no_webview_runtime_dialog.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/utils/platform.dart'; - -Future Function() useLoginCallback(WidgetRef ref) { - final context = useContext(); - final theme = Theme.of(context); - final authNotifier = ref.read(authenticationProvider.notifier); - - return useCallback(() async { - if (kIsMobile || kIsMacOS) { - context.pushNamed(WebViewLogin.name); - return; - } - - try { - final exp = RegExp(r"https:\/\/accounts.spotify.com\/.+\/status"); - final applicationSupportDir = await getApplicationSupportDirectory(); - final userDataFolder = Directory( - join(applicationSupportDir.path, "webview_window_Webview2")); - - if (!await userDataFolder.exists()) { - await userDataFolder.create(); - } - - final webview = await WebviewWindow.create( - configuration: CreateConfiguration( - title: "Spotify Login", - titleBarTopPadding: kIsMacOS ? 20 : 0, - windowHeight: 720, - windowWidth: 1280, - userDataFolderWindows: userDataFolder.path, - ), - ); - webview - ..setBrightness(theme.colorScheme.brightness) - ..launch("https://accounts.spotify.com/") - ..setOnUrlRequestCallback((url) { - if (exp.hasMatch(url)) { - webview.getAllCookies().then((cookies) async { - final cookieHeader = - "sp_dc=${cookies.firstWhere((element) => element.name.contains("sp_dc")).value.replaceAll("\u0000", "")}"; - - await authNotifier.login(cookieHeader); - - webview.close(); - if (context.mounted) { - context.go("/"); - } - }); - } - - return true; - }); - } on PlatformException catch (_) { - if (!await WebviewWindow.isWebviewAvailable()) { - WidgetsBinding.instance.addPostFrameCallback((timeStamp) { - showDialog( - context: context, - builder: (context) { - return const NoWebviewRuntimeDialog(); - }, - ); - }); - } - } - }, [authNotifier, theme, context.go, context.pushNamed]); -} diff --git a/lib/pages/mobile_login/mobile_login.dart b/lib/pages/mobile_login/mobile_login.dart deleted file mode 100644 index c45c2184..00000000 --- a/lib/pages/mobile_login/mobile_login.dart +++ /dev/null @@ -1,71 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_inappwebview/flutter_inappwebview.dart'; -import 'package:go_router/go_router.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/components/titlebar/titlebar.dart'; - -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/utils/platform.dart'; - -class WebViewLogin extends HookConsumerWidget { - static const name = "login"; - const WebViewLogin({super.key}); - - @override - Widget build(BuildContext context, ref) { - final authenticationNotifier = ref.watch(authenticationProvider.notifier); - - if (kIsDesktop) { - const Scaffold( - body: Center( - child: Text('This feature is not available on desktop'), - ), - ); - } - - return Scaffold( - appBar: const PageWindowTitleBar( - leading: BackButton(color: Colors.white), - backgroundColor: Colors.transparent, - ), - extendBodyBehindAppBar: true, - body: InAppWebView( - initialSettings: InAppWebViewSettings( - userAgent: - "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 safari/537.36", - ), - initialUrlRequest: URLRequest( - url: WebUri("https://accounts.spotify.com/"), - ), - onPermissionRequest: (controller, permissionRequest) async { - return PermissionResponse( - resources: permissionRequest.resources, - action: PermissionResponseAction.GRANT, - ); - }, - onLoadStop: (controller, action) async { - if (action == null) return; - String url = action.toString(); - if (url.endsWith("/")) { - url = url.substring(0, url.length - 1); - } - - final exp = RegExp(r"https:\/\/accounts.spotify.com\/.+\/status"); - - if (exp.hasMatch(url)) { - final cookies = - await CookieManager.instance().getCookies(url: action); - final cookieHeader = - "sp_dc=${cookies.firstWhere((element) => element.name == "sp_dc").value}"; - - await authenticationNotifier.login(cookieHeader); - if (context.mounted) { - // ignore: use_build_context_synchronously - GoRouter.of(context).go("/"); - } - } - }, - ), - ); - } -} diff --git a/lib/pages/mobile_login/no_webview_runtime_dialog.dart b/lib/pages/mobile_login/no_webview_runtime_dialog.dart deleted file mode 100644 index a6cc5ffb..00000000 --- a/lib/pages/mobile_login/no_webview_runtime_dialog.dart +++ /dev/null @@ -1,50 +0,0 @@ -import 'package:flutter/material.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:url_launcher/url_launcher_string.dart'; - -class NoWebviewRuntimeDialog extends StatelessWidget { - const NoWebviewRuntimeDialog({super.key}); - - @override - Widget build(BuildContext context) { - final ThemeData(:platform) = Theme.of(context); - - return AlertDialog( - title: Text(context.l10n.webview_not_found), - content: Text(context.l10n.webview_not_found_description), - actions: [ - TextButton( - onPressed: () { - Navigator.of(context).pop(); - }, - child: Text(context.l10n.cancel), - ), - FilledButton( - onPressed: () async { - final url = switch (platform) { - TargetPlatform.windows => - 'https://developer.microsoft.com/en-us/microsoft-edge/webview2', - TargetPlatform.macOS => 'https://www.apple.com/safari/', - TargetPlatform.linux => - 'https://webkitgtk.org/reference/webkit2gtk/stable/', - _ => "", - }; - if (url.isEmpty) { - ScaffoldMessenger.of(context).showSnackBar( - const SnackBar(content: Text('Unsupported platform')), - ); - } - - await launchUrlString(url); - }, - child: Text(switch (platform) { - TargetPlatform.windows => 'Download Edge WebView2', - TargetPlatform.macOS => 'Download Safari', - TargetPlatform.linux => 'Download Webkit2Gtk', - _ => 'Download Webview', - }), - ), - ], - ); - } -} diff --git a/lib/pages/player/lyrics.dart b/lib/pages/player/lyrics.dart new file mode 100644 index 00000000..093b0aa2 --- /dev/null +++ b/lib/pages/player/lyrics.dart @@ -0,0 +1,62 @@ +import 'package:auto_route/annotations.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/button/back_button.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/hooks/utils/use_palette_color.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/pages/lyrics/plain_lyrics.dart'; +import 'package:spotube/pages/lyrics/synced_lyrics.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; + +@RoutePage() +class PlayerLyricsPage extends HookConsumerWidget { + const PlayerLyricsPage({super.key}); + + @override + Widget build(BuildContext context, ref) { + final playlist = ref.watch(audioPlayerProvider); + String albumArt = useMemoized( + () => (playlist.activeTrack?.album.images).asUrlString( + index: (playlist.activeTrack?.album.images.length ?? 1) - 1, + placeholder: ImagePlaceholder.albumArt, + ), + [playlist.activeTrack?.album.images], + ); + final selectedIndex = useState(0); + final palette = usePaletteColor(albumArt, ref); + + final tabbar = TabList( + index: selectedIndex.value, + onChanged: (index) => selectedIndex.value = index, + children: [ + TabItem( + child: Text(context.l10n.synced), + ), + TabItem( + child: Text(context.l10n.plain), + ), + ], + ); + + return Scaffold( + headers: [ + AppBar( + leading: [tabbar], + trailing: const [ + BackButton(icon: SpotubeIcons.angleDown), + ], + ), + ], + child: IndexedStack( + index: selectedIndex.value, + children: [ + SyncedLyrics(palette: palette, isModal: false), + PlainLyrics(palette: palette, isModal: false), + ], + ), + ); + } +} diff --git a/lib/pages/player/queue.dart b/lib/pages/player/queue.dart new file mode 100644 index 00000000..829db6eb --- /dev/null +++ b/lib/pages/player/queue.dart @@ -0,0 +1,28 @@ +import 'package:auto_route/annotations.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/modules/player/player_queue.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; + +@RoutePage() +class PlayerQueuePage extends HookConsumerWidget { + const PlayerQueuePage({super.key}); + + @override + Widget build(BuildContext context, ref) { + final playlist = ref.watch( + audioPlayerProvider, + ); + final playlistNotifier = ref.read(audioPlayerProvider.notifier); + return Scaffold( + child: SafeArea( + bottom: false, + child: PlayerQueue.fromAudioPlayerNotifier( + floating: false, + playlist: playlist, + notifier: playlistNotifier, + ), + ), + ); + } +} diff --git a/lib/pages/player/sources.dart b/lib/pages/player/sources.dart new file mode 100644 index 00000000..8e370daf --- /dev/null +++ b/lib/pages/player/sources.dart @@ -0,0 +1,15 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/modules/player/sibling_tracks_sheet.dart'; + +@RoutePage() +class PlayerTrackSourcesPage extends StatelessWidget { + const PlayerTrackSourcesPage({super.key}); + + @override + Widget build(BuildContext context) { + return const Scaffold( + child: SiblingTracksSheet(floating: false), + ); + } +} diff --git a/lib/pages/playlist/liked_playlist.dart b/lib/pages/playlist/liked_playlist.dart index 942f46d5..3897acef 100644 --- a/lib/pages/playlist/liked_playlist.dart +++ b/lib/pages/playlist/liked_playlist.dart @@ -1,15 +1,20 @@ +import 'package:flutter/material.dart' as material; import 'package:flutter/widgets.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/components/tracks_view/track_view.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; +import 'package:spotube/collections/assets.gen.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/track_presentation.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/pages/playlist/playlist.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/library/tracks.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +@RoutePage() class LikedPlaylistPage extends HookConsumerWidget { static const name = PlaylistPage.name; - final PlaylistSimple playlist; + final SpotubeSimplePlaylistObject playlist; const LikedPlaylistPage({ super.key, required this.playlist, @@ -17,31 +22,43 @@ class LikedPlaylistPage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final likedTracks = ref.watch(likedTracksProvider); - final tracks = likedTracks.asData?.value ?? []; + final likedTracks = ref.watch(metadataPluginSavedTracksProvider); + final likedTracksNotifier = + ref.watch(metadataPluginSavedTracksProvider.notifier); + final tracks = likedTracks.asData?.value.items ?? []; - return InheritedTrackView( - collection: playlist, - image: "assets/liked-tracks.jpg", - pagination: PaginationProps( - hasNextPage: false, - isLoading: false, - onFetchMore: () {}, - onFetchAll: () async { - return tracks.toList(); - }, - onRefresh: () async { - ref.invalidate(likedTracksProvider); - }, + return material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginSavedTracksProvider); + }, + child: TrackPresentation( + options: TrackPresentationOptions( + collection: playlist, + image: Assets.images.likedTracks.path, + pagination: PaginationProps( + hasNextPage: likedTracks.asData?.value.hasMore ?? false, + isLoading: likedTracks.isLoadingNextPage && !likedTracks.isLoading, + onFetchMore: () async { + await likedTracksNotifier.fetchMore(); + }, + onFetchAll: () async { + return await likedTracksNotifier.fetchAll(); + }, + onRefresh: () async { + ref.invalidate(metadataPluginSavedTracksProvider); + }, + ), + title: playlist.name, + description: playlist.description, + tracks: tracks, + error: likedTracks.error, + routePath: '/playlist/${playlist.id}', + isLiked: false, + shareUrl: null, + onHeart: null, + owner: playlist.owner.name, + ), ), - title: playlist.name!, - description: playlist.description, - tracks: tracks, - routePath: '/playlist/${playlist.id}', - isLiked: false, - shareUrl: "", - onHeart: null, - child: const TrackView(), ); } } diff --git a/lib/pages/playlist/playlist.dart b/lib/pages/playlist/playlist.dart index e1b33e98..4aca5945 100644 --- a/lib/pages/playlist/playlist.dart +++ b/lib/pages/playlist/playlist.dart @@ -1,29 +1,35 @@ +import 'package:flutter/material.dart' as material; import 'package:collection/collection.dart'; import 'package:flutter/material.dart' hide Page; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/components/dialogs/prompt_dialog.dart'; -import 'package:spotube/components/tracks_view/sections/body/use_is_user_playlist.dart'; -import 'package:spotube/components/tracks_view/track_view.dart'; -import 'package:spotube/components/tracks_view/track_view_props.dart'; +import 'package:spotube/components/track_presentation/presentation_props.dart'; +import 'package:spotube/components/track_presentation/track_presentation.dart'; +import 'package:spotube/components/track_presentation/use_is_user_playlist.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/playlist.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +@RoutePage() class PlaylistPage extends HookConsumerWidget { static const name = "playlist"; - final PlaylistSimple _playlist; + final SpotubeSimplePlaylistObject _playlist; + final String id; const PlaylistPage({ super.key, - required PlaylistSimple playlist, + @PathParam("id") required this.id, + required SpotubeSimplePlaylistObject playlist, }) : _playlist = playlist; @override Widget build(BuildContext context, ref) { final playlist = ref .watch( - favoritePlaylistsProvider.select( + metadataPluginSavedPlaylistsProvider.select( (value) => value.whenData( (value) => value.items.firstWhereOrNull((s) => s.id == _playlist.id), @@ -34,60 +40,74 @@ class PlaylistPage extends HookConsumerWidget { ?.value ?? _playlist; - final tracks = ref.watch(playlistTracksProvider(playlist.id!)); + final tracks = ref.watch(metadataPluginPlaylistTracksProvider(playlist.id)); final tracksNotifier = - ref.watch(playlistTracksProvider(playlist.id!).notifier); + ref.watch(metadataPluginPlaylistTracksProvider(playlist.id).notifier); final isFavoritePlaylist = - ref.watch(isFavoritePlaylistProvider(playlist.id!)); + ref.watch(metadataPluginIsSavedPlaylistProvider(playlist.id)); final favoritePlaylistsNotifier = - ref.watch(favoritePlaylistsProvider.notifier); + ref.watch(metadataPluginSavedPlaylistsProvider.notifier); - final isUserPlaylist = useIsUserPlaylist(ref, playlist.id!); + final isUserPlaylist = useIsUserPlaylist(ref, playlist.id); - return InheritedTrackView( - collection: playlist, - image: playlist.images.asUrlString( - placeholder: ImagePlaceholder.collection, - ), - pagination: PaginationProps( - hasNextPage: tracks.asData?.value.hasMore ?? false, - isLoading: tracks.isLoadingNextPage, - onFetchMore: tracksNotifier.fetchMore, - onRefresh: () async { - ref.invalidate(playlistTracksProvider(playlist.id!)); - }, - onFetchAll: () async { - return await tracksNotifier.fetchAll(); - }, - ), - title: playlist.name!, - description: playlist.description, - tracks: tracks.asData?.value.items ?? [], - routePath: '/playlist/${playlist.id}', - isLiked: isFavoritePlaylist.asData?.value ?? false, - shareUrl: playlist.externalUrls?.spotify ?? - "https://open.spotify.com/playlist/${playlist.id}", - onHeart: isFavoritePlaylist.asData?.value == null - ? null - : () async { - final confirmed = isUserPlaylist - ? await showPromptDialog( - context: context, - title: context.l10n.delete_playlist, - message: context.l10n.delete_playlist_confirmation, - ) - : true; - if (!confirmed) return null; - - if (isFavoritePlaylist.asData!.value) { - await favoritePlaylistsNotifier.removeFavorite(playlist); - } else { - await favoritePlaylistsNotifier.addFavorite(playlist); - } - return isUserPlaylist; + return material.RefreshIndicator.adaptive( + onRefresh: () async { + ref.invalidate(metadataPluginPlaylistTracksProvider(playlist.id)); + ref.invalidate(metadataPluginSavedPlaylistsProvider); + ref.invalidate(metadataPluginIsSavedPlaylistProvider(playlist.id)); + }, + child: TrackPresentation( + options: TrackPresentationOptions( + collection: playlist, + image: playlist.images.asUrlString( + placeholder: ImagePlaceholder.collection, + ), + pagination: PaginationProps( + hasNextPage: tracks.asData?.value.hasMore ?? false, + isLoading: tracks.isLoading || tracks.isLoadingNextPage, + onFetchMore: tracksNotifier.fetchMore, + onRefresh: () async { + ref.invalidate(metadataPluginPlaylistTracksProvider(playlist.id)); }, - child: const TrackView(), + onFetchAll: () async { + return await tracksNotifier.fetchAll(); + }, + ), + title: playlist.name, + description: playlist.description, + owner: playlist.owner.name, + ownerImage: playlist.owner.images.lastOrNull?.url, + tracks: tracks.asData?.value.items ?? [], + error: tracks.error, + routePath: '/playlist/${playlist.id}', + isLiked: isFavoritePlaylist.asData?.value ?? false, + shareUrl: playlist.externalUri, + onHeart: isFavoritePlaylist.asData?.value == null + ? null + : () async { + final confirmed = isUserPlaylist + ? await showPromptDialog( + context: context, + title: context.l10n.delete_playlist, + message: context.l10n.delete_playlist_confirmation, + ) + : true; + if (!confirmed) return null; + + if (isFavoritePlaylist.asData!.value) { + if (isUserPlaylist) { + await favoritePlaylistsNotifier.delete(playlist.id); + } else { + await favoritePlaylistsNotifier.removeFavorite(playlist); + } + } else { + await favoritePlaylistsNotifier.addFavorite(playlist); + } + return isUserPlaylist; + }, + ), + ), ); } } diff --git a/lib/pages/profile/profile.dart b/lib/pages/profile/profile.dart index 9e51793d..eb3dec2a 100644 --- a/lib/pages/profile/profile.dart +++ b/lib/pages/profile/profile.dart @@ -1,19 +1,18 @@ -import 'package:flutter/material.dart'; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:sliver_tools/sliver_tools.dart'; import 'package:spotube/collections/fake.dart'; -import 'package:spotube/collections/spotify_markets.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; import 'package:url_launcher/url_launcher_string.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class ProfilePage extends HookConsumerWidget { static const name = "profile"; @@ -21,34 +20,31 @@ class ProfilePage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme) = Theme.of(context); - - final me = ref.watch(meProvider); + final me = ref.watch(metadataPluginUserProvider); final meData = me.asData?.value ?? FakeData.user; - final userProperties = useMemoized( - () => { - context.l10n.email: meData.email ?? "N/A", - context.l10n.profile_followers: - meData.followers?.total.toString() ?? "N/A", - context.l10n.birthday: meData.birthdate ?? context.l10n.not_born, - context.l10n.country: spotifyMarkets - .firstWhere((market) => market.$1 == meData.country) - .$2, - context.l10n.subscription: meData.product ?? context.l10n.hacker, - }, - [meData], - ); + // final userProperties = useMemoized( + // () => { + // context.l10n.email: meData.email ?? "N/A", + // context.l10n.profile_followers: + // meData.followers?.total.toString() ?? "N/A", + // context.l10n.birthday: meData.birthdate ?? context.l10n.not_born, + // context.l10n.country: markets + // .firstWhere((market) => market.$1 == meData.country) + // .$2, + // context.l10n.subscription: meData.product ?? context.l10n.hacker, + // }, + // [meData], + // ); return SafeArea( child: Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.profile), - titleSpacing: 0, - automaticallyImplyLeading: true, - centerTitle: false, - ), - body: Skeletonizer( + headers: [ + TitleBar( + title: Text(context.l10n.profile), + ) + ], + child: Skeletonizer( enabled: me.isLoading, child: CustomScrollView( slivers: [ @@ -74,10 +70,9 @@ class ProfilePage extends HookConsumerWidget { const SliverGap(10), SliverToBoxAdapter( child: Text( - meData.displayName ?? context.l10n.no_name, - style: textTheme.titleLarge, + meData.name, textAlign: TextAlign.center, - ), + ).h4(), ), const SliverGap(20), SliverCrossAxisConstrained( @@ -86,59 +81,57 @@ class ProfilePage extends HookConsumerWidget { child: Row( mainAxisAlignment: MainAxisAlignment.end, children: [ - TextButton.icon( - label: Text(context.l10n.edit), - icon: const Icon(SpotubeIcons.edit), + Button.text( + leading: const Icon(SpotubeIcons.edit), onPressed: () { launchUrlString( - "https://www.spotify.com/account/profile/", + meData.externalUri, mode: LaunchMode.externalApplication, ); }, + child: Text(context.l10n.edit), ), ], ), ), ), - SliverCrossAxisConstrained( - maxCrossAxisExtent: 500, - child: SliverToBoxAdapter( - child: Card( - margin: const EdgeInsets.all(10), - child: Padding( - padding: const EdgeInsets.all(8.0), - child: Table( - columnWidths: const { - 0: FixedColumnWidth(110), - }, - children: [ - for (final MapEntry(:key, :value) - in userProperties.entries) - TableRow( - children: [ - TableCell( - child: Padding( - padding: const EdgeInsets.all(6), - child: Text( - key, - style: textTheme.titleSmall, - ), - ), - ), - TableCell( - child: Padding( - padding: const EdgeInsets.all(6), - child: Text(value), - ), - ), - ], - ) - ], - ), - ), - ), - ), - ), + // SliverCrossAxisConstrained( + // maxCrossAxisExtent: 500, + // child: SliverToBoxAdapter( + // child: Card( + // child: Padding( + // padding: const EdgeInsets.all(8.0), + // child: Table( + // columnWidths: const { + // 0: FixedTableSize(120), + // }, + // defaultRowHeight: const FixedTableSize(40), + // rows: [ + // for (final MapEntry(:key, :value) + // in userProperties.entries) + // TableRow( + // cells: [ + // TableCell( + // child: Padding( + // padding: const EdgeInsets.all(6), + // child: Text(key).large(), + // ), + // ), + // TableCell( + // child: Padding( + // padding: const EdgeInsets.all(6), + // child: Text(value), + // ), + // ), + // ], + // ) + // ], + // ), + // ), + // ), + // ), + // ), + const SliverGap(200), ], ), ), diff --git a/lib/pages/root/root_app.dart b/lib/pages/root/root_app.dart index 0274de00..44b8416f 100644 --- a/lib/pages/root/root_app.dart +++ b/lib/pages/root/root_app.dart @@ -1,235 +1,68 @@ -import 'dart:async'; - -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; import 'package:flutter/services.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/side_bar_tiles.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/framework/app_pop_scope.dart'; -import 'package:spotube/modules/player/player_queue.dart'; -import 'package:spotube/components/dialogs/replace_downloaded_dialog.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/hooks/configurators/use_check_yt_dlp_installed.dart'; import 'package:spotube/modules/root/bottom_player.dart'; -import 'package:spotube/modules/root/sidebar.dart'; +import 'package:spotube/modules/root/sidebar/sidebar.dart'; import 'package:spotube/modules/root/spotube_navigation_bar.dart'; -import 'package:spotube/extensions/context.dart'; import 'package:spotube/hooks/configurators/use_endless_playback.dart'; -import 'package:spotube/pages/home/home.dart'; -import 'package:spotube/provider/download_manager_provider.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/server/routes/connect.dart'; -import 'package:spotube/services/connectivity_adapter.dart'; -import 'package:spotube/utils/platform.dart'; -import 'package:spotube/utils/service_utils.dart'; +import 'package:spotube/modules/root/use_global_subscriptions.dart'; +import 'package:spotube/provider/glance/glance.dart'; -class RootApp extends HookConsumerWidget { - final Widget child; - const RootApp({ - required this.child, - super.key, - }); +@RoutePage() +class RootAppPage extends HookConsumerWidget { + const RootAppPage({super.key}); @override Widget build(BuildContext context, ref) { - final theme = Theme.of(context); + final backgroundColor = Theme.of(context).colorScheme.background; + final brightness = Theme.of(context).brightness; - final showingDialogCompleter = useRef(Completer()..complete()); - final downloader = ref.watch(downloadManagerProvider); - final scaffoldMessenger = ScaffoldMessenger.of(context); - final connectRoutes = ref.watch(serverConnectRoutesProvider); - - useEffect(() { - WidgetsBinding.instance.addPostFrameCallback((_) async { - ServiceUtils.checkForUpdates(context, ref); - }); - - final subscriptions = [ - ConnectionCheckerService.instance.onConnectivityChanged - .listen((status) { - if (status) { - scaffoldMessenger.showSnackBar( - SnackBar( - content: Row( - children: [ - Icon( - SpotubeIcons.wifi, - color: theme.colorScheme.onPrimary, - ), - const SizedBox(width: 10), - Text(context.l10n.connection_restored), - ], - ), - backgroundColor: theme.colorScheme.primary, - showCloseIcon: true, - width: 350, - ), - ); - } else { - scaffoldMessenger.showSnackBar( - SnackBar( - content: Row( - children: [ - Icon( - SpotubeIcons.noWifi, - color: theme.colorScheme.onError, - ), - const SizedBox(width: 10), - Text(context.l10n.you_are_offline), - ], - ), - backgroundColor: theme.colorScheme.error, - showCloseIcon: true, - width: 300, - ), - ); - } - }), - connectRoutes.connectClientStream.listen((clientOrigin) { - scaffoldMessenger.showSnackBar( - SnackBar( - backgroundColor: Colors.yellow[600], - behavior: SnackBarBehavior.floating, - content: Row( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon( - SpotubeIcons.error, - color: Colors.black, - ), - const SizedBox(width: 10), - Text( - context.l10n.connect_client_alert(clientOrigin), - style: const TextStyle(color: Colors.black), - ), - ], - ), - ), - ); - }) - ]; - - return () { - for (final subscription in subscriptions) { - subscription.cancel(); - } - }; - }, []); - - useEffect(() { - downloader.onFileExists = (track) async { - if (!context.mounted) return false; - - if (!showingDialogCompleter.value.isCompleted) { - await showingDialogCompleter.value.future; - } - - final replaceAll = ref.read(replaceDownloadedFileState); - - if (replaceAll != null) return replaceAll; - - showingDialogCompleter.value = Completer(); - - if (context.mounted) { - final result = await showDialog( - context: context, - builder: (context) => ReplaceDownloadedDialog( - track: track, - ), - ) ?? - false; - - showingDialogCompleter.value.complete(); - return result; - } - - // it'll never reach here as root_app is always mounted - return false; - }; - return null; - }, [downloader]); - - // checks for latest version of the application + ref.listen(glanceProvider, (_, __) {}); + useGlobalSubscriptions(ref); useEndlessPlayback(ref); - - final backgroundColor = Theme.of(context).scaffoldBackgroundColor; + useCheckYtDlpInstalled(ref); useEffect(() { SystemChrome.setSystemUIOverlayStyle( SystemUiOverlayStyle( statusBarColor: backgroundColor, // status bar color - statusBarIconBrightness: backgroundColor.computeLuminance() > 0.179 - ? Brightness.dark - : Brightness.light, + statusBarIconBrightness: brightness == Brightness.dark + ? Brightness.light + : Brightness.dark, ), ); return null; - }, [backgroundColor]); + }, [backgroundColor, brightness]); - final navTileNames = useMemoized(() { - return getSidebarTileList(context.l10n).map((s) => s.name).toList(); - }, []); - - final scaffold = Scaffold( - body: Sidebar(child: child), - extendBody: true, - drawerScrimColor: Colors.transparent, - endDrawer: kIsDesktop - ? Container( - constraints: const BoxConstraints(maxWidth: 800), - decoration: BoxDecoration( - boxShadow: theme.brightness == Brightness.light - ? null - : kElevationToShadow[8], + final scaffold = MediaQuery.removeViewInsets( + context: context, + removeBottom: true, + child: SafeArea( + top: false, + child: Scaffold( + footers: const [ + BottomPlayer(), + SpotubeNavigationBar(), + ], + floatingFooter: true, + child: Sidebar( + child: MediaQuery( + data: MediaQuery.of(context).copyWith( + padding: MediaQuery.paddingOf(context) + .copyWith(bottom: 100 * context.theme.scaling), ), - margin: const EdgeInsets.only( - top: 40, - bottom: 100, - ), - child: Consumer( - builder: (context, ref, _) { - final playlist = ref.watch(audioPlayerProvider); - final playlistNotifier = - ref.read(audioPlayerProvider.notifier); - - return PlayerQueue.fromAudioPlayerNotifier( - floating: true, - playlist: playlist, - notifier: playlistNotifier, - ); - }, - ), - ) - : null, - bottomNavigationBar: const Column( - mainAxisSize: MainAxisSize.min, - children: [ - BottomPlayer(), - SpotubeNavigationBar(), - ], + child: const AutoRouter(), + ), + ), + ), ), ); - if (!kIsAndroid) { - return scaffold; - } - - final topRoute = GoRouterState.of(context).topRoute; - final canPop = topRoute != null && !navTileNames.contains(topRoute.name); - - return AppPopScope( - canPop: canPop, - onPopInvoked: (didPop) { - if (didPop) return; - - if (topRoute?.name == HomePage.name) { - SystemNavigator.pop(); - } else { - context.goNamed(HomePage.name); - } - }, - child: scaffold, - ); + return scaffold; } } diff --git a/lib/pages/search/search.dart b/lib/pages/search/search.dart index d5de12f0..cb4f4a0b 100644 --- a/lib/pages/search/search.dart +++ b/lib/pages/search/search.dart @@ -1,31 +1,33 @@ -import 'dart:async'; +import 'package:flutter/services.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; -import 'package:collection/collection.dart'; -import 'package:flutter/material.dart' hide Page; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:fuzzywuzzy/fuzzywuzzy.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/collections/routes.gr.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; -import 'package:spotube/components/fallbacks/anonymous_fallback.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/fallbacks/no_default_metadata_plugin.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/hooks/utils/use_force_update.dart'; -import 'package:spotube/pages/search/sections/albums.dart'; -import 'package:spotube/pages/search/sections/artists.dart'; -import 'package:spotube/pages/search/sections/playlists.dart'; -import 'package:spotube/pages/search/sections/tracks.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/extensions/string.dart'; +import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart'; +import 'package:spotube/pages/search/tabs/albums.dart'; +import 'package:spotube/pages/search/tabs/all.dart'; +import 'package:spotube/pages/search/tabs/artists.dart'; +import 'package:spotube/pages/search/tabs/playlists.dart'; +import 'package:spotube/pages/search/tabs/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/search/all.dart'; import 'package:spotube/services/kv_store/kv_store.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; -import 'package:spotube/utils/platform.dart'; +final searchTermStateProvider = StateProvider((ref) { + return ""; +}); +@RoutePage() class SearchPage extends HookConsumerWidget { static const name = "search"; @@ -33,21 +35,21 @@ class SearchPage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final theme = Theme.of(context); + final controller = useShadcnTextEditingController(); + final focusNode = useFocusNode(); + final searchTerm = ref.watch(searchTermStateProvider); - final controller = useSearchController(); + final searchChipSnapshot = ref.watch(metadataPluginSearchChipsProvider); + final selectedChip = useState( + searchChipSnapshot.asData?.value.first ?? "all", + ); - final auth = ref.watch(authenticationProvider); - final mediaQuery = MediaQuery.of(context); - - final searchTrack = ref.watch(searchProvider(SearchType.track)); - final searchAlbum = ref.watch(searchProvider(SearchType.album)); - final searchPlaylist = ref.watch(searchProvider(SearchType.playlist)); - final searchArtist = ref.watch(searchProvider(SearchType.artist)); - - final queries = [searchTrack, searchAlbum, searchPlaylist, searchArtist]; - - final isFetching = queries.every((s) => s.isLoading); + ref.listen( + metadataPluginSearchChipsProvider, + (previous, next) { + selectedChip.value = next.asData?.value.first ?? "all"; + }, + ); useEffect(() { controller.text = searchTerm; @@ -55,210 +57,189 @@ class SearchPage extends HookConsumerWidget { return null; }, []); - final resultWidget = HookBuilder( - builder: (context) { - final controller = useScrollController(); + void onSubmitted(String value) { + ref.read(searchTermStateProvider.notifier).state = value; + if (value.trim().isEmpty) { + return; + } + KVStoreService.setRecentSearches( + { + value, + ...KVStoreService.recentSearches, + }.toList(), + ); + } - return InterScrollbar( - controller: controller, - child: SingleChildScrollView( - controller: controller, - child: const Padding( - padding: EdgeInsets.symmetric(vertical: 8), - child: SafeArea( - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - SearchTracksSection(), - SearchPlaylistsSection(), - Gap(20), - SearchArtistsSection(), - Gap(20), - SearchAlbumsSection(), - ], - ), - ), - ), - ), - ); + return PopScope( + canPop: false, + onPopInvokedWithResult: (didPop, result) { + context.navigateTo(const HomeRoute()); }, - ); + child: SafeArea( + bottom: false, + child: Scaffold( + headers: [ + if (kTitlebarVisible) + const TitleBar(automaticallyImplyLeading: false, height: 30) + ], + child: Builder(builder: (context) { + if (searchChipSnapshot.error + case MetadataPluginException( + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + message: _ + )) { + return const NoDefaultMetadataPlugin(); + } - return SafeArea( - bottom: false, - child: Scaffold( - appBar: kIsDesktop && !kIsMacOS - ? const PageWindowTitleBar(automaticallyImplyLeading: true) - : null, - body: auth.asData?.value == null - ? const AnonymousFallback() - : Column( - children: [ - Row( - crossAxisAlignment: CrossAxisAlignment.center, - children: [ - if ((kIsMobile || kIsMacOS) && context.canPop()) - const BackButton() - else - const Gap(20), - Expanded( - child: Padding( - padding: const EdgeInsets.only( - right: 20, - top: 20, - bottom: 20, - ), - child: SearchAnchor( - searchController: controller, - viewBuilder: (_) => HookBuilder(builder: (context) { - final searchController = - useListenable(controller); - final update = useForceUpdate(); - final suggestions = searchController.text.isEmpty + if (searchChipSnapshot.hasError) { + return ErrorBox( + error: searchChipSnapshot.error!, + onRetry: () { + ref.invalidate(metadataPluginSearchChipsProvider); + }, + ); + } + + return Column( + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Expanded( + child: Padding( + padding: const EdgeInsets.symmetric( + horizontal: 20, + vertical: 10, + ), + child: ListenableBuilder( + listenable: controller, + builder: (context, _) { + final suggestions = controller.text.isEmpty ? KVStoreService.recentSearches : KVStoreService.recentSearches .where( (s) => weightedRatio( s.toLowerCase(), - searchController.text - .toLowerCase(), + controller.text.toLowerCase(), ) > 50, ) .toList(); - return ListView.builder( - itemCount: suggestions.length, - itemBuilder: (context, index) { - final suggestion = suggestions[index]; + return KeyboardListener( + focusNode: focusNode, + autofocus: true, + onKeyEvent: (value) { + final isEnter = value.logicalKey == + LogicalKeyboardKey.enter; - return ListTile( - leading: const Icon(SpotubeIcons.history), - title: Text(suggestion), - trailing: IconButton( - icon: const Icon(SpotubeIcons.trash), - onPressed: () { - KVStoreService.setRecentSearches( - KVStoreService.recentSearches - .where((s) => s != suggestion) - .toList(), - ); - update(); - }, - ), - onTap: () { - controller.closeView(suggestion); - ref - .read( - searchTermStateProvider.notifier) - .state = suggestion; - }, - ); + if (isEnter) { + onSubmitted(controller.text); + focusNode.unfocus(); + } }, + child: AutoComplete( + suggestions: suggestions.length <= 2 + ? [ + ...suggestions, + "Twenty One Pilots", + "Linkin Park", + "d4vd" + ] + : suggestions, + completer: (suggestion) => suggestion, + mode: AutoCompleteMode.replaceAll, + child: TextField( + autofocus: true, + controller: controller, + features: [ + const InputFeature.leading( + Icon(SpotubeIcons.search), + ), + InputFeature.trailing( + AnimatedCrossFade( + duration: + const Duration(milliseconds: 300), + crossFadeState: + controller.text.isNotEmpty + ? CrossFadeState.showFirst + : CrossFadeState.showSecond, + firstChild: IconButton.ghost( + size: ButtonSize.small, + icon: + const Icon(SpotubeIcons.close), + onPressed: () { + controller.clear(); + }, + ), + secondChild: const SizedBox.square( + dimension: 28), + ), + ) + ], + textInputAction: TextInputAction.search, + placeholder: Text(context.l10n.search), + onSubmitted: onSubmitted, + ), + ), ); }), - suggestionsBuilder: (context, controller) { - return []; - }, - viewOnSubmitted: (value) async { - controller.closeView(value); - Timer( - const Duration(milliseconds: 50), - () { - ref - .read(searchTermStateProvider.notifier) - .state = value; - if (value.trim().isEmpty) { - return; - } - KVStoreService.setRecentSearches( - { - value, - ...KVStoreService.recentSearches, - }.toList(), - ); - }, - ); - }, - builder: (context, controller) { - return SearchBar( - autoFocus: queries.none((s) => - s.asData?.value != null && - !s.hasError) && - !kIsMobile, - controller: controller, - leading: const Icon(SpotubeIcons.search), - hintText: "${context.l10n.search}...", - onTap: controller.openView, - onChanged: (_) => controller.openView(), - ); - }, - ), - ), ), - ], - ), - Expanded( - child: AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: searchTerm.isEmpty - ? Column( - children: [ - SizedBox( - height: mediaQuery.size.height * 0.2, - ), - Icon( - SpotubeIcons.web, - size: 120, - color: theme.colorScheme.onSurface - .withOpacity(0.7), - ), - const SizedBox(height: 20), - Text( - context.l10n.search_to_get_results, - style: theme.textTheme.titleLarge?.copyWith( - fontWeight: FontWeight.w900, - color: theme.colorScheme.onSurface - .withOpacity(0.5), - ), - ), - ], - ) - : isFetching - ? Container( - constraints: BoxConstraints( - maxWidth: mediaQuery.lgAndUp - ? mediaQuery.size.width * 0.5 - : mediaQuery.size.width, - ), - padding: const EdgeInsets.symmetric( - horizontal: 20, - ), - child: Column( - mainAxisAlignment: MainAxisAlignment.center, - crossAxisAlignment: - CrossAxisAlignment.center, - children: [ - Text( - context.l10n.crunching_results, - style: TextStyle( - fontSize: 20, - fontWeight: FontWeight.w900, - color: theme.colorScheme.onSurface - .withOpacity(0.7), - ), - ), - const SizedBox(height: 20), - const LinearProgressIndicator(), - ], - ), - ) - : resultWidget, ), + ], + ), + Row( + spacing: 8, + children: [ + const Gap(12), + if (searchChipSnapshot.asData?.value != null) + for (final chip in searchChipSnapshot.asData!.value) + Chip( + style: selectedChip.value == chip + ? ButtonVariance.primary.copyWith( + decoration: (context, states, value) { + return ButtonVariance.primary + .decoration(context, states) + .copyWithIfBoxDecoration( + borderRadius: + BorderRadius.circular(100), + ); + }, + ) + : ButtonVariance.secondary.copyWith( + decoration: (context, states, value) { + return ButtonVariance.secondary + .decoration(context, states) + .copyWithIfBoxDecoration( + borderRadius: + BorderRadius.circular(100), + ); + }, + ), + child: Text(chip.capitalize()), + onPressed: () { + selectedChip.value = chip; + }, + ), + ], + ), + Expanded( + child: AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + child: switch (selectedChip.value) { + "tracks" => const SearchPageTracksTab(), + "albums" => const SearchPageAlbumsTab(), + "artists" => const SearchPageArtistsTab(), + "playlists" => const SearchPagePlaylistsTab(), + _ => const SearchPageAllTab(), + }, ), - ], - ), + ), + ], + ); + }), + ), ), ); } diff --git a/lib/pages/search/sections/albums.dart b/lib/pages/search/sections/albums.dart deleted file mode 100644 index 857eb59c..00000000 --- a/lib/pages/search/sections/albums.dart +++ /dev/null @@ -1,38 +0,0 @@ -import 'package:flutter/material.dart' hide Page; -import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; - -import 'package:spotify/spotify.dart'; -import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; -import 'package:spotube/extensions/album_simple.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class SearchAlbumsSection extends HookConsumerWidget { - const SearchAlbumsSection({ - super.key, - }); - - @override - Widget build(BuildContext context, ref) { - final query = ref.watch(searchProvider(SearchType.album)); - final notifier = ref.watch(searchProvider(SearchType.album).notifier); - final albums = useMemoized( - () => - query.asData?.value.items - .cast() - .map((e) => e.toAlbum()) - .toList() ?? - [], - [query.asData?.value], - ); - - return HorizontalPlaybuttonCardView( - isLoadingNextPage: query.isLoadingNextPage, - hasNextPage: query.asData?.value.hasMore == true, - items: albums, - onFetchMore: notifier.fetchMore, - title: Text(context.l10n.albums), - ); - } -} diff --git a/lib/pages/search/sections/artists.dart b/lib/pages/search/sections/artists.dart deleted file mode 100644 index 16295580..00000000 --- a/lib/pages/search/sections/artists.dart +++ /dev/null @@ -1,29 +0,0 @@ -import 'package:flutter/material.dart' hide Page; -import 'package:hooks_riverpod/hooks_riverpod.dart'; - -import 'package:spotify/spotify.dart'; -import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class SearchArtistsSection extends HookConsumerWidget { - const SearchArtistsSection({ - super.key, - }); - - @override - Widget build(BuildContext context, ref) { - final query = ref.watch(searchProvider(SearchType.artist)); - final notifier = ref.watch(searchProvider(SearchType.artist).notifier); - - final artists = query.asData?.value.items.cast() ?? []; - - return HorizontalPlaybuttonCardView( - isLoadingNextPage: query.isLoadingNextPage, - hasNextPage: query.asData?.value.hasMore == true, - items: artists, - onFetchMore: notifier.fetchMore, - title: Text(context.l10n.artists), - ); - } -} diff --git a/lib/pages/search/sections/playlists.dart b/lib/pages/search/sections/playlists.dart deleted file mode 100644 index 3799f9fa..00000000 --- a/lib/pages/search/sections/playlists.dart +++ /dev/null @@ -1,29 +0,0 @@ -import 'package:flutter/material.dart' hide Page; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; - -class SearchPlaylistsSection extends HookConsumerWidget { - const SearchPlaylistsSection({ - super.key, - }); - - @override - Widget build(BuildContext context, ref) { - final playlistsQuery = ref.watch(searchProvider(SearchType.playlist)); - final playlistsQueryNotifier = - ref.watch(searchProvider(SearchType.playlist).notifier); - final playlists = - playlistsQuery.asData?.value.items.cast() ?? []; - - return HorizontalPlaybuttonCardView( - isLoadingNextPage: playlistsQuery.isLoadingNextPage, - hasNextPage: playlistsQuery.asData?.value.hasMore == true, - items: playlists, - onFetchMore: playlistsQueryNotifier.fetchMore, - title: Text(context.l10n.playlists), - ); - } -} diff --git a/lib/pages/search/tabs/albums.dart b/lib/pages/search/tabs/albums.dart new file mode 100644 index 00000000..e27772c6 --- /dev/null +++ b/lib/pages/search/tabs/albums.dart @@ -0,0 +1,58 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/fake.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/playbutton_view/playbutton_view.dart'; +import 'package:spotube/modules/album/album_card.dart'; +import 'package:spotube/modules/search/loading.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/metadata_plugin/search/albums.dart'; + +class SearchPageAlbumsTab extends HookConsumerWidget { + const SearchPageAlbumsTab({super.key}); + + @override + Widget build(BuildContext context, ref) { + final controller = useScrollController(); + + final searchTerm = ref.watch(searchTermStateProvider); + final searchAlbumsSnapshot = + ref.watch(metadataPluginSearchAlbumsProvider(searchTerm)); + final searchAlbumsNotifier = + ref.read(metadataPluginSearchAlbumsProvider(searchTerm).notifier); + final searchAlbums = + searchAlbumsSnapshot.asData?.value.items ?? [FakeData.albumSimple]; + + if (searchAlbumsSnapshot.hasError) { + return ErrorBox( + error: searchAlbumsSnapshot.error!, + onRetry: () { + ref.invalidate(metadataPluginSearchAlbumsProvider(searchTerm)); + }, + ); + } + + return SearchPlaceholder( + snapshot: searchAlbumsSnapshot, + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: CustomScrollView( + slivers: [ + PlaybuttonView( + controller: controller, + itemCount: searchAlbums.length, + hasMore: searchAlbumsSnapshot.asData?.value.hasMore == true, + isLoading: searchAlbumsSnapshot.isLoading, + onRequestMore: searchAlbumsNotifier.fetchMore, + gridItemBuilder: (context, index) => + AlbumCard(searchAlbums[index]), + listItemBuilder: (context, index) => + AlbumCard.tile(searchAlbums[index]), + ), + ], + ), + ), + ); + } +} diff --git a/lib/pages/search/tabs/all.dart b/lib/pages/search/tabs/all.dart new file mode 100644 index 00000000..306bdfce --- /dev/null +++ b/lib/pages/search/tabs/all.dart @@ -0,0 +1,58 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; +import 'package:spotube/modules/search/loading.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/modules/search/sections/albums.dart'; +import 'package:spotube/modules/search/sections/artists.dart'; +import 'package:spotube/modules/search/sections/playlists.dart'; +import 'package:spotube/modules/search/sections/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/search/all.dart'; + +class SearchPageAllTab extends HookConsumerWidget { + const SearchPageAllTab({super.key}); + + @override + Widget build(BuildContext context, ref) { + final scrollController = ScrollController(); + final searchTerm = ref.watch(searchTermStateProvider); + final searchSnapshot = + ref.watch(metadataPluginSearchAllProvider(searchTerm)); + + if (searchSnapshot.hasError) { + return ErrorBox( + error: searchSnapshot.error!, + onRetry: () { + ref.invalidate(metadataPluginSearchAllProvider(searchTerm)); + }, + ); + } + + return SearchPlaceholder( + snapshot: searchSnapshot, + child: InterScrollbar( + controller: scrollController, + child: SingleChildScrollView( + controller: scrollController, + child: const Padding( + padding: EdgeInsets.symmetric(vertical: 8), + child: SafeArea( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + SearchTracksSection(), + SearchPlaylistsSection(), + Gap(20), + SearchArtistsSection(), + Gap(20), + SearchAlbumsSection(), + ], + ), + ), + ), + ), + ), + ); + } +} diff --git a/lib/pages/search/tabs/artists.dart b/lib/pages/search/tabs/artists.dart new file mode 100644 index 00000000..8cea7b58 --- /dev/null +++ b/lib/pages/search/tabs/artists.dart @@ -0,0 +1,104 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/fake.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/waypoint.dart'; +import 'package:spotube/extensions/constrains.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/artist/artist_card.dart'; +import 'package:spotube/modules/search/loading.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/metadata_plugin/search/artists.dart'; + +class SearchPageArtistsTab extends HookConsumerWidget { + const SearchPageArtistsTab({super.key}); + + @override + Widget build(BuildContext context, ref) { + final controller = useScrollController(); + + final searchTerm = ref.watch(searchTermStateProvider); + final searchArtistsSnapshot = + ref.watch(metadataPluginSearchArtistsProvider(searchTerm)); + final searchArtistsNotifier = + ref.read(metadataPluginSearchArtistsProvider(searchTerm).notifier); + final searchArtists = searchArtistsSnapshot.asData?.value.items ?? []; + + if (searchArtistsSnapshot.hasError) { + return ErrorBox( + error: searchArtistsSnapshot.error!, + onRetry: () { + ref.invalidate(metadataPluginSearchArtistsProvider(searchTerm)); + }, + ); + } + + return SearchPlaceholder( + snapshot: searchArtistsSnapshot, + child: AnimatedSwitcher( + duration: const Duration(milliseconds: 300), + child: LayoutBuilder(builder: (context, constrains) { + if (searchArtistsSnapshot.hasValue && searchArtists.isEmpty) { + return Padding( + padding: const EdgeInsets.symmetric(horizontal: 8), + child: Column( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Undraw( + height: 200 * context.theme.scaling, + illustration: UndrawIllustration.taken, + color: Theme.of(context).colorScheme.primary, + ), + Text( + context.l10n.nothing_found, + textAlign: TextAlign.center, + ).muted().small() + ], + ), + ); + } + + return GridView.builder( + padding: const EdgeInsets.all(16), + itemCount: searchArtists.length + 1, + gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent( + maxCrossAxisExtent: 200, + mainAxisExtent: constrains.smAndDown ? 225 : 250, + crossAxisSpacing: 8, + mainAxisSpacing: 8, + ), + itemBuilder: (context, index) { + if (searchArtists.isNotEmpty && index == searchArtists.length) { + if (searchArtistsSnapshot.asData?.value.hasMore != true) { + return const SizedBox.shrink(); + } + + return Waypoint( + controller: controller, + isGrid: true, + onTouchEdge: searchArtistsNotifier.fetchMore, + child: Skeletonizer( + enabled: true, + child: ArtistCard(FakeData.artist), + ), + ); + } + + return Skeletonizer( + enabled: searchArtistsSnapshot.isLoading, + child: ArtistCard( + searchArtists.elementAtOrNull(index) ?? FakeData.artist, + ), + ); + }, + ); + }), + ), + ); + } +} diff --git a/lib/pages/search/tabs/playlists.dart b/lib/pages/search/tabs/playlists.dart new file mode 100644 index 00000000..f00153cb --- /dev/null +++ b/lib/pages/search/tabs/playlists.dart @@ -0,0 +1,58 @@ +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/fake.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/playbutton_view/playbutton_view.dart'; +import 'package:spotube/modules/playlist/playlist_card.dart'; +import 'package:spotube/modules/search/loading.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/metadata_plugin/search/playlists.dart'; + +class SearchPagePlaylistsTab extends HookConsumerWidget { + const SearchPagePlaylistsTab({super.key}); + + @override + Widget build(BuildContext context, ref) { + final controller = useScrollController(); + + final searchTerm = ref.watch(searchTermStateProvider); + final searchPlaylistsSnapshot = + ref.watch(metadataPluginSearchPlaylistsProvider(searchTerm)); + final searchPlaylistsNotifier = + ref.read(metadataPluginSearchPlaylistsProvider(searchTerm).notifier); + final searchPlaylists = searchPlaylistsSnapshot.asData?.value.items ?? + [FakeData.playlistSimple]; + + if (searchPlaylistsSnapshot.hasError) { + return ErrorBox( + error: searchPlaylistsSnapshot.error!, + onRetry: () { + ref.invalidate(metadataPluginSearchPlaylistsProvider(searchTerm)); + }, + ); + } + + return SearchPlaceholder( + snapshot: searchPlaylistsSnapshot, + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: CustomScrollView( + slivers: [ + PlaybuttonView( + controller: controller, + itemCount: searchPlaylists.length, + hasMore: searchPlaylistsSnapshot.asData?.value.hasMore == true, + isLoading: searchPlaylistsSnapshot.isLoading, + onRequestMore: searchPlaylistsNotifier.fetchMore, + gridItemBuilder: (context, index) => + PlaylistCard(searchPlaylists[index]), + listItemBuilder: (context, index) => + PlaylistCard.tile(searchPlaylists[index]), + ), + ], + ), + ), + ); + } +} diff --git a/lib/pages/search/tabs/tracks.dart b/lib/pages/search/tabs/tracks.dart new file mode 100644 index 00000000..e4c56891 --- /dev/null +++ b/lib/pages/search/tabs/tracks.dart @@ -0,0 +1,129 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/fake.dart'; +import 'package:spotube/components/dialogs/prompt_dialog.dart'; +import 'package:spotube/components/dialogs/select_device_dialog.dart'; +import 'package:spotube/components/fallbacks/error_box.dart'; +import 'package:spotube/components/track_tile/track_tile.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/modules/search/loading.dart'; +import 'package:spotube/pages/search/search.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/connect/connect.dart'; +import 'package:spotube/provider/metadata_plugin/search/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:very_good_infinite_list/very_good_infinite_list.dart'; + +class SearchPageTracksTab extends HookConsumerWidget { + const SearchPageTracksTab({super.key}); + + @override + Widget build(BuildContext context, ref) { + final searchTerm = ref.watch(searchTermStateProvider); + final searchTracksSnapshot = + ref.watch(metadataPluginSearchTracksProvider(searchTerm)); + final searchTracksNotifier = + ref.read(metadataPluginSearchTracksProvider(searchTerm).notifier); + final searchTracks = + searchTracksSnapshot.asData?.value.items ?? [FakeData.track]; + + final playlist = ref.watch(audioPlayerProvider); + final playlistNotifier = ref.watch(audioPlayerProvider.notifier); + + if (searchTracksSnapshot.hasError) { + return ErrorBox( + error: searchTracksSnapshot.error!, + onRetry: () { + ref.invalidate(metadataPluginSearchTracksProvider(searchTerm)); + }, + ); + } + + return SearchPlaceholder( + snapshot: searchTracksSnapshot, + child: InfiniteList( + itemCount: searchTracksSnapshot.asData?.value.items.length ?? 0, + hasReachedMax: searchTracksSnapshot.asData?.value.hasMore != true, + isLoading: searchTracksSnapshot.isLoading && + !searchTracksSnapshot.isLoadingNextPage, + loadingBuilder: (context) { + return Skeletonizer( + enabled: true, + child: TrackTile(track: FakeData.track, playlist: playlist), + ); + }, + onFetchData: () { + searchTracksNotifier.fetchMore(); + }, + itemBuilder: (context, index) { + final track = searchTracks[index]; + + return TrackTile( + track: track, + playlist: playlist, + index: index, + onTap: () async { + final isRemoteDevice = await showSelectDeviceDialog(context, ref); + + if (isRemoteDevice == null) return; + + if (isRemoteDevice) { + final remotePlayback = ref.read(connectProvider.notifier); + final remotePlaylist = ref.read(queueProvider); + + final isTrackPlaying = + remotePlaylist.activeTrack?.id == track.id; + + if (!isTrackPlaying && context.mounted) { + final shouldPlay = (playlist.tracks.length) > 20 + ? await showPromptDialog( + context: context, + title: context.l10n.playing_track( + track.name, + ), + message: context.l10n.queue_clear_alert( + playlist.tracks.length, + ), + ) + : true; + + if (shouldPlay) { + await remotePlayback.load( + WebSocketLoadEventData.playlist( + tracks: [track], + ), + ); + } + } + } else { + final isTrackPlaying = playlist.activeTrack?.id == track.id; + if (!isTrackPlaying && context.mounted) { + final shouldPlay = (playlist.tracks.length) > 20 + ? await showPromptDialog( + context: context, + title: context.l10n.playing_track( + track.name, + ), + message: context.l10n.queue_clear_alert( + playlist.tracks.length, + ), + ) + : true; + + if (shouldPlay) { + await playlistNotifier.load( + [track], + autoPlay: true, + ); + } + } + } + }, + ); + }, + ), + ); + } +} diff --git a/lib/pages/settings/about.dart b/lib/pages/settings/about.dart index 1357c52f..5a95c0eb 100644 --- a/lib/pages/settings/about.dart +++ b/lib/pages/settings/about.dart @@ -1,7 +1,8 @@ -import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/assets.gen.dart'; import 'package:spotube/collections/env.dart'; +import 'package:spotube/components/button/back_button.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/links/hyper_link.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; @@ -10,15 +11,17 @@ import 'package:spotube/hooks/controllers/use_package_info.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:url_launcher/url_launcher.dart'; +import 'package:auto_route/auto_route.dart'; final _licenseProvider = FutureProvider((ref) async { return await rootBundle.loadString("LICENSE"); }); -class AboutSpotube extends HookConsumerWidget { +@RoutePage() +class AboutSpotubePage extends HookConsumerWidget { static const name = "about"; - const AboutSpotube({super.key}); + const AboutSpotubePage({super.key}); @override Widget build(BuildContext context, ref) { @@ -26,154 +29,180 @@ class AboutSpotube extends HookConsumerWidget { final license = ref.watch(_licenseProvider); final theme = Theme.of(context); - const colon = Text(":"); + const colon = TableCell(child: Text(":")); - return Scaffold( - appBar: PageWindowTitleBar( - leading: const BackButton(), - title: Text(context.l10n.about_spotube), - ), - body: SingleChildScrollView( - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 16.0), - child: Column( - children: [ - Assets.spotubeLogoPng.image( - height: 200, - width: 200, - ), - Center( - child: Column( - children: [ - Text( - context.l10n.spotube_description, - style: theme.textTheme.titleLarge, + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + leading: const [BackButton()], + title: Text(context.l10n.about_spotube), + ) + ], + child: SingleChildScrollView( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16.0), + child: Column( + children: [ + Assets.branding.spotubeLogoPng.image( + height: 200, + width: 200, + ), + Center( + child: Column( + children: [ + Text(context.l10n.spotube_description).semiBold().large(), + const SizedBox(height: 20), + Table( + columnWidths: const { + 0: FixedTableSize(95), + 1: FixedTableSize(10), + 2: IntrinsicTableSize(), + }, + defaultRowHeight: const FixedTableSize(40), + rows: [ + TableRow( + cells: [ + TableCell(child: Text(context.l10n.founder)), + colon, + TableCell( + child: Hyperlink( + context.l10n.kingkor_roy_tirtho, + "https://github.com/KRTirtho", + ), + ) + ], + ), + TableRow( + cells: [ + TableCell(child: Text(context.l10n.version)), + colon, + TableCell(child: Text("v${packageInfo.version}")) + ], + ), + TableRow( + cells: [ + TableCell(child: Text(context.l10n.channel)), + colon, + TableCell(child: Text(Env.releaseChannel.name)) + ], + ), + TableRow( + cells: [ + TableCell(child: Text(context.l10n.build_number)), + colon, + TableCell( + child: Text(packageInfo.buildNumber + .replaceAll(".", " ")), + ) + ], + ), + const TableRow( + cells: [ + TableCell(child: Text("Website")), + colon, + TableCell( + child: Hyperlink( + "spotube.krtirtho.dev", + "https://spotube.krtirtho.dev", + ), + ), + ], + ), + TableRow( + cells: [ + TableCell(child: Text(context.l10n.repository)), + colon, + const TableCell( + child: Hyperlink( + "github.com/KRTirtho/spotube", + "https://github.com/KRTirtho/spotube", + ), + ), + ], + ), + TableRow( + cells: [ + TableCell(child: Text(context.l10n.license)), + colon, + const TableCell( + child: Hyperlink( + "BSD-4-Clause", + "https://raw.githubusercontent.com/KRTirtho/spotube/master/LICENSE", + ), + ), + ], + ), + TableRow( + cells: [ + TableCell(child: Text(context.l10n.bug_issues)), + colon, + const TableCell( + child: Hyperlink( + "Discord#chat", + "https://discord.gg/uJ94vxB6vg", + ), + ), + ], + ), + ], + ), + ], + ), + ), + const SizedBox(height: 20), + MouseRegion( + cursor: SystemMouseCursors.click, + child: GestureDetector( + onTap: () => launchUrl( + Uri.parse("https://discord.gg/uJ94vxB6vg"), + mode: LaunchMode.externalApplication, ), - const SizedBox(height: 20), - Table( - columnWidths: const { - 0: FixedColumnWidth(95), - 1: FixedColumnWidth(10), - 2: IntrinsicColumnWidth(), + child: const UniversalImage( + path: + "https://discord.com/api/guilds/1012234096237350943/widget.png?style=banner2", + ), + ), + ), + const SizedBox(height: 20), + Text( + context.l10n.made_with, + textAlign: TextAlign.center, + style: theme.typography.small, + ), + Text( + context.l10n.copyright(DateTime.now().year), + textAlign: TextAlign.center, + style: theme.typography.small, + ), + const SizedBox(height: 20), + ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 750), + child: SafeArea( + child: license.when( + data: (data) { + return Text( + data, + style: theme.typography.small, + ); + }, + loading: () { + return const Center( + child: CircularProgressIndicator(), + ); + }, + error: (e, s) { + return Text( + e.toString(), + style: theme.typography.small, + ); }, - children: [ - TableRow( - children: [ - Text(context.l10n.founder), - colon, - Hyperlink( - context.l10n.kingkor_roy_tirtho, - "https://github.com/KRTirtho", - ) - ], - ), - TableRow( - children: [ - Text(context.l10n.version), - colon, - Text("v${packageInfo.version}") - ], - ), - TableRow( - children: [ - Text(context.l10n.channel), - colon, - Text(Env.releaseChannel.name) - ], - ), - TableRow( - children: [ - Text(context.l10n.build_number), - colon, - Text(packageInfo.buildNumber.replaceAll(".", " ")) - ], - ), - TableRow( - children: [ - Text(context.l10n.repository), - colon, - const Hyperlink( - "github.com/KRTirtho/spotube", - "https://github.com/KRTirtho/spotube", - ), - ], - ), - TableRow( - children: [ - Text(context.l10n.license), - colon, - const Hyperlink( - "BSD-4-Clause", - "https://raw.githubusercontent.com/KRTirtho/spotube/master/LICENSE", - ), - ], - ), - TableRow( - children: [ - Text(context.l10n.bug_issues), - colon, - const Hyperlink( - "github.com/KRTirtho/spotube/issues", - "https://github.com/KRTirtho/spotube/issues", - ), - ], - ), - ], ), - ], - ), - ), - const SizedBox(height: 20), - MouseRegion( - cursor: SystemMouseCursors.click, - child: GestureDetector( - onTap: () => launchUrl( - Uri.parse("https://discord.gg/uJ94vxB6vg"), - mode: LaunchMode.externalApplication, - ), - child: const UniversalImage( - path: - "https://discord.com/api/guilds/1012234096237350943/widget.png?style=banner2", ), ), - ), - const SizedBox(height: 20), - Text( - context.l10n.made_with, - textAlign: TextAlign.center, - style: theme.textTheme.bodySmall, - ), - Text( - context.l10n.copyright(DateTime.now().year), - textAlign: TextAlign.center, - style: theme.textTheme.bodySmall, - ), - const SizedBox(height: 20), - ConstrainedBox( - constraints: const BoxConstraints(maxWidth: 750), - child: SafeArea( - child: license.when( - data: (data) { - return Text( - data, - style: theme.textTheme.bodySmall, - ); - }, - loading: () { - return const Center( - child: CircularProgressIndicator(), - ); - }, - error: (e, s) { - return Text( - e.toString(), - style: theme.textTheme.bodySmall, - ); - }, - ), - ), - ), - ], + ], + ), ), ), ), diff --git a/lib/pages/settings/blacklist.dart b/lib/pages/settings/blacklist.dart index 1f018dab..2af899f3 100644 --- a/lib/pages/settings/blacklist.dart +++ b/lib/pages/settings/blacklist.dart @@ -1,15 +1,19 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:collection/collection.dart'; import 'package:fuzzywuzzy/fuzzywuzzy.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/button/back_button.dart'; import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/components/ui/button_tile.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/blacklist_provider.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class BlackListPage extends HookConsumerWidget { static const name = "blacklist"; @@ -43,50 +47,52 @@ class BlackListPage extends HookConsumerWidget { [blacklist, searchText.value], ); - return Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.blacklist), - centerTitle: true, - leading: const BackButton(), - ), - body: Column( - mainAxisSize: MainAxisSize.min, - children: [ - Padding( - padding: const EdgeInsets.all(8.0), - child: TextField( - onChanged: (value) => searchText.value = value, - decoration: InputDecoration( - hintText: context.l10n.search, - prefixIcon: const Icon(SpotubeIcons.search), + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.blacklist), + leading: const [BackButton()], + ) + ], + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + Padding( + padding: const EdgeInsets.all(8.0), + child: TextField( + onChanged: (value) => searchText.value = value, + placeholder: Text(context.l10n.search), + // prefixIcon: const Icon(SpotubeIcons.search), ), ), - ), - InterScrollbar( - controller: controller, - child: ListView.builder( + InterScrollbar( controller: controller, - shrinkWrap: true, - itemCount: filteredBlacklist.length, - itemBuilder: (context, index) { - final item = filteredBlacklist.elementAt(index); - return ListTile( - leading: Text("${index + 1}."), - title: Text("${item.name} (${item.elementType.name})"), - subtitle: Text(item.elementId), - trailing: IconButton( - icon: Icon(SpotubeIcons.trash, color: Colors.red[400]), - onPressed: () { - ref - .read(blacklistProvider.notifier) - .remove(filteredBlacklist.elementAt(index).elementId); - }, - ), - ); - }, + child: ListView.builder( + controller: controller, + shrinkWrap: true, + itemCount: filteredBlacklist.length, + itemBuilder: (context, index) { + final item = filteredBlacklist.elementAt(index); + return ButtonTile( + style: ButtonVariance.ghost, + leading: Text("${index + 1}."), + title: Text("${item.name} (${item.elementType.name})"), + subtitle: Text(item.elementId), + trailing: IconButton.ghost( + icon: Icon(SpotubeIcons.trash, color: Colors.red[400]), + onPressed: () { + ref.read(blacklistProvider.notifier).remove( + filteredBlacklist.elementAt(index).elementId); + }, + ), + ); + }, + ), ), - ), - ], + ], + ), ), ); } diff --git a/lib/pages/settings/logs.dart b/lib/pages/settings/logs.dart index 6ccbe32f..61269456 100644 --- a/lib/pages/settings/logs.dart +++ b/lib/pages/settings/logs.dart @@ -1,14 +1,19 @@ -import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:flutter_undraw/flutter_undraw.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/button/back_button.dart'; import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/logs/logs_provider.dart'; import 'package:spotube/services/logger/logger.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class LogsPage extends HookConsumerWidget { static const name = "logs"; @@ -21,54 +26,76 @@ class LogsPage extends HookConsumerWidget { final logsQuery = ref.watch(logsProvider); return Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.logs), - leading: const BackButton(), - actions: [ - IconButton( - icon: const Icon(SpotubeIcons.clipboard), - iconSize: 16, - onPressed: () async { - final logsSnapshot = await ref.read(logsProvider.future); + headers: [ + SafeArea( + bottom: false, + child: TitleBar( + title: Text(context.l10n.logs), + leading: const [BackButton()], + trailing: [ + IconButton.ghost( + icon: const Icon(SpotubeIcons.clipboard, size: 16), + onPressed: () async { + final logsSnapshot = await ref.read(logsProvider.future); - await Clipboard.setData(ClipboardData(text: logsSnapshot)); - if (context.mounted) { - ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - content: Text(context.l10n.copied_to_clipboard("")), - ), - ); - } - }, - ), - IconButton( - icon: const Icon(SpotubeIcons.trash), - iconSize: 16, - onPressed: () async { - ref.invalidate(logsProvider); - - final logsFile = await AppLogger.getLogsPath(); - - await logsFile.writeAsString(""); - }, - ) - ], - ), - body: SafeArea( - child: switch (logsQuery) { - AsyncData(:final value) => Card( - child: InterScrollbar( - controller: controller, - child: Padding( - padding: const EdgeInsets.all(8.0), - child: SingleChildScrollView( - controller: controller, - child: Text(value), - ), + await Clipboard.setData(ClipboardData(text: logsSnapshot)); + if (context.mounted) { + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + title: Text(context.l10n.copied_to_clipboard("")), + ), + ); + }, + ); + } + }, + ), + IconButton.ghost( + icon: const Icon( + SpotubeIcons.trash, + size: 16, ), + onPressed: () async { + ref.invalidate(logsProvider); + + final logsFile = await AppLogger.getLogsPath(); + + await logsFile.writeAsString(""); + }, + ) + ], + ), + ) + ], + child: SafeArea( + child: switch (logsQuery) { + AsyncData(:final value) => InterScrollbar( + controller: controller, + child: SingleChildScrollView( + padding: const EdgeInsets.all(8.0), + controller: controller, + child: Card(child: SelectableText(value)), ), ), - AsyncError(:final error) => Center(child: Text(error.toString())), + AsyncError(:final error) => switch (error) { + StateError() => Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Undraw( + illustration: UndrawIllustration.noData, + height: 200 * context.theme.scaling, + width: 200 * context.theme.scaling, + color: context.theme.colorScheme.primary, + ), + Text(context.l10n.no_logs_found).muted().small(), + ], + ), + _ => Center(child: Text(error.toString())), + }, _ => const Center(child: CircularProgressIndicator()), }, ), diff --git a/lib/pages/settings/metadata/metadata_form.dart b/lib/pages/settings/metadata/metadata_form.dart new file mode 100644 index 00000000..b0aeb8bb --- /dev/null +++ b/lib/pages/settings/metadata/metadata_form.dart @@ -0,0 +1,146 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:form_builder_validators/form_builder_validators.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/components/markdown/markdown.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +@RoutePage() +class SettingsMetadataProviderFormPage extends HookConsumerWidget { + final String title; + final List fields; + const SettingsMetadataProviderFormPage({ + super.key, + required this.title, + required this.fields, + }); + + @override + Widget build(BuildContext context, ref) { + final formKey = useMemoized(() => GlobalKey(), []); + + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(title), + ), + ], + child: FormBuilder( + key: formKey, + child: Center( + child: Container( + padding: const EdgeInsets.all(16), + constraints: const BoxConstraints(maxWidth: 600), + child: CustomScrollView( + shrinkWrap: true, + slivers: [ + SliverToBoxAdapter( + child: Text( + title, + textAlign: TextAlign.center, + style: context.theme.typography.h2, + ), + ), + const SliverGap(24), + SliverList.separated( + itemCount: fields.length, + separatorBuilder: (context, index) => const Gap(12), + itemBuilder: (context, index) { + if (fields[index] is MetadataFormFieldTextObject) { + final field = + fields[index] as MetadataFormFieldTextObject; + return AppMarkdown(data: field.text); + } + + final field = + fields[index] as MetadataFormFieldInputObject; + return FormBuilderField( + name: field.id, + initialValue: field.defaultValue, + validator: FormBuilderValidators.compose([ + if (field.required == true) + FormBuilderValidators.required( + errorText: 'This field is required', + ), + if (field.regex != null) + FormBuilderValidators.match( + RegExp(field.regex!), + errorText: + context.l10n.input_does_not_match_format, + ), + ]), + builder: (formField) { + return Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 4, + children: [ + TextField( + placeholder: field.placeholder == null + ? null + : Text(field.placeholder!), + initialValue: formField.value, + onChanged: (value) { + formField.didChange(value); + }, + obscureText: + field.variant == FormFieldVariant.password, + keyboardType: + field.variant == FormFieldVariant.number + ? TextInputType.number + : TextInputType.text, + features: [ + if (field.variant == + FormFieldVariant.password) + const InputFeature.passwordToggle(), + ], + ), + if (formField.hasError) + Text( + formField.errorText ?? '', + style: const TextStyle( + color: Colors.red, fontSize: 12), + ), + ], + ); + }, + ); + }, + ), + const SliverGap(24), + SliverToBoxAdapter( + child: Button.primary( + onPressed: () { + if (formKey.currentState?.saveAndValidate() != true) { + return; + } + + final data = formKey.currentState!.value.entries + .map((e) => { + "id": e.key, + "value": e.value, + }) + .toList(); + + context.router.maybePop(data); + }, + child: Text(context.l10n.submit), + ), + ), + const SliverGap(200) + ], + ), + ), + ), + ), + ), + ); + } +} diff --git a/lib/pages/settings/metadata_plugins.dart b/lib/pages/settings/metadata_plugins.dart new file mode 100644 index 00000000..d4cb1ecf --- /dev/null +++ b/lib/pages/settings/metadata_plugins.dart @@ -0,0 +1,358 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:collection/collection.dart'; +import 'package:file_selector/file_selector.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_form_builder/flutter_form_builder.dart'; +import 'package:flutter_hooks/flutter_hooks.dart'; +import 'package:form_builder_validators/form_builder_validators.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:skeletonizer/skeletonizer.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/form/text_form_field.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/modules/metadata_plugins/installed_plugin.dart'; +import 'package:spotube/modules/metadata_plugins/plugin_repository.dart'; +import 'package:spotube/provider/metadata_plugin/core/repositories.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:file_picker/file_picker.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/logger/logger.dart'; +import 'package:spotube/utils/platform.dart'; +import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:sliver_tools/sliver_tools.dart'; + +@RoutePage() +class SettingsMetadataProviderPage extends HookConsumerWidget { + const SettingsMetadataProviderPage({super.key}); + + @override + Widget build(BuildContext context, ref) { + final tabState = useState(0); + final formKey = useMemoized(() => GlobalKey(), []); + + final plugins = ref.watch(metadataPluginsProvider); + final pluginsNotifier = ref.watch(metadataPluginsProvider.notifier); + + final pluginReposSnapshot = ref.watch(metadataPluginRepositoriesProvider); + final pluginReposNotifier = + ref.watch(metadataPluginRepositoriesProvider.notifier); + + final pluginRepos = useMemoized( + () { + final installedPluginIds = plugins.asData?.value.plugins + .map((e) => e.repository) + .nonNulls + .toList() ?? + []; + + final pluginRepos = pluginReposSnapshot.asData?.value.items ?? []; + if (installedPluginIds.isEmpty) return pluginRepos; + final availablePlugins = pluginRepos + .whereNot((repo) => installedPluginIds.contains(repo.repoUrl)) + .toList(); + + if (tabState.value != 0) { + // metadata only plugins + return availablePlugins.where( + (d) { + return d.topics.contains( + tabState.value == 1 + ? "spotube-metadata-plugin" + : "spotube-audio-source-plugin", + ); + }, + ).toList(); + } + + return availablePlugins; // all plugins + }, + [ + plugins.asData?.value.plugins, + pluginReposSnapshot.asData?.value, + tabState.value, + ], + ); + + final installedPlugins = useMemoized?>(() { + if (tabState.value == 0) return plugins.asData?.value.plugins; + + return plugins.asData?.value.plugins.where((d) { + return d.abilities.contains( + tabState.value == 1 + ? PluginAbilities.metadata + : PluginAbilities.audioSource, + ); + }).toList(); + }, [tabState.value, plugins.asData?.value]); + + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.plugins), + ) + ], + child: Padding( + padding: const EdgeInsets.all(8), + child: CustomScrollView( + slivers: [ + SliverToBoxAdapter( + child: Row( + spacing: 8, + children: [ + Expanded( + child: FormBuilder( + key: formKey, + child: TextFormBuilderField( + name: "plugin_url", + validator: FormBuilderValidators.url( + protocols: ["http", "https"]), + placeholder: + Text(context.l10n.paste_plugin_download_url), + ), + ), + ), + HookBuilder(builder: (context) { + final isLoading = useState(false); + + return Tooltip( + tooltip: TooltipContainer( + child: Text(context + .l10n.download_and_install_plugin_from_url), + ).call, + child: IconButton.secondary( + icon: isLoading.value + ? const SizedBox.square( + dimension: 22, + child: + CircularProgressIndicator(strokeWidth: 2), + ) + : const Icon(SpotubeIcons.download), + enabled: !isLoading.value, + onPressed: () async { + try { + if (formKey.currentState?.saveAndValidate() ?? + false) { + final url = formKey.currentState + ?.fields["plugin_url"]?.value as String; + + if (url.isNotEmpty) { + isLoading.value = true; + final pluginConfig = await pluginsNotifier + .downloadAndCachePlugin(url); + + await pluginsNotifier.addPlugin(pluginConfig); + + formKey.currentState?.fields["plugin_url"] + ?.reset(); + } + } + } catch (e, stackTrace) { + AppLogger.reportError(e, stackTrace); + if (context.mounted) { + showToast( + showDuration: const Duration(seconds: 5), + context: context, + builder: (context, overlay) { + return SurfaceCard( + child: Basic( + leading: const Icon( + SpotubeIcons.error, + color: Colors.red, + ), + title: Text( + context.l10n + .failed_to_add_plugin_error( + e.toString()), + ), + ), + ); + }, + ); + } + } finally { + isLoading.value = false; + } + }, + ), + ); + }), + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.upload_plugin_from_file), + ).call, + child: IconButton.primary( + icon: const Icon(SpotubeIcons.upload), + onPressed: () async { + Uint8List bytes; + + if (kIsFlatpak) { + final result = await openFile( + acceptedTypeGroups: [ + const XTypeGroup( + label: 'Spotube Metadata Plugin', + extensions: ['smplug'], + ), + ], + ); + if (result == null) return; + bytes = await result.readAsBytes(); + } else { + final result = await FilePicker.platform.pickFiles( + type: kIsAndroid ? FileType.any : FileType.custom, + allowedExtensions: kIsAndroid ? [] : ["smplug"], + withData: true, + ); + + if (result == null) return; + + final file = result.files.first; + if (file.bytes == null) return; + bytes = file.bytes!; + } + + final pluginConfig = + await pluginsNotifier.extractPluginArchive(bytes); + await pluginsNotifier.addPlugin(pluginConfig); + }, + ), + ), + ], + ), + ), + const SliverGap(12), + SliverToBoxAdapter( + child: TabList( + index: tabState.value, + onChanged: (value) { + tabState.value = value; + }, + children: const [ + TabItem(child: Text("All")), + TabItem(child: Text("Metadata")), + TabItem(child: Text("Audio Source")), + ], + ), + ), + const SliverGap(12), + if (plugins.asData?.value.plugins.isNotEmpty ?? false) + SliverToBoxAdapter( + child: Row( + children: [ + const Gap(8), + Text(context.l10n.installed).h4, + const Gap(8), + const Expanded(child: Divider()), + const Gap(8), + ], + ), + ), + const SliverGap(20), + SliverList.separated( + itemCount: installedPlugins?.length ?? 0, + separatorBuilder: (context, index) => const Gap(12), + itemBuilder: (context, index) { + final plugin = installedPlugins![index]; + final isDefaultMetadata = + plugins.asData!.value.defaultMetadataPluginConfig?.slug == + plugin.slug; + final isDefaultAudioSource = plugins + .asData!.value.defaultAudioSourcePluginConfig?.slug == + plugin.slug; + return MetadataInstalledPluginItem( + plugin: plugin, + isDefaultMetadata: isDefaultMetadata, + isDefaultAudioSource: isDefaultAudioSource, + ); + }, + ), + const SliverGap(12), + SliverToBoxAdapter( + child: Row( + children: [ + const Gap(8), + Text(context.l10n.available_plugins).h4, + const Gap(8), + const Expanded(child: Divider()), + const Gap(8), + ], + ), + ), + const SliverGap(12), + SliverInfiniteList( + isLoading: pluginReposSnapshot.isLoading && + !pluginReposSnapshot.isLoadingNextPage, + itemCount: pluginRepos.length, + onFetchData: pluginReposNotifier.fetchMore, + separatorBuilder: (context, index) { + return const Gap(12); + }, + loadingBuilder: (context) { + return Skeletonizer( + enabled: true, + child: MetadataPluginRepositoryItem( + pluginRepo: MetadataPluginRepository( + name: "Loading...", + description: "Loading...", + repoUrl: "", + owner: "", + topics: [], + ), + ), + ); + }, + itemBuilder: (context, index) { + final pluginRepo = pluginRepos[index]; + + return MetadataPluginRepositoryItem( + pluginRepo: pluginRepo, + ); + }, + ), + const SliverGap(20), + SliverCrossAxisConstrained( + maxCrossAxisExtent: 720, + child: SliverFillRemaining( + hasScrollBody: false, + child: Container( + alignment: Alignment.bottomCenter, + margin: const EdgeInsets.only(bottom: 20), + child: SafeArea( + child: Card( + child: Column( + mainAxisSize: MainAxisSize.min, + crossAxisAlignment: CrossAxisAlignment.start, + spacing: 12, + children: [ + Row( + spacing: 8, + children: [ + const Icon(SpotubeIcons.warning, size: 16), + Text( + context.l10n.disclaimer, + style: const TextStyle( + fontWeight: FontWeight.bold), + ).bold, + ], + ), + Text(context.l10n.third_party_plugin_dmca_notice) + .muted + .xSmall, + ], + ), + ), + ), + ), + ), + ), + ], + ), + ), + ), + ); + } +} diff --git a/lib/pages/settings/scrobbling/scrobbling.dart b/lib/pages/settings/scrobbling/scrobbling.dart new file mode 100644 index 00000000..9c7f3296 --- /dev/null +++ b/lib/pages/settings/scrobbling/scrobbling.dart @@ -0,0 +1,67 @@ +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/material.dart' + show ListTile, ListTileTheme, ListTileThemeData, Material, MaterialType; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shadcn_flutter/shadcn_flutter_extension.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/extensions/context.dart'; + +@RoutePage() +class SettingsScrobblingPage extends HookConsumerWidget { + static const name = "settings_scrobbling"; + + const SettingsScrobblingPage({super.key}); + + @override + Widget build(BuildContext context, ref) { + return Material( + type: MaterialType.transparency, + child: ListTileTheme( + data: ListTileThemeData( + contentPadding: EdgeInsets.zero, + minVerticalPadding: 0, + shape: RoundedRectangleBorder( + borderRadius: context.theme.borderRadiusLg, + side: BorderSide( + color: context.theme.colorScheme.border, + width: .5, + ), + ), + textColor: context.theme.colorScheme.foreground, + iconColor: context.theme.colorScheme.foreground, + selectedColor: context.theme.colorScheme.accent, + subtitleTextStyle: context.theme.typography.xSmall, + ), + child: SafeArea( + bottom: false, + child: Scaffold( + headers: [TitleBar(title: Text(context.l10n.scrobbling))], + child: ListView( + padding: const EdgeInsets.all(8), + children: [ + Card( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: ListTile( + leading: const Icon(SpotubeIcons.lastFm, color: Colors.red), + title: Text(context.l10n.login_with_lastfm), + subtitle: Text(context.l10n.scrobble_to_lastfm), + trailing: Button.secondary( + leading: const Icon(SpotubeIcons.lastFm), + onPressed: () { + context.navigateTo(const LastFMLoginRoute()); + }, + child: Text(context.l10n.connect), + ), + ), + ), + ], + ), + ), + ), + ), + ); + } +} diff --git a/lib/pages/settings/sections/about.dart b/lib/pages/settings/sections/about.dart index a0a5bf30..82c98e90 100644 --- a/lib/pages/settings/sections/about.dart +++ b/lib/pages/settings/sections/about.dart @@ -1,8 +1,11 @@ +import 'package:auto_route/auto_route.dart'; import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; -import 'package:go_router/go_router.dart'; +import 'package:flutter/material.dart' show ListTile; + import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide ButtonStyle; import 'package:spotube/collections/env.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/settings/section_card_with_heading.dart'; import 'package:spotube/components/adaptive/adaptive_list_tile.dart'; @@ -42,12 +45,25 @@ class SettingsAboutSection extends HookConsumerWidget { ), ), ), - trailing: (context, update) => FilledButton( - style: ButtonStyle( - backgroundColor: WidgetStatePropertyAll(Colors.red[100]), - foregroundColor: - const WidgetStatePropertyAll(Colors.pinkAccent), - padding: const WidgetStatePropertyAll(EdgeInsets.all(15)), + trailing: (context, update) => Button( + style: ButtonVariance.primary.copyWith( + decoration: (context, states, value) { + final decoration = ButtonVariance.primary + .decoration(context, states) as BoxDecoration; + + if (states.contains(WidgetState.hovered)) { + return decoration.copyWith(color: Colors.pink[400]); + } else if (states.contains(WidgetState.focused)) { + return decoration.copyWith(color: Colors.pink[300]); + } else if (states.isNotEmpty) { + return decoration; + } + + return decoration.copyWith(color: Colors.pink); + }, + textStyle: (context, states, value) => ButtonVariance.primary + .textStyle(context, states) + .copyWith(color: Colors.white), ), onPressed: () { launchUrlString( @@ -55,29 +71,26 @@ class SettingsAboutSection extends HookConsumerWidget { mode: LaunchMode.externalApplication, ); }, - child: Row( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon(SpotubeIcons.heart), - const SizedBox(width: 5), - Text(context.l10n.please_sponsor), - ], - ), + leading: const Icon(SpotubeIcons.heart), + child: Text(context.l10n.please_sponsor), ), ), if (Env.enableUpdateChecker) - SwitchListTile( - secondary: const Icon(SpotubeIcons.update), + ListTile( + leading: const Icon(SpotubeIcons.update), title: Text(context.l10n.check_for_updates), - value: preferences.checkUpdate, - onChanged: (checked) => preferencesNotifier.setCheckUpdate(checked), + trailing: Switch( + value: preferences.checkUpdate, + onChanged: (checked) => + preferencesNotifier.setCheckUpdate(checked), + ), ), ListTile( leading: const Icon(SpotubeIcons.info), title: Text(context.l10n.about_spotube), trailing: const Icon(SpotubeIcons.angleRight), onTap: () { - GoRouter.of(context).push("/settings/about"); + context.navigateTo(const AboutSpotubeRoute()); }, ) ], diff --git a/lib/pages/settings/sections/accounts.dart b/lib/pages/settings/sections/accounts.dart index b9a26147..ca859ada 100644 --- a/lib/pages/settings/sections/accounts.dart +++ b/lib/pages/settings/sections/accounts.dart @@ -1,147 +1,50 @@ -import 'package:auto_size_text/auto_size_text.dart'; -import 'package:flutter/material.dart'; -import 'package:go_router/go_router.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/material.dart' show ListTile; + import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/settings/section_card_with_heading.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/constrains.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/pages/profile/profile.dart'; -import 'package:spotube/pages/mobile_login/hooks/login_callback.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; import 'package:spotube/provider/scrobbler/scrobbler.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; -import 'package:spotube/utils/service_utils.dart'; class SettingsAccountSection extends HookConsumerWidget { const SettingsAccountSection({super.key}); @override Widget build(context, ref) { - final theme = Theme.of(context); - final router = GoRouter.of(context); - - final auth = ref.watch(authenticationProvider); final scrobbler = ref.watch(scrobblerProvider); - final me = ref.watch(meProvider); - final meData = me.asData?.value; - - final logoutBtnStyle = FilledButton.styleFrom( - backgroundColor: Colors.red, - foregroundColor: Colors.white, - ); - - final onLogin = useLoginCallback(ref); return SectionCardWithHeading( heading: context.l10n.account, children: [ - if (auth.asData?.value != null) - ListTile( - leading: const Icon(SpotubeIcons.user), - title: Text(context.l10n.user_profile), - trailing: Padding( - padding: const EdgeInsets.all(8.0), - child: CircleAvatar( - backgroundImage: UniversalImage.imageProvider( - (meData?.images).asUrlString( - placeholder: ImagePlaceholder.artist, - ), - ), - ), - ), - onTap: () { - ServiceUtils.pushNamed(context, ProfilePage.name); - }, - ), - if (auth.asData?.value == null) - LayoutBuilder(builder: (context, constrains) { - return ListTile( - leading: Icon( - SpotubeIcons.spotify, - color: theme.colorScheme.primary, - ), - title: Align( - alignment: Alignment.centerLeft, - child: AutoSizeText( - context.l10n.login_with_spotify, - maxLines: 1, - style: TextStyle( - color: theme.colorScheme.primary, - ), - ), - ), - onTap: constrains.mdAndUp ? null : onLogin, - trailing: constrains.smAndDown - ? null - : FilledButton( - onPressed: onLogin, - style: ButtonStyle( - shape: WidgetStateProperty.all( - RoundedRectangleBorder( - borderRadius: BorderRadius.circular(25.0), - ), - ), - ), - child: Text( - context.l10n.connect_with_spotify.toUpperCase(), - ), - ), - ); - }) - else - Builder(builder: (context) { - return ListTile( - leading: const Icon(SpotubeIcons.spotify), - title: SizedBox( - height: 50, - width: 180, - child: Align( - alignment: Alignment.centerLeft, - child: AutoSizeText( - context.l10n.logout_of_this_account, - maxLines: 1, - ), - ), - ), - trailing: FilledButton( - style: logoutBtnStyle, - onPressed: () async { - ref.read(authenticationProvider.notifier).logout(); - GoRouter.of(context).pop(); - }, - child: Text(context.l10n.logout), - ), - ); - }), + ListTile( + leading: const Icon(SpotubeIcons.extensions), + title: Text(context.l10n.plugins), + subtitle: Text(context.l10n.configure_plugins), + onTap: () { + context.pushRoute(const SettingsMetadataProviderRoute()); + }, + trailing: const Icon(SpotubeIcons.angleRight), + ), if (scrobbler.asData?.value == null) ListTile( - leading: const Icon(SpotubeIcons.lastFm), - title: Text(context.l10n.login_with_lastfm), - subtitle: Text(context.l10n.scrobble_to_lastfm), - trailing: FilledButton.icon( - icon: const Icon(SpotubeIcons.lastFm), - label: Text(context.l10n.connect), - onPressed: () { - router.push("/lastfm-login"); - }, - style: FilledButton.styleFrom( - backgroundColor: const Color.fromARGB(255, 186, 0, 0), - foregroundColor: Colors.white, - ), - ), + leading: const Icon(SpotubeIcons.music), + title: Text(context.l10n.audio_scrobblers), + onTap: () { + context.pushRoute(const SettingsScrobblingRoute()); + }, + trailing: const Icon(SpotubeIcons.angleRight), ) else ListTile( leading: const Icon(SpotubeIcons.lastFm), title: Text(context.l10n.disconnect_lastfm), - trailing: FilledButton( + trailing: Button.destructive( onPressed: () { ref.read(scrobblerProvider.notifier).logout(); }, - style: logoutBtnStyle, child: Text(context.l10n.disconnect), ), ), diff --git a/lib/pages/settings/sections/appearance.dart b/lib/pages/settings/sections/appearance.dart index f97add42..88f39a25 100644 --- a/lib/pages/settings/sections/appearance.dart +++ b/lib/pages/settings/sections/appearance.dart @@ -1,6 +1,6 @@ -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' show ListTile; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/models/database/database.dart'; @@ -41,15 +41,15 @@ class SettingsAppearanceSection extends HookConsumerWidget { } }, options: [ - DropdownMenuItem( + SelectItemButton( value: LayoutMode.adaptive, child: Text(context.l10n.adaptive), ), - DropdownMenuItem( + SelectItemButton( value: LayoutMode.compact, child: Text(context.l10n.compact), ), - DropdownMenuItem( + SelectItemButton( value: LayoutMode.extended, child: Text(context.l10n.extended), ), @@ -60,15 +60,15 @@ class SettingsAppearanceSection extends HookConsumerWidget { title: Text(context.l10n.theme), value: preferences.themeMode, options: [ - DropdownMenuItem( + SelectItemButton( value: ThemeMode.dark, child: Text(context.l10n.dark), ), - DropdownMenuItem( + SelectItemButton( value: ThemeMode.light, child: Text(context.l10n.light), ), - DropdownMenuItem( + SelectItemButton( value: ThemeMode.system, child: Text(context.l10n.system), ), @@ -79,13 +79,14 @@ class SettingsAppearanceSection extends HookConsumerWidget { } }, ), - SwitchListTile( - secondary: const Icon(SpotubeIcons.amoled), - title: Text(context.l10n.use_amoled_mode), - subtitle: Text(context.l10n.pitch_dark_theme), - value: preferences.amoledDarkTheme, - onChanged: preferencesNotifier.setAmoledDarkTheme, - ), + // ListTile( + // leading: const Icon(SpotubeIcons.amoled), + // title: Text(context.l10n.use_amoled_mode), + // subtitle: Text(context.l10n.pitch_dark_theme), + // trailing: Switch( + // value: preferences.amoledDarkTheme, + // onChanged: preferencesNotifier.setAmoledDarkTheme, + // )), ListTile( leading: const Icon(SpotubeIcons.palette), title: Text(context.l10n.accent_color), @@ -93,20 +94,22 @@ class SettingsAppearanceSection extends HookConsumerWidget { horizontal: 15, vertical: 5, ), - trailing: ColorTile.compact( + trailing: ColorChip( color: preferences.accentColorScheme, + name: preferences.accentColorScheme.name, onPressed: pickColorScheme(), - isActive: true, + isActive: false, ), onTap: pickColorScheme(), ), - SwitchListTile( - secondary: const Icon(SpotubeIcons.colorSync), - title: Text(context.l10n.sync_album_color), - subtitle: Text(context.l10n.sync_album_color_description), - value: preferences.albumColorSync, - onChanged: preferencesNotifier.setAlbumColorSync, - ), + // ListTile( + // leading: const Icon(SpotubeIcons.colorSync), + // title: Text(context.l10n.sync_album_color), + // subtitle: Text(context.l10n.sync_album_color_description), + // trailing: Switch( + // value: preferences.albumColorSync, + // onChanged: preferencesNotifier.setAlbumColorSync, + // )), ]; if (isGettingStarted) { diff --git a/lib/pages/settings/sections/desktop.dart b/lib/pages/settings/sections/desktop.dart index c61f0150..ad45c689 100644 --- a/lib/pages/settings/sections/desktop.dart +++ b/lib/pages/settings/sections/desktop.dart @@ -1,6 +1,6 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; +import 'package:flutter/material.dart' show ListTile; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/models/database/database.dart'; import 'package:spotube/modules/settings/section_card_with_heading.dart'; @@ -25,11 +25,11 @@ class SettingsDesktopSection extends HookConsumerWidget { title: Text(context.l10n.close_behavior), value: preferences.closeBehavior, options: [ - DropdownMenuItem( + SelectItemButton( value: CloseBehavior.close, child: Text(context.l10n.close), ), - DropdownMenuItem( + SelectItemButton( value: CloseBehavior.minimizeToTray, child: Text(context.l10n.minimize_to_tray), ), @@ -40,23 +40,29 @@ class SettingsDesktopSection extends HookConsumerWidget { } }, ), - SwitchListTile( - secondary: const Icon(SpotubeIcons.tray), + ListTile( + leading: const Icon(SpotubeIcons.tray), title: Text(context.l10n.show_tray_icon), - value: preferences.showSystemTrayIcon, - onChanged: preferencesNotifier.setShowSystemTrayIcon, + trailing: Switch( + value: preferences.showSystemTrayIcon, + onChanged: preferencesNotifier.setShowSystemTrayIcon, + ), ), - SwitchListTile( - secondary: const Icon(SpotubeIcons.window), + ListTile( + leading: const Icon(SpotubeIcons.window), title: Text(context.l10n.use_system_title_bar), - value: preferences.systemTitleBar, - onChanged: preferencesNotifier.setSystemTitleBar, + trailing: Switch( + value: preferences.systemTitleBar, + onChanged: preferencesNotifier.setSystemTitleBar, + ), ), - SwitchListTile( - secondary: const Icon(SpotubeIcons.discord), + ListTile( + leading: const Icon(SpotubeIcons.discord), title: Text(context.l10n.discord_rich_presence), - value: preferences.discordPresence, - onChanged: preferencesNotifier.setDiscordPresence, + trailing: Switch( + value: preferences.discordPresence, + onChanged: preferencesNotifier.setDiscordPresence, + ), ), ], ); diff --git a/lib/pages/settings/sections/developers.dart b/lib/pages/settings/sections/developers.dart index f33fe843..0862e023 100644 --- a/lib/pages/settings/sections/developers.dart +++ b/lib/pages/settings/sections/developers.dart @@ -1,6 +1,9 @@ -import 'package:flutter/material.dart'; +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/material.dart' show ListTile; import 'package:flutter_hooks/flutter_hooks.dart'; -import 'package:go_router/go_router.dart'; + +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/settings/section_card_with_heading.dart'; import 'package:spotube/extensions/context.dart'; @@ -18,7 +21,7 @@ class SettingsDevelopersSection extends HookWidget { title: Text(context.l10n.logs), trailing: const Icon(SpotubeIcons.angleRight), onTap: () { - GoRouter.of(context).push("/settings/logs"); + context.navigateTo(const LogsRoute()); }, ) ], diff --git a/lib/pages/settings/sections/downloads.dart b/lib/pages/settings/sections/downloads.dart index 8e679a7d..516d2aca 100644 --- a/lib/pages/settings/sections/downloads.dart +++ b/lib/pages/settings/sections/downloads.dart @@ -1,8 +1,9 @@ import 'package:file_picker/file_picker.dart'; import 'package:file_selector/file_selector.dart'; -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' show ListTile; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/modules/settings/section_card_with_heading.dart'; import 'package:spotube/extensions/context.dart'; @@ -40,9 +41,9 @@ class SettingsDownloadsSection extends HookConsumerWidget { leading: const Icon(SpotubeIcons.download), title: Text(context.l10n.download_location), subtitle: Text(preferences.downloadLocation), - trailing: FilledButton( + trailing: IconButton.secondary( onPressed: pickDownloadLocation, - child: const Icon(SpotubeIcons.folder), + icon: const Icon(SpotubeIcons.folder), ), onTap: pickDownloadLocation, ), diff --git a/lib/pages/settings/sections/language_region.dart b/lib/pages/settings/sections/language_region.dart index 18c2d088..920b0df7 100644 --- a/lib/pages/settings/sections/language_region.dart +++ b/lib/pages/settings/sections/language_region.dart @@ -1,10 +1,10 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; +import 'package:collection/collection.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/collections/language_codes.dart'; -import 'package:spotube/collections/spotify_markets.dart'; +import 'package:spotube/collections/markets.dart'; import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/models/metadata/market.dart'; import 'package:spotube/modules/settings/section_card_with_heading.dart'; import 'package:spotube/components/adaptive/adaptive_select_tile.dart'; import 'package:spotube/extensions/constrains.dart'; @@ -12,6 +12,15 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/l10n/l10n.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +final localWithName = L10n.all.map((e) { + final isoCodeName = + LanguageLocals.getDisplayLanguage(e.languageCode, e.countryCode); + return ( + locale: e, + name: "${isoCodeName.name} (${isoCodeName.nativeName})", + ); +}).sortedBy((e) => e.name); + class SettingsLanguageRegionSection extends HookConsumerWidget { const SettingsLanguageRegionSection({super.key}); @@ -24,7 +33,6 @@ class SettingsLanguageRegionSection extends HookConsumerWidget { return SectionCardWithHeading( heading: context.l10n.language_region, children: [ - const Gap(10), AdaptiveSelectTile( value: preferences.locale, onChanged: (locale) { @@ -34,22 +42,12 @@ class SettingsLanguageRegionSection extends HookConsumerWidget { title: Text(context.l10n.language), secondary: const Icon(SpotubeIcons.language), options: [ - DropdownMenuItem( + SelectItemButton( value: const Locale("system", "system"), child: Text(context.l10n.system_default), ), - for (final locale in L10n.all) - DropdownMenuItem( - value: locale, - child: Builder(builder: (context) { - final isoCodeName = LanguageLocals.getDisplayLanguage( - locale.languageCode, - ); - return Text( - "${isoCodeName.name} (${isoCodeName.nativeName})", - ); - }), - ), + for (final (:locale, :name) in localWithName) + SelectItemButton(value: locale, child: Text(name)), ], ), AdaptiveSelectTile( @@ -62,9 +60,9 @@ class SettingsLanguageRegionSection extends HookConsumerWidget { if (value == null) return; preferencesNotifier.setRecommendationMarket(value); }, - options: spotifyMarkets + options: marketsMap .map( - (country) => DropdownMenuItem( + (country) => SelectItemButton( value: country.$1, child: Text(country.$2), ), diff --git a/lib/pages/settings/sections/playback.dart b/lib/pages/settings/sections/playback.dart index f8868789..0a29c991 100644 --- a/lib/pages/settings/sections/playback.dart +++ b/lib/pages/settings/sections/playback.dart @@ -1,21 +1,24 @@ -import 'package:collection/collection.dart'; -import 'package:flutter/gestures.dart'; -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; -import 'package:go_router/go_router.dart'; -import 'package:google_fonts/google_fonts.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:piped_client/piped_client.dart'; -import 'package:spotube/collections/spotube_icons.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/modules/settings/section_card_with_heading.dart'; -import 'package:spotube/components/adaptive/adaptive_select_tile.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/provider/audio_player/sources/invidious_instances_provider.dart'; -import 'package:spotube/provider/audio_player/sources/piped_instances_provider.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +import 'dart:io'; + +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/gestures.dart'; +import 'package:flutter/material.dart' show ListTile; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/components/adaptive/adaptive_select_tile.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/modules/settings/playback/edit_connect_port_dialog.dart'; +import 'package:spotube/modules/settings/section_card_with_heading.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/modules/settings/youtube_engine_not_installed_dialog.dart'; +import 'package:spotube/provider/metadata_plugin/audio_source/quality_presets.dart'; +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +import 'package:spotube/services/kv_store/kv_store.dart'; +import 'package:spotube/services/youtube_engine/yt_dlp_engine.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; import 'package:spotube/utils/platform.dart'; class SettingsPlaybackSection extends HookConsumerWidget { @@ -25,223 +28,108 @@ class SettingsPlaybackSection extends HookConsumerWidget { Widget build(BuildContext context, ref) { final preferences = ref.watch(userPreferencesProvider); final preferencesNotifier = ref.watch(userPreferencesProvider.notifier); + final sourcePresets = ref.watch(audioSourcePresetsProvider); + final sourcePresetsNotifier = + ref.watch(audioSourcePresetsProvider.notifier); final theme = Theme.of(context); return SectionCardWithHeading( heading: context.l10n.playback, children: [ - const Gap(10), - AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.audioQuality), - title: Text(context.l10n.audio_quality), - value: preferences.audioQuality, - options: [ - DropdownMenuItem( - value: SourceQualities.high, - child: Text(context.l10n.high), - ), - DropdownMenuItem( - value: SourceQualities.medium, - child: Text(context.l10n.medium), - ), - DropdownMenuItem( - value: SourceQualities.low, - child: Text(context.l10n.low), - ), - ], - onChanged: (value) { - if (value != null) { - preferencesNotifier.setAudioQuality(value); - } - }, - ), - const Gap(5), - AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.api), - title: Text(context.l10n.audio_source), - value: preferences.audioSource, - options: AudioSource.values - .map((e) => DropdownMenuItem( + AdaptiveSelectTile( + secondary: const Icon(SpotubeIcons.engine), + title: Text(context.l10n.youtube_engine), + value: preferences.youtubeClientEngine, + options: YoutubeClientEngine.values + .where((e) => e.isAvailableForPlatform()) + .map((e) => SelectItemButton( value: e, child: Text(e.label), )) .toList(), - onChanged: (value) { + onChanged: (value) async { if (value == null) return; - preferencesNotifier.setAudioSource(value); + if (value == YoutubeClientEngine.ytDlp) { + final customPath = KVStoreService.getYoutubeEnginePath(value); + if (!await YtDlpEngine.isInstalled() && + (customPath == null || !await File(customPath).exists()) && + context.mounted) { + final hasInstalled = await showDialog( + context: context, + builder: (context) => + YouTubeEngineNotInstalledDialog(engine: value), + ); + if (hasInstalled != true) return; + } + } + preferencesNotifier.setYoutubeClientEngine(value); }, ), - AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: preferences.audioSource != AudioSource.piped - ? const SizedBox.shrink() - : Consumer(builder: (context, ref, child) { - final instanceList = ref.watch(pipedInstancesFutureProvider); - - return instanceList.when( - data: (data) { - return AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.piped), - title: Text(context.l10n.piped_instance), - subtitle: RichText( - text: TextSpan( - children: [ - TextSpan( - text: context.l10n.piped_description, - style: theme.textTheme.bodyMedium, - ), - const TextSpan(text: "\n"), - TextSpan( - text: context.l10n.piped_warning, - style: theme.textTheme.labelMedium, - ) - ], - ), - ), - value: preferences.pipedInstance, - showValueWhenUnfolded: false, - options: data - .sortedBy((e) => e.name) - .map( - (e) => DropdownMenuItem( - value: e.apiUrl, - child: RichText( - text: TextSpan( - children: [ - TextSpan( - text: "${e.name.trim()}\n", - style: theme.textTheme.labelLarge, - ), - TextSpan( - text: e.locations - .map(countryCodeToEmoji) - .join(""), - style: GoogleFonts.notoColorEmoji(), - ), - ], - ), - ), - ), - ) - .toList(), - onChanged: (value) { - if (value != null) { - preferencesNotifier.setPipedInstance(value); - } - }, - ); - }, - loading: () => const Center( - child: CircularProgressIndicator(), - ), - error: (error, stackTrace) => Text(error.toString()), - ); - }), - ), - AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: preferences.audioSource != AudioSource.invidious - ? const SizedBox.shrink() - : Consumer(builder: (context, ref, child) { - final instanceList = ref.watch(invidiousInstancesProvider); - - return instanceList.when( - data: (data) { - return AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.piped), - title: Text(context.l10n.invidious_instance), - subtitle: RichText( - text: TextSpan( - children: [ - TextSpan( - text: context.l10n.invidious_description, - style: theme.textTheme.bodyMedium, - ), - const TextSpan(text: "\n"), - TextSpan( - text: context.l10n.invidious_warning, - style: theme.textTheme.labelMedium, - ) - ], - ), - ), - value: preferences.invidiousInstance, - showValueWhenUnfolded: false, - options: data - .sortedBy((e) => e.name) - .map( - (e) => DropdownMenuItem( - value: e.details.uri, - child: RichText( - text: TextSpan( - children: [ - TextSpan( - text: "${e.name.trim()}\n", - style: theme.textTheme.labelLarge, - ), - TextSpan( - text: countryCodeToEmoji( - e.details.region, - ), - style: GoogleFonts.notoColorEmoji(), - ), - ], - ), - ), - ), - ) - .toList(), - onChanged: (value) { - if (value != null) { - preferencesNotifier.setInvidiousInstance(value); - } - }, - ); - }, - loading: () => const Center( - child: CircularProgressIndicator(), - ), - error: (error, stackTrace) => Text(error.toString()), - ); - }), - ), - AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: preferences.audioSource != AudioSource.piped - ? const SizedBox.shrink() - : AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.search), - title: Text(context.l10n.search_mode), - value: preferences.searchMode, - options: SearchMode.values - .map((e) => DropdownMenuItem( - value: e, - child: Text(e.label), - )) - .toList(), - onChanged: (value) { - if (value == null) return; - preferencesNotifier.setSearchMode(value); - }, - ), - ), - AnimatedSwitcher( - duration: const Duration(milliseconds: 300), - child: preferences.searchMode == SearchMode.youtube && - (preferences.audioSource == AudioSource.piped || - preferences.audioSource == AudioSource.youtube || - preferences.audioSource == AudioSource.invidious) - ? SwitchListTile( - secondary: const Icon(SpotubeIcons.skip), - title: Text(context.l10n.skip_non_music), - value: preferences.skipNonMusic, - onChanged: (state) { - preferencesNotifier.setSkipNonMusic(state); - }, - ) - : const SizedBox.shrink(), - ), - SwitchListTile( + if (sourcePresets.presets.isNotEmpty) ...[ + AdaptiveSelectTile( + secondary: const Icon(SpotubeIcons.plugin), + title: Text(context.l10n.streaming_music_format), + value: sourcePresets.selectedStreamingContainerIndex, + options: [ + for (final MapEntry(:key, value: preset) + in sourcePresets.presets.asMap().entries) + SelectItemButton(value: key, child: Text(preset.name)), + ], + onChanged: (value) { + if (value == null) return; + sourcePresetsNotifier.setSelectedStreamingContainerIndex(value); + }, + ), + AdaptiveSelectTile( + secondary: const Icon(SpotubeIcons.audioQuality), + title: Text(context.l10n.streaming_music_quality), + value: sourcePresets.selectedStreamingQualityIndex, + options: [ + for (final MapEntry(:key, value: quality) in sourcePresets + .presets[sourcePresets.selectedStreamingContainerIndex] + .qualities + .asMap() + .entries) + SelectItemButton(value: key, child: Text(quality.toString())), + ], + onChanged: (value) { + if (value == null) return; + sourcePresetsNotifier.setSelectedStreamingQualityIndex(value); + }, + ), + AdaptiveSelectTile( + secondary: const Icon(SpotubeIcons.plugin), + title: Text(context.l10n.download_music_format), + value: sourcePresets.selectedDownloadingContainerIndex, + options: [ + for (final MapEntry(:key, value: preset) + in sourcePresets.presets.asMap().entries) + SelectItemButton(value: key, child: Text(preset.name)), + ], + onChanged: (value) { + if (value == null) return; + sourcePresetsNotifier.setSelectedDownloadingContainerIndex(value); + }, + ), + AdaptiveSelectTile( + secondary: const Icon(SpotubeIcons.audioQuality), + title: Text(context.l10n.download_music_quality), + value: sourcePresets.selectedStreamingQualityIndex, + options: [ + for (final MapEntry(:key, value: quality) in sourcePresets + .presets[sourcePresets.selectedDownloadingContainerIndex] + .qualities + .asMap() + .entries) + SelectItemButton(value: key, child: Text(quality.toString())), + ], + onChanged: (value) { + if (value == null) return; + sourcePresetsNotifier.setSelectedStreamingQualityIndex(value); + }, + ), + ], + ListTile( title: Text(context.l10n.cache_music), subtitle: kIsMobile ? null @@ -253,7 +141,7 @@ class SettingsPlaybackSection extends HookConsumerWidget { text: context.l10n.cache_folder.toLowerCase(), recognizer: TapGestureRecognizer() ..onTap = preferencesNotifier.openCacheFolder, - style: theme.textTheme.bodyMedium?.copyWith( + style: theme.typography.normal.copyWith( color: theme.colorScheme.primary, decoration: TextDecoration.underline, ), @@ -261,79 +149,67 @@ class SettingsPlaybackSection extends HookConsumerWidget { ], ), ), - secondary: const Icon(SpotubeIcons.cache), - value: preferences.cacheMusic, - onChanged: preferencesNotifier.setCacheMusic, + leading: const Icon(SpotubeIcons.cache), + trailing: Switch( + value: preferences.cacheMusic, + onChanged: preferencesNotifier.setCacheMusic, + ), ), ListTile( leading: const Icon(SpotubeIcons.playlistRemove), title: Text(context.l10n.blacklist), subtitle: Text(context.l10n.blacklist_description), onTap: () { - GoRouter.of(context).push("/settings/blacklist"); + context.navigateTo(const BlackListRoute()); }, trailing: const Icon(SpotubeIcons.angleRight), ), - SwitchListTile( - secondary: const Icon(SpotubeIcons.normalize), + ListTile( + leading: const Icon(SpotubeIcons.normalize), title: Text(context.l10n.normalize_audio), - value: preferences.normalizeAudio, - onChanged: preferencesNotifier.setNormalizeAudio, - ), - if (preferences.audioSource != AudioSource.jiosaavn) ...[ - const Gap(5), - AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.stream), - title: Text(context.l10n.streaming_music_codec), - value: preferences.streamMusicCodec, - showValueWhenUnfolded: false, - options: SourceCodecs.values - .map((e) => DropdownMenuItem( - value: e, - child: Text( - e.label, - style: theme.textTheme.labelMedium, - ), - )) - .toList(), - onChanged: (value) { - if (value == null) return; - preferencesNotifier.setStreamMusicCodec(value); - }, + trailing: Switch( + value: preferences.normalizeAudio, + onChanged: preferencesNotifier.setNormalizeAudio, ), - const Gap(5), - AdaptiveSelectTile( - secondary: const Icon(SpotubeIcons.file), - title: Text(context.l10n.download_music_codec), - value: preferences.downloadMusicCodec, - showValueWhenUnfolded: false, - options: SourceCodecs.values - .map((e) => DropdownMenuItem( - value: e, - child: Text( - e.label, - style: theme.textTheme.labelMedium, - ), - )) - .toList(), - onChanged: (value) { - if (value == null) return; - preferencesNotifier.setDownloadMusicCodec(value); - }, - ) - ], - SwitchListTile( - secondary: const Icon(SpotubeIcons.repeat), - title: Text(context.l10n.endless_playback), - value: preferences.endlessPlayback, - onChanged: preferencesNotifier.setEndlessPlayback, ), - SwitchListTile( + ListTile( + leading: const Icon(SpotubeIcons.repeat), + title: Text(context.l10n.endless_playback), + trailing: Switch( + value: preferences.endlessPlayback, + onChanged: preferencesNotifier.setEndlessPlayback, + )), + ListTile( title: Text(context.l10n.enable_connect), subtitle: Text(context.l10n.enable_connect_description), - secondary: const Icon(SpotubeIcons.connect), - value: preferences.enableConnect, - onChanged: preferencesNotifier.setEnableConnect, + leading: const Icon(SpotubeIcons.connect), + trailing: Row( + mainAxisSize: MainAxisSize.min, + spacing: 10, + children: [ + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.edit_port), + ).call, + child: IconButton.outline( + icon: const Icon(SpotubeIcons.edit), + size: ButtonSize.small, + onPressed: () { + showDialog( + context: context, + barrierColor: Colors.black.withValues(alpha: 0.5), + builder: (context) => + const SettingsPlaybackEditConnectPortDialog(), + ); + }, + ), + ), + Switch( + value: preferences.enableConnect, + onChanged: preferencesNotifier.setEnableConnect, + ), + ], + ), ), ], ); diff --git a/lib/pages/settings/settings.dart b/lib/pages/settings/settings.dart index 8bce4bcf..0948bdeb 100644 --- a/lib/pages/settings/settings.dart +++ b/lib/pages/settings/settings.dart @@ -1,7 +1,8 @@ import 'package:flutter/foundation.dart'; -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' show Material, MaterialType; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/extensions/context.dart'; import 'package:spotube/pages/settings/sections/about.dart'; @@ -14,7 +15,9 @@ import 'package:spotube/pages/settings/sections/language_region.dart'; import 'package:spotube/pages/settings/sections/playback.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/utils/platform.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class SettingsPage extends HookConsumerWidget { static const name = "settings"; @@ -28,37 +31,40 @@ class SettingsPage extends HookConsumerWidget { return SafeArea( bottom: false, child: Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.settings), - centerTitle: true, - automaticallyImplyLeading: true, - ), - body: Scrollbar( + headers: [ + TitleBar( + title: Text(context.l10n.settings), + ) + ], + child: Scrollbar( controller: controller, child: Center( child: ConstrainedBox( constraints: const BoxConstraints(maxWidth: 1366), child: ScrollConfiguration( behavior: const ScrollBehavior().copyWith(scrollbars: false), - child: ListView( - controller: controller, - children: [ - const SettingsAccountSection(), - const SettingsLanguageRegionSection(), - const SettingsAppearanceSection(), - const SettingsPlaybackSection(), - const SettingsDownloadsSection(), - if (kIsDesktop) const SettingsDesktopSection(), - if (!kIsWeb) const SettingsDevelopersSection(), - const SettingsAboutSection(), - Center( - child: FilledButton( - onPressed: preferencesNotifier.reset, - child: Text(context.l10n.restore_defaults), + child: Material( + type: MaterialType.transparency, + child: ListView( + controller: controller, + children: [ + const SettingsAccountSection(), + const SettingsLanguageRegionSection(), + const SettingsAppearanceSection(), + const SettingsPlaybackSection(), + const SettingsDownloadsSection(), + if (kIsDesktop) const SettingsDesktopSection(), + if (!kIsWeb) const SettingsDevelopersSection(), + const SettingsAboutSection(), + Center( + child: Button.destructive( + onPressed: preferencesNotifier.reset, + child: Text(context.l10n.restore_defaults), + ), ), - ), - const SizedBox(height: 10), - ], + const SizedBox(height: 200), + ], + ), ), ), ), diff --git a/lib/pages/stats/albums/albums.dart b/lib/pages/stats/albums/albums.dart index e14a2f32..363e7962 100644 --- a/lib/pages/stats/albums/albums.dart +++ b/lib/pages/stats/albums/albums.dart @@ -1,5 +1,5 @@ -import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; @@ -8,9 +8,11 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/albums.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsAlbumsPage extends HookConsumerWidget { static const name = "stats_albums"; const StatsAlbumsPage({super.key}); @@ -24,30 +26,33 @@ class StatsAlbumsPage extends HookConsumerWidget { final albumsData = topAlbums.asData?.value.items ?? []; - return Scaffold( - appBar: PageWindowTitleBar( - automaticallyImplyLeading: true, - centerTitle: false, - title: Text(context.l10n.albums), - ), - body: Skeletonizer( - enabled: topAlbums.isLoading && !topAlbums.isLoadingNextPage, - child: InfiniteList( - onFetchData: () async { - await topAlbumsNotifier.fetchMore(); - }, - hasError: topAlbums.hasError, - isLoading: topAlbums.isLoading && !topAlbums.isLoadingNextPage, - hasReachedMax: topAlbums.asData?.value.hasMore ?? true, - itemCount: albumsData.length, - itemBuilder: (context, index) { - final album = albumsData[index]; - return StatsAlbumItem( - album: album.album, - info: Text(context.l10n - .count_plays(compactNumberFormatter.format(album.count))), - ); - }, + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.albums), + ) + ], + child: Skeletonizer( + enabled: topAlbums.isLoading && !topAlbums.isLoadingNextPage, + child: InfiniteList( + onFetchData: () async { + await topAlbumsNotifier.fetchMore(); + }, + hasError: topAlbums.hasError, + isLoading: topAlbums.isLoading && !topAlbums.isLoadingNextPage, + hasReachedMax: topAlbums.asData?.value.hasMore ?? true, + itemCount: albumsData.length, + itemBuilder: (context, index) { + final album = albumsData[index]; + return StatsAlbumItem( + album: album.album, + info: Text(context.l10n + .count_plays(compactNumberFormatter.format(album.count))), + ); + }, + ), ), ), ); diff --git a/lib/pages/stats/artists/artists.dart b/lib/pages/stats/artists/artists.dart index 436bbb57..340f7b4b 100644 --- a/lib/pages/stats/artists/artists.dart +++ b/lib/pages/stats/artists/artists.dart @@ -1,6 +1,6 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; @@ -9,9 +9,11 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsArtistsPage extends HookConsumerWidget { static const name = "stats_artists"; const StatsArtistsPage({super.key}); @@ -25,32 +27,37 @@ class StatsArtistsPage extends HookConsumerWidget { ref.watch(historyTopTracksProvider(HistoryDuration.allTime).notifier); final artistsData = useMemoized( - () => topTracks.asData?.value.artists ?? [], [topTracks.asData?.value]); + () => topTracksNotifier.artists, + [topTracks.asData?.value], + ); - return Scaffold( - appBar: PageWindowTitleBar( - automaticallyImplyLeading: true, - centerTitle: false, - title: Text(context.l10n.artists), - ), - body: Skeletonizer( - enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, - child: InfiniteList( - onFetchData: () async { - await topTracksNotifier.fetchMore(); - }, - hasError: topTracks.hasError, - isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, - hasReachedMax: topTracks.asData?.value.hasMore ?? true, - itemCount: artistsData.length, - itemBuilder: (context, index) { - final artist = artistsData[index]; - return StatsArtistItem( - artist: artist.artist, - info: Text(context.l10n - .count_plays(compactNumberFormatter.format(artist.count))), - ); - }, + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.artists), + ) + ], + child: Skeletonizer( + enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, + child: InfiniteList( + onFetchData: () async { + await topTracksNotifier.fetchMore(); + }, + hasError: topTracks.hasError, + isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, + hasReachedMax: topTracks.asData?.value.hasMore ?? true, + itemCount: artistsData.length, + itemBuilder: (context, index) { + final artist = artistsData[index]; + return StatsArtistItem( + artist: artist.artist, + info: Text(context.l10n + .count_plays(compactNumberFormatter.format(artist.count))), + ); + }, + ), ), ), ); diff --git a/lib/pages/stats/fees/fees.dart b/lib/pages/stats/fees/fees.dart index da62fb30..15b93057 100644 --- a/lib/pages/stats/fees/fees.dart +++ b/lib/pages/stats/fees/fees.dart @@ -1,6 +1,6 @@ -import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:sliver_tools/sliver_tools.dart'; import 'package:spotube/collections/formatters.dart'; @@ -10,9 +10,11 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsStreamFeesPage extends HookConsumerWidget { static const name = "stats_stream_fees"; @@ -20,7 +22,6 @@ class StatsStreamFeesPage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :hintColor) = Theme.of(context); final duration = useState(HistoryDuration.days30); final topTracks = ref.watch( @@ -30,7 +31,9 @@ class StatsStreamFeesPage extends HookConsumerWidget { ref.watch(historyTopTracksProvider(duration.value).notifier); final artistsData = useMemoized( - () => topTracks.asData?.value.artists ?? [], [topTracks.asData?.value]); + () => topTracksNotifier.artists, + [topTracks.asData?.value], + ); final total = useMemoized( () => artistsData.fold( @@ -40,98 +43,97 @@ class StatsStreamFeesPage extends HookConsumerWidget { [artistsData], ); - return Scaffold( - appBar: PageWindowTitleBar( - automaticallyImplyLeading: true, - centerTitle: false, - title: Text(context.l10n.streaming_fees_hypothetical), - ), - body: CustomScrollView( - slivers: [ - SliverCrossAxisConstrained( - maxCrossAxisExtent: 600, - alignment: -1, - child: SliverPadding( - padding: const EdgeInsets.all(16.0), - sliver: SliverToBoxAdapter( - child: Text( - context.l10n.spotify_hipotetical_calculation, - style: textTheme.bodySmall?.copyWith( - color: hintColor, - ), + final translations = { + HistoryDuration.days7: context.l10n.this_week, + HistoryDuration.days30: context.l10n.this_month, + HistoryDuration.months6: context.l10n.last_6_months, + HistoryDuration.year: context.l10n.this_year, + HistoryDuration.years2: context.l10n.last_2_years, + HistoryDuration.allTime: context.l10n.all_time, + }; + + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.streaming_fees_hypothetical), + ) + ], + child: CustomScrollView( + slivers: [ + SliverCrossAxisConstrained( + maxCrossAxisExtent: 600, + alignment: -1, + child: SliverPadding( + padding: const EdgeInsets.all(16.0), + sliver: SliverToBoxAdapter( + child: Text( + context.l10n.hipotetical_calculation, + ).small().muted(), ), ), ), - ), - SliverToBoxAdapter( - child: Padding( - padding: const EdgeInsets.symmetric(horizontal: 16), - child: Row( - mainAxisAlignment: MainAxisAlignment.spaceBetween, - children: [ - Text( - context.l10n.total_money(usdFormatter.format(total)), - style: textTheme.titleLarge, - ), - DropdownButton( - value: duration.value, - onChanged: (value) { - if (value == null) return; - duration.value = value; - }, - items: [ - DropdownMenuItem( - value: HistoryDuration.days7, - child: Text(context.l10n.this_week), - ), - DropdownMenuItem( - value: HistoryDuration.days30, - child: Text(context.l10n.this_month), - ), - DropdownMenuItem( - value: HistoryDuration.months6, - child: Text(context.l10n.last_6_months), - ), - DropdownMenuItem( - value: HistoryDuration.year, - child: Text(context.l10n.this_year), - ), - DropdownMenuItem( - value: HistoryDuration.years2, - child: Text(context.l10n.last_2_years), - ), - DropdownMenuItem( - value: HistoryDuration.allTime, - child: Text(context.l10n.all_time), - ), - ], - ), - ], + SliverToBoxAdapter( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + context.l10n.total_money(usdFormatter.format(total)), + ).semiBold().large(), + Select( + value: duration.value, + onChanged: (value) { + if (value == null) return; + duration.value = value; + }, + itemBuilder: (context, value) => + Text(translations[value]!), + constraints: const BoxConstraints(maxWidth: 150), + popupWidthConstraint: PopoverConstraint.anchorMaxSize, + popup: SelectPopup( + items: SelectItemBuilder( + childCount: translations.length, + builder: (context, index) { + final entry = translations.entries.elementAt(index); + return SelectItemButton( + value: entry.key, + child: Text(entry.value), + ); + }, + ), + ).call, + ), + ], + ), ), ), - ), - SliverSafeArea( - sliver: Skeletonizer.sliver( - enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, - child: SliverInfiniteList( - onFetchData: () async { - await topTracksNotifier.fetchMore(); - }, - hasError: topTracks.hasError, - isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, - hasReachedMax: topTracks.asData?.value.hasMore ?? true, - itemCount: artistsData.length, - itemBuilder: (context, index) { - final artist = artistsData[index]; - return StatsArtistItem( - artist: artist.artist, - info: Text(usdFormatter.format(artist.count * 0.005)), - ); - }, + SliverSafeArea( + sliver: Skeletonizer.sliver( + enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, + child: SliverInfiniteList( + onFetchData: () async { + await topTracksNotifier.fetchMore(); + }, + hasError: topTracks.hasError, + isLoading: + topTracks.isLoading && !topTracks.isLoadingNextPage, + hasReachedMax: topTracks.asData?.value.hasMore ?? true, + itemCount: artistsData.length, + itemBuilder: (context, index) { + final artist = artistsData[index]; + return StatsArtistItem( + artist: artist.artist, + info: Text(usdFormatter.format(artist.count * 0.005)), + ); + }, + ), ), ), - ), - ], + ], + ), ), ); } diff --git a/lib/pages/stats/minutes/minutes.dart b/lib/pages/stats/minutes/minutes.dart index 3ad0984b..a6c95992 100644 --- a/lib/pages/stats/minutes/minutes.dart +++ b/lib/pages/stats/minutes/minutes.dart @@ -1,6 +1,5 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; @@ -9,9 +8,11 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsMinutesPage extends HookConsumerWidget { static const name = "stats_minutes"; @@ -27,33 +28,41 @@ class StatsMinutesPage extends HookConsumerWidget { final tracksData = topTracks.asData?.value.items ?? []; - return Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.minutes_listened), - centerTitle: false, - automaticallyImplyLeading: true, - ), - body: Skeletonizer( - enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, - child: InfiniteList( - separatorBuilder: (context, index) => const Gap(8), - onFetchData: () async { - await topTracksNotifier.fetchMore(); - }, - hasError: topTracks.hasError, - isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, - hasReachedMax: topTracks.asData?.value.hasMore ?? true, - itemCount: tracksData.length, - itemBuilder: (context, index) { - final track = tracksData[index]; - return StatsTrackItem( - track: track.track, - info: Text( - context.l10n.count_mins(compactNumberFormatter - .format(track.count * track.track.duration!.inMinutes)), - ), - ); - }, + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.minutes_listened), + ) + ], + child: Skeletonizer( + enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, + child: InfiniteList( + separatorBuilder: (context, index) => const Gap(8), + onFetchData: () async { + await topTracksNotifier.fetchMore(); + }, + hasError: topTracks.hasError, + isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, + hasReachedMax: topTracks.asData?.value.hasMore ?? true, + itemCount: tracksData.length, + itemBuilder: (context, index) { + final track = tracksData[index]; + return StatsTrackItem( + track: track.track, + info: Text( + context.l10n.count_mins( + compactNumberFormatter.format( + track.count * + Duration(milliseconds: track.track.durationMs) + .inMinutes, + ), + ), + ), + ); + }, + ), ), ), ); diff --git a/lib/pages/stats/playlists/playlists.dart b/lib/pages/stats/playlists/playlists.dart index 4e83b0a2..369066f7 100644 --- a/lib/pages/stats/playlists/playlists.dart +++ b/lib/pages/stats/playlists/playlists.dart @@ -1,5 +1,5 @@ -import 'package:flutter/material.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; @@ -8,9 +8,11 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/playlists.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsPlaylistsPage extends HookConsumerWidget { static const name = "stats_playlists"; const StatsPlaylistsPage({super.key}); @@ -25,32 +27,36 @@ class StatsPlaylistsPage extends HookConsumerWidget { final playlistsData = topPlaylists.asData?.value.items ?? []; - return Scaffold( - appBar: PageWindowTitleBar( - automaticallyImplyLeading: true, - centerTitle: false, - title: Text(context.l10n.playlists), - ), - body: Skeletonizer( - enabled: topPlaylists.isLoading && !topPlaylists.isLoadingNextPage, - child: InfiniteList( - onFetchData: () async { - await topPlaylistsNotifier.fetchMore(); - }, - hasError: topPlaylists.hasError, - isLoading: topPlaylists.isLoading && !topPlaylists.isLoadingNextPage, - hasReachedMax: topPlaylists.asData?.value.hasMore ?? true, - itemCount: playlistsData.length, - itemBuilder: (context, index) { - final playlist = playlistsData[index]; - return StatsPlaylistItem( - playlist: playlist.playlist, - info: Text( - context.l10n - .count_plays(compactNumberFormatter.format(playlist.count)), - ), - ); - }, + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.playlists), + ) + ], + child: Skeletonizer( + enabled: topPlaylists.isLoading && !topPlaylists.isLoadingNextPage, + child: InfiniteList( + onFetchData: () async { + await topPlaylistsNotifier.fetchMore(); + }, + hasError: topPlaylists.hasError, + isLoading: + topPlaylists.isLoading && !topPlaylists.isLoadingNextPage, + hasReachedMax: topPlaylists.asData?.value.hasMore ?? true, + itemCount: playlistsData.length, + itemBuilder: (context, index) { + final playlist = playlistsData[index]; + return StatsPlaylistItem( + playlist: playlist.playlist, + info: Text( + context.l10n.count_plays( + compactNumberFormatter.format(playlist.count)), + ), + ); + }, + ), ), ), ); diff --git a/lib/pages/stats/stats.dart b/lib/pages/stats/stats.dart index b2dc03c2..da7c64f3 100644 --- a/lib/pages/stats/stats.dart +++ b/lib/pages/stats/stats.dart @@ -1,11 +1,13 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; import 'package:spotube/modules/stats/summary/summary.dart'; import 'package:spotube/modules/stats/top/top.dart'; import 'package:spotube/utils/platform.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsPage extends HookConsumerWidget { static const name = "stats"; @@ -13,21 +15,30 @@ class StatsPage extends HookConsumerWidget { @override Widget build(BuildContext context, ref) { - return SafeArea( - bottom: false, - child: Scaffold( - appBar: kIsMacOS || kIsMobile ? null : const PageWindowTitleBar(), - body: CustomScrollView( - slivers: [ - if (kIsMacOS) const SliverGap(20), - const StatsPageSummarySection(), - const StatsPageTopSection(), - const SliverToBoxAdapter( - child: SafeArea( - child: SizedBox(), - ), - ) + return PopScope( + canPop: false, + onPopInvokedWithResult: (didPop, result) { + context.navigateTo(const HomeRoute()); + }, + child: SafeArea( + bottom: false, + child: Scaffold( + headers: [ + if (kTitlebarVisible) + const TitleBar(automaticallyImplyLeading: false), ], + child: CustomScrollView( + slivers: [ + if (kIsMacOS) const SliverGap(20), + const StatsPageSummarySection(), + const StatsPageTopSection(), + const SliverToBoxAdapter( + child: SafeArea( + child: SizedBox(), + ), + ) + ], + ), ), ), ); diff --git a/lib/pages/stats/streams/streams.dart b/lib/pages/stats/streams/streams.dart index 059366e0..b2cc671d 100644 --- a/lib/pages/stats/streams/streams.dart +++ b/lib/pages/stats/streams/streams.dart @@ -1,6 +1,5 @@ -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/formatters.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; @@ -9,9 +8,11 @@ import 'package:spotube/extensions/context.dart'; import 'package:spotube/provider/history/top.dart'; import 'package:spotube/provider/history/top/tracks.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class StatsStreamsPage extends HookConsumerWidget { static const name = "stats_streams"; @@ -27,33 +28,36 @@ class StatsStreamsPage extends HookConsumerWidget { final tracksData = topTracks.asData?.value.items ?? []; - return Scaffold( - appBar: PageWindowTitleBar( - title: Text(context.l10n.streamed_songs), - centerTitle: false, - automaticallyImplyLeading: true, - ), - body: Skeletonizer( - enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, - child: InfiniteList( - separatorBuilder: (context, index) => const Gap(8), - onFetchData: () async { - await topTracksNotifier.fetchMore(); - }, - hasError: topTracks.hasError, - isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, - hasReachedMax: topTracks.asData?.value.hasMore ?? true, - itemCount: tracksData.length, - itemBuilder: (context, index) { - final track = tracksData[index]; - return StatsTrackItem( - track: track.track, - info: Text( - context.l10n - .count_plays(compactNumberFormatter.format(track.count)), - ), - ); - }, + return SafeArea( + bottom: false, + child: Scaffold( + headers: [ + TitleBar( + title: Text(context.l10n.streamed_songs), + ) + ], + child: Skeletonizer( + enabled: topTracks.isLoading && !topTracks.isLoadingNextPage, + child: InfiniteList( + separatorBuilder: (context, index) => const Gap(8), + onFetchData: () async { + await topTracksNotifier.fetchMore(); + }, + hasError: topTracks.hasError, + isLoading: topTracks.isLoading && !topTracks.isLoadingNextPage, + hasReachedMax: topTracks.asData?.value.hasMore ?? true, + itemCount: tracksData.length, + itemBuilder: (context, index) { + final track = tracksData[index]; + return StatsTrackItem( + track: track.track, + info: Text( + context.l10n + .count_plays(compactNumberFormatter.format(track.count)), + ), + ); + }, + ), ), ), ); diff --git a/lib/pages/track/track.dart b/lib/pages/track/track.dart index 84c53b74..44453ebd 100644 --- a/lib/pages/track/track.dart +++ b/lib/pages/track/track.dart @@ -1,38 +1,40 @@ import 'dart:ui'; -import 'package:flutter/material.dart'; -import 'package:gap/gap.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:skeletonizer/skeletonizer.dart'; import 'package:spotube/collections/fake.dart'; +import 'package:spotube/collections/routes.gr.dart'; import 'package:spotube/collections/spotube_icons.dart'; import 'package:spotube/components/heart_button/heart_button.dart'; import 'package:spotube/components/image/universal_image.dart'; import 'package:spotube/components/links/artist_link.dart'; import 'package:spotube/components/links/link_text.dart'; import 'package:spotube/components/titlebar/titlebar.dart'; -import 'package:spotube/components/track_tile/track_options.dart'; +import 'package:spotube/components/track_tile/track_options_button.dart'; import 'package:spotube/extensions/context.dart'; -import 'package:spotube/extensions/image.dart'; import 'package:spotube/extensions/list.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/track.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/extensions/constrains.dart'; +import 'package:auto_route/auto_route.dart'; +@RoutePage() class TrackPage extends HookConsumerWidget { static const name = "track"; final String trackId; const TrackPage({ super.key, - required this.trackId, + @PathParam("id") required this.trackId, }); @override Widget build(BuildContext context, ref) { - final ThemeData(:textTheme, :colorScheme) = Theme.of(context); + final ThemeData(:typography, :colorScheme) = Theme.of(context); final mediaQuery = MediaQuery.of(context); final playlist = ref.watch(audioPlayerProvider); @@ -40,7 +42,7 @@ class TrackPage extends HookConsumerWidget { final isActive = playlist.activeTrack?.id == trackId; - final trackQuery = ref.watch(trackProvider(trackId)); + final trackQuery = ref.watch(metadataPluginTrackProvider(trackId)); final track = trackQuery.asData?.value ?? FakeData.track; @@ -52,182 +54,203 @@ class TrackPage extends HookConsumerWidget { } } - return Scaffold( - appBar: const PageWindowTitleBar( - automaticallyImplyLeading: true, - backgroundColor: Colors.transparent, - ), - extendBodyBehindAppBar: true, - body: Stack( - children: [ - Positioned.fill( - child: Container( - decoration: BoxDecoration( - image: DecorationImage( - image: UniversalImage.imageProvider( - track.album!.images.asUrlString( - placeholder: ImagePlaceholder.albumArt, + return SafeArea( + bottom: false, + child: Scaffold( + headers: const [ + TitleBar( + backgroundColor: Colors.transparent, + surfaceBlur: 0, + ) + ], + floatingHeader: true, + child: Stack( + children: [ + Positioned.fill( + child: Container( + decoration: BoxDecoration( + image: DecorationImage( + image: UniversalImage.imageProvider( + track.album.images.asUrlString( + placeholder: ImagePlaceholder.albumArt, + ), ), + fit: BoxFit.cover, + colorFilter: ColorFilter.mode( + colorScheme.background.withValues(alpha: 0.5), + BlendMode.srcOver, + ), + alignment: Alignment.topCenter, ), - fit: BoxFit.cover, - colorFilter: ColorFilter.mode( - colorScheme.surface.withOpacity(0.5), - BlendMode.srcOver, - ), - alignment: Alignment.topCenter, ), ), ), - ), - Positioned.fill( - child: BackdropFilter( - filter: ImageFilter.blur(sigmaX: 10, sigmaY: 10), - child: Skeletonizer( - enabled: trackQuery.isLoading, - child: Container( - alignment: Alignment.topCenter, - decoration: BoxDecoration( - gradient: LinearGradient( - colors: [ - colorScheme.surface, - Colors.transparent, - ], - begin: Alignment.topCenter, - end: Alignment.bottomCenter, - stops: const [0.2, 1], + Positioned.fill( + child: BackdropFilter( + filter: ImageFilter.blur(sigmaX: 10, sigmaY: 10), + child: Skeletonizer( + enabled: trackQuery.isLoading, + child: Container( + alignment: Alignment.topCenter, + decoration: BoxDecoration( + gradient: LinearGradient( + colors: [ + colorScheme.background, + Colors.transparent, + ], + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + stops: const [0.2, 1], + ), ), - ), - child: SafeArea( - child: Wrap( - spacing: 20, - runSpacing: 20, - alignment: WrapAlignment.center, - crossAxisAlignment: WrapCrossAlignment.center, - runAlignment: WrapAlignment.center, - children: [ - ClipRRect( - borderRadius: BorderRadius.circular(10), - child: UniversalImage( - path: track.album!.images.asUrlString( - placeholder: ImagePlaceholder.albumArt, + child: SafeArea( + child: Wrap( + spacing: 20, + runSpacing: 20, + alignment: WrapAlignment.center, + crossAxisAlignment: WrapCrossAlignment.center, + runAlignment: WrapAlignment.center, + children: [ + Padding( + padding: const EdgeInsets.only(top: 20), + child: ClipRRect( + borderRadius: BorderRadius.circular(10), + child: UniversalImage( + path: track.album.images.asUrlString( + placeholder: ImagePlaceholder.albumArt, + ), + height: 200, + width: 200, + ), ), - height: 200, - width: 200, ), - ), - Padding( - padding: const EdgeInsets.symmetric(horizontal: 16.0), - child: Column( - crossAxisAlignment: mediaQuery.smAndDown - ? CrossAxisAlignment.center - : CrossAxisAlignment.start, - mainAxisSize: MainAxisSize.min, - children: [ - Text( - track.name!, - style: textTheme.titleLarge, - ), - const Gap(10), - Row( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon(SpotubeIcons.album), - const Gap(5), - Flexible( - child: LinkText( - track.album!.name!, - '/album/${track.album!.id}', - push: true, - extra: track.album, - ), - ), - ], - ), - const Gap(10), - Row( - mainAxisSize: MainAxisSize.min, - children: [ - const Icon(SpotubeIcons.artist), - const Gap(5), - Flexible( - child: ArtistLink( - artists: track.artists!, - hideOverflowArtist: false, - ), - ), - ], - ), - const Gap(10), - ConstrainedBox( - constraints: - const BoxConstraints(maxWidth: 350), - child: Row( - mainAxisSize: mediaQuery.smAndDown - ? MainAxisSize.max - : MainAxisSize.min, + Padding( + padding: + const EdgeInsets.symmetric(horizontal: 16.0), + child: Column( + crossAxisAlignment: mediaQuery.smAndDown + ? CrossAxisAlignment.center + : CrossAxisAlignment.start, + mainAxisSize: MainAxisSize.min, + children: [ + Text( + track.name, + ).large().semiBold(), + const Gap(10), + Row( + mainAxisSize: MainAxisSize.min, children: [ + const Icon(SpotubeIcons.album), const Gap(5), - if (!isActive && - !playlist.tracks - .containsBy(track, (t) => t.id)) - OutlinedButton.icon( - icon: const Icon(SpotubeIcons.queueAdd), - label: Text(context.l10n.queue), - onPressed: () { - playlistNotifier.addTrack(track); - }, + Flexible( + child: LinkText( + track.album.name, + AlbumRoute( + id: track.album.id, + album: track.album, + ), + push: true, ), - const Gap(5), - if (!isActive && - !playlist.tracks - .containsBy(track, (t) => t.id)) - IconButton.outlined( - icon: - const Icon(SpotubeIcons.lightning), - tooltip: context.l10n.play_next, - onPressed: () { - playlistNotifier - .addTracksAtFirst([track]); - }, - ), - const Gap(5), - IconButton.filled( - tooltip: isActive - ? context.l10n.pause_playback - : context.l10n.play, - icon: Icon( - isActive - ? SpotubeIcons.pause - : SpotubeIcons.play, - color: colorScheme.onPrimary, - ), - onPressed: onPlay, ), - const Gap(5), - if (mediaQuery.smAndDown) - const Spacer() - else - const Gap(20), - TrackHeartButton(track: track), - TrackOptions( - track: track, - userPlaylist: false, - ), - const Gap(5), ], ), - ), - ], + const Gap(10), + Row( + mainAxisSize: MainAxisSize.min, + children: [ + const Icon(SpotubeIcons.artist), + const Gap(5), + Flexible( + child: ArtistLink( + artists: track.artists, + hideOverflowArtist: false, + ), + ), + ], + ), + const Gap(10), + ConstrainedBox( + constraints: + const BoxConstraints(maxWidth: 350), + child: Row( + mainAxisSize: mediaQuery.smAndDown + ? MainAxisSize.max + : MainAxisSize.min, + children: [ + const Gap(5), + if (!isActive && + !playlist.tracks + .containsBy(track, (t) => t.id)) + Button.outline( + leading: + const Icon(SpotubeIcons.queueAdd), + child: Text(context.l10n.queue), + onPressed: () { + playlistNotifier.addTrack(track); + }, + ), + const Gap(5), + if (!isActive && + !playlist.tracks + .containsBy(track, (t) => t.id)) + Tooltip( + tooltip: TooltipContainer( + child: Text(context.l10n.play_next), + ).call, + child: IconButton.outline( + icon: const Icon( + SpotubeIcons.lightning), + onPressed: () { + playlistNotifier + .addTracksAtFirst([track]); + }, + ), + ), + const Gap(5), + Tooltip( + tooltip: TooltipContainer( + child: Text( + isActive + ? context.l10n.pause_playback + : context.l10n.play, + ), + ).call, + child: IconButton.primary( + shape: ButtonShape.circle, + icon: Icon( + isActive + ? SpotubeIcons.pause + : SpotubeIcons.play, + ), + onPressed: onPlay, + ), + ), + const Gap(5), + if (mediaQuery.smAndDown) + const Spacer() + else + const Gap(20), + TrackHeartButton(track: track), + TrackOptionsButton( + track: track, + userPlaylist: false, + ), + const Gap(5), + ], + ), + ), + ], + ), ), - ), - ], + ], + ), ), ), ), ), ), - ), - ], + ], + ), ), ); } diff --git a/lib/provider/audio_player/audio_player.dart b/lib/provider/audio_player/audio_player.dart index b4892a0c..66878714 100644 --- a/lib/provider/audio_player/audio_player.dart +++ b/lib/provider/audio_player/audio_player.dart @@ -1,24 +1,40 @@ import 'dart:math'; +import 'package:collection/collection.dart'; import 'package:drift/drift.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:media_kit/media_kit.dart' hide Track; -import 'package:spotify/spotify.dart' hide Playlist; +import 'package:media_kit/media_kit.dart'; import 'package:spotube/extensions/list.dart'; -import 'package:spotube/extensions/track.dart'; import 'package:spotube/models/database/database.dart'; -import 'package:spotube/models/local_track.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/state.dart'; import 'package:spotube/provider/blacklist_provider.dart'; import 'package:spotube/provider/database/database.dart'; import 'package:spotube/provider/discord_provider.dart'; -import 'package:spotube/provider/server/sourced_track.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/logger/logger.dart'; class AudioPlayerNotifier extends Notifier { BlackListNotifier get _blacklist => ref.read(blacklistProvider.notifier); + void _assertAllowedTracks(Iterable tracks) { + assert( + tracks.every( + (track) => + track is SpotubeFullTrackObject || track is SpotubeLocalTrackObject, + ), + 'All tracks must be either SpotubeFullTrackObject or SpotubeLocalTrackObject', + ); + } + + void _assertAllowedTrack(SpotubeTrackObject tracks) { + assert( + tracks is SpotubeFullTrackObject || tracks is SpotubeLocalTrackObject, + 'Track must be either SpotubeFullTrackObject or SpotubeLocalTrackObject', + ); + } + Future _syncSavedState() async { final database = ref.read(databaseProvider); @@ -32,6 +48,8 @@ class AudioPlayerNotifier extends Notifier { loopMode: audioPlayer.loopMode, shuffled: audioPlayer.isShuffled, collections: [], + tracks: const Value([]), + currentIndex: const Value(0), id: const Value(0), ), ); @@ -43,51 +61,24 @@ class AudioPlayerNotifier extends Notifier { await audioPlayer.setShuffle(playerState.shuffled); } - var playlist = - await database.select(database.playlistTable).getSingleOrNull(); - var medias = await database.select(database.playlistMediaTable).get(); + final tracks = playerState.tracks; + final currentIndex = playerState.currentIndex; - if (playlist == null) { - await database.into(database.playlistTable).insert( - PlaylistTableCompanion.insert( - audioPlayerStateId: 0, - index: audioPlayer.playlist.index, - id: const Value(0), - ), - ); - - playlist = await database.select(database.playlistTable).getSingle(); - } - - if (medias.isEmpty && audioPlayer.playlist.medias.isNotEmpty) { - await database.batch((batch) { - batch.insertAll( - database.playlistMediaTable, - [ - for (final media in audioPlayer.playlist.medias) - PlaylistMediaTableCompanion.insert( - playlistId: playlist!.id, - uri: media.uri, - extras: Value(media.extras), - httpHeaders: Value(media.httpHeaders), - ), - ], - ); - }); - } else if (medias.isNotEmpty) { + if (tracks.isEmpty && state.tracks.isNotEmpty) { + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(currentIndex), + ), + ); + } else if (tracks.isNotEmpty) { + state = state.copyWith( + tracks: tracks, + currentIndex: currentIndex, + ); await audioPlayer.openPlaylist( - medias - .map( - (media) => SpotubeMedia.fromMedia( - Media( - media.uri, - extras: media.extras, - httpHeaders: media.httpHeaders, - ), - ), - ) - .toList(), - initialIndex: playlist.index, + tracks.asMediaList(), + initialIndex: currentIndex, autoPlay: false, ); } @@ -109,36 +100,6 @@ class AudioPlayerNotifier extends Notifier { .write(companion); } - Future _updatePlaylist( - Playlist playlist, - ) async { - final database = ref.read(databaseProvider); - - await database.batch((batch) { - batch.update( - database.playlistTable, - PlaylistTableCompanion(index: Value(playlist.index)), - where: (tb) => tb.id.equals(0), - ); - - batch.deleteAll(database.playlistMediaTable); - - if (playlist.medias.isEmpty) return; - batch.insertAll( - database.playlistMediaTable, - [ - for (final media in playlist.medias) - PlaylistMediaTableCompanion.insert( - playlistId: 0, - uri: media.uri, - extras: Value(media.extras), - httpHeaders: Value(media.httpHeaders), - ), - ], - ); - }); - } - @override build() { final subscriptions = [ @@ -183,9 +144,20 @@ class AudioPlayerNotifier extends Notifier { }), audioPlayer.playlistStream.listen((playlist) async { try { - state = state.copyWith(playlist: playlist); + final tracks = + playlist.medias.map((e) => SpotubeMedia.media(e).track).toList(); - await _updatePlaylist(playlist); + state = state.copyWith( + tracks: tracks, + currentIndex: playlist.index, + ); + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + currentIndex: Value(state.currentIndex), + tracks: Value(state.tracks), + ), + ); } catch (e, stack) { AppLogger.reportError(e, stack); } @@ -203,8 +175,8 @@ class AudioPlayerNotifier extends Notifier { return AudioPlayerState( loopMode: audioPlayer.loopMode, playing: audioPlayer.isPlaying, - playlist: audioPlayer.playlist, shuffled: audioPlayer.isShuffled, + tracks: [], collections: [], ); } @@ -245,44 +217,83 @@ class AudioPlayerNotifier extends Notifier { await removeCollections([collectionId]); } - // Tracks related methods - Future addTracksAtFirst( - Iterable tracks, { + Iterable tracks, { bool allowDuplicates = false, }) async { + _assertAllowedTracks(tracks); if (state.tracks.length == 1) { return addTracks(tracks); } - tracks = _blacklist.filter(tracks).toList() as List; + final addableTracks = _blacklist + .filter(tracks) + .where( + (track) => + allowDuplicates || + !state.tracks.any((element) => _compareTracks(element, track)), + ) + .toList(); - for (int i = 0; i < tracks.length; i++) { - final track = tracks.elementAt(i); + state = state.copyWith( + tracks: [...addableTracks, ...state.tracks], + ); - if (!allowDuplicates && - state.tracks.any((element) => _compareTracks(element, track))) { - continue; - } + for (int i = 0; i < addableTracks.length; i++) { + final track = addableTracks.elementAt(i); await audioPlayer.addTrackAt( SpotubeMedia(track), - max(state.playlist.index, 0) + i + 1, + max(state.currentIndex, 0) + i + 1, ); } + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(max(state.currentIndex, 0)), + ), + ); } - Future addTrack(Track track) async { + Future addTrack(SpotubeTrackObject track) async { + _assertAllowedTrack(track); + if (_blacklist.contains(track)) return; if (state.tracks.any((element) => _compareTracks(element, track))) return; + + state = state.copyWith( + tracks: [...state.tracks, track], + ); + await audioPlayer.addTrack(SpotubeMedia(track)); + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(max(state.currentIndex, 0)), + ), + ); } - Future addTracks(Iterable tracks) async { - tracks = _blacklist.filter(tracks).toList() as List; + Future addTracks(Iterable tracks) async { + _assertAllowedTracks(tracks); + + tracks = _blacklist.filter(tracks).toList(); + state = state.copyWith( + tracks: [...state.tracks, ...tracks], + ); + for (final track in tracks) { await audioPlayer.addTrack(SpotubeMedia(track)); } + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(max(state.currentIndex, 0)), + ), + ); } Future removeTrack(String trackId) async { @@ -290,52 +301,135 @@ class AudioPlayerNotifier extends Notifier { if (index == -1) return; + state = state.copyWith( + tracks: List.of(state.tracks)..removeAt(index), + ); + await audioPlayer.removeTrack(index); + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(max(state.currentIndex, 0)), + ), + ); } Future removeTracks(Iterable trackIds) async { - for (final trackId in trackIds) { - await removeTrack(trackId); + final trackIndexes = state.tracks + .where((element) => trackIds.any((trackId) => trackId == element.id)) + .mapIndexed((index, element) => index); + + final tracks = state.tracks.where( + (element) => !trackIds.contains(element.id), + ); + + state = state.copyWith( + tracks: tracks.toList(), + ); + + for (final index in trackIndexes) { + await audioPlayer.removeTrack(index); } + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(max(state.currentIndex, 0)), + ), + ); } - bool _compareTracks(Track a, Track b) { - if ((a is LocalTrack && b is! LocalTrack) || - (a is! LocalTrack && b is LocalTrack)) return false; + bool _compareTracks(SpotubeTrackObject a, SpotubeTrackObject b) { + if (a.runtimeType != b.runtimeType) { + return false; + } - return a is LocalTrack && b is LocalTrack - ? (a).path == (b).path + return a is SpotubeLocalTrackObject && b is SpotubeLocalTrackObject + ? a.path == b.path : a.id == b.id; } Future load( - List tracks, { + List tracks, { int initialIndex = 0, bool autoPlay = false, }) async { - final medias = (_blacklist.filter(tracks).toList() as List) + _assertAllowedTracks(tracks); + + final medias = _blacklist + .filter(tracks) + .toList() .asMediaList() - .unique((a, b) => _compareTracks(a.track, b.track)); + .unique((a, b) => a.uri == b.uri); // Giving the initial track a boost so MediaKit won't skip // because of timeout final intendedActiveTrack = medias.elementAt(initialIndex); - if (intendedActiveTrack.track is! LocalTrack) { - await ref.read(sourcedTrackProvider(intendedActiveTrack).future); + if (intendedActiveTrack.track is! SpotubeLocalTrackObject) { + ref.read( + sourcedTrackProvider( + intendedActiveTrack.track as SpotubeFullTrackObject, + ).future, + ); } if (medias.isEmpty) return; - await removeCollections(state.collections); + state = state.copyWith( + // These are filtered tracks as well + tracks: medias.map((media) => media.track).toList(), + currentIndex: initialIndex, + collections: [], + ); await audioPlayer.openPlaylist( - medias.map((s) => s as Media).toList(), + medias, initialIndex: initialIndex, autoPlay: autoPlay, ); + + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(max(state.currentIndex, 0)), + ), + ); } - Future jumpToTrack(Track track) async { + Future swapActiveSource() async { + if (state.tracks.isEmpty || state.activeTrack is! SpotubeFullTrackObject) { + return; + } + + final oldState = state; + await audioPlayer.stop(); + + await load( + oldState.tracks, + initialIndex: oldState.currentIndex, + autoPlay: true, + ); + state = state.copyWith( + collections: oldState.collections, + loopMode: oldState.loopMode, + playing: oldState.playing, + shuffled: false, + ); + await audioPlayer.setLoopMode(oldState.loopMode); + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: Value(state.currentIndex), + collections: Value(state.collections), + loopMode: Value(state.loopMode), + playing: Value(state.playing), + shuffled: Value(state.shuffled), + ), + ); + } + + Future jumpToTrack(SpotubeTrackObject track) async { final index = state.tracks.toList().indexWhere((element) => element.id == track.id); if (index == -1) return; @@ -347,14 +441,33 @@ class AudioPlayerNotifier extends Notifier { newIndex < 0 || oldIndex < 0 || newIndex > state.tracks.length - 1 || - oldIndex > state.tracks.length - 1) return; + oldIndex > state.tracks.length - 1) { + return; + } await audioPlayer.moveTrack(oldIndex, newIndex); } Future stop() async { + state = state.copyWith( + tracks: [], + currentIndex: 0, + collections: [], + loopMode: PlaylistMode.none, + playing: false, + shuffled: false, + ); await audioPlayer.stop(); - await removeCollections(state.collections); + await _updatePlayerState( + AudioPlayerStateTableCompanion( + tracks: Value(state.tracks), + currentIndex: const Value(0), + collections: const Value([]), + loopMode: const Value(PlaylistMode.none), + playing: const Value(false), + shuffled: const Value(false), + ), + ); ref.read(discordProvider.notifier).clear(); } } diff --git a/lib/provider/audio_player/audio_player_streams.dart b/lib/provider/audio_player/audio_player_streams.dart index e52b6109..eff13134 100644 --- a/lib/provider/audio_player/audio_player_streams.dart +++ b/lib/provider/audio_player/audio_player_streams.dart @@ -1,18 +1,17 @@ import 'dart:async'; +import 'dart:math'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:palette_generator/palette_generator.dart'; -import 'package:spotube/components/image/universal_image.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/models/local_track.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/audio_player/state.dart'; import 'package:spotube/provider/discord_provider.dart'; import 'package:spotube/provider/history/history.dart'; -import 'package:spotube/provider/palette_provider.dart'; +import 'package:spotube/provider/metadata_plugin/core/scrobble.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; import 'package:spotube/provider/skip_segments/skip_segments.dart'; import 'package:spotube/provider/scrobbler/scrobbler.dart'; -import 'package:spotube/provider/server/sourced_track.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/audio_services/audio_services.dart'; @@ -48,36 +47,12 @@ class AudioPlayerStreamListeners { PlaybackHistoryActions get history => ref.read(playbackHistoryActionsProvider); - Future updatePalette() async { - final palette = ref.read(paletteProvider); - if (!preferences.albumColorSync) { - if (palette != null) ref.read(paletteProvider.notifier).state = null; - return; - } - return Future.microtask(() async { - final activeTrack = ref.read(audioPlayerProvider).activeTrack; - if (activeTrack == null) return; - - final palette = await PaletteGenerator.fromImageProvider( - UniversalImage.imageProvider( - (activeTrack.album?.images).asUrlString( - placeholder: ImagePlaceholder.albumArt, - ), - height: 50, - width: 50, - ), - ); - ref.read(paletteProvider.notifier).state = palette; - }); - } - StreamSubscription subscribeToPlaylist() { return audioPlayer.playlistStream.listen((mpvPlaylist) { try { if (audioPlayerState.activeTrack == null) return; notificationService.addTrack(audioPlayerState.activeTrack!); discord.updatePresence(audioPlayerState.activeTrack!); - updatePalette(); } catch (e, stack) { AppLogger.reportError(e, stack); } @@ -90,7 +65,9 @@ class AudioPlayerStreamListeners { final currentSegments = await ref.read(segmentProvider.future); if (currentSegments?.segments.isNotEmpty != true || - position < const Duration(seconds: 3)) return; + position < const Duration(seconds: 3)) { + return; + } for (final segment in currentSegments!.segments) { final seconds = position.inSeconds; @@ -107,21 +84,48 @@ class AudioPlayerStreamListeners { StreamSubscription subscribeToScrobbleChanged() { String? lastScrobbled; - return audioPlayer.positionStream.listen((position) { + return audioPlayer.positionStream.listen((position) async { try { - final uid = audioPlayerState.activeTrack is LocalTrack - ? (audioPlayerState.activeTrack as LocalTrack).path + final uid = audioPlayerState.activeTrack is SpotubeLocalTrackObject + ? (audioPlayerState.activeTrack as SpotubeLocalTrackObject).path : audioPlayerState.activeTrack?.id; + /// According to Listenbrainz and Last.fm, a scrobble should be sent + /// after 4 minutes of listening or 50% of the track duration, + /// whichever is less. + final minimumListenTime = min(audioPlayer.duration.inSeconds ~/ 2, 240); + if (audioPlayerState.activeTrack == null || lastScrobbled == uid || - position.inSeconds < 30) { + position.inSeconds < minimumListenTime || + audioPlayer.duration == Duration.zero || + position == Duration.zero) { return; } scrobbler.scrobble(audioPlayerState.activeTrack!); - history.addTrack(audioPlayerState.activeTrack!); + ref + .read(metadataPluginScrobbleProvider.notifier) + .scrobble(audioPlayerState.activeTrack!); lastScrobbled = uid; + + /// The [Track] from Playlist.getTracks doesn't contain artist images + /// so we need to fetch them from the API + var activeTrack = audioPlayerState.activeTrack!; + if (activeTrack.artists.any((a) => a.images == null)) { + final metadataPlugin = await ref.read(metadataPluginProvider.future); + final artists = await Future.wait( + activeTrack.artists + .map((artist) => metadataPlugin!.artist.getArtist(artist.id)), + ); + activeTrack = activeTrack.copyWith( + artists: artists + .map((e) => SpotubeSimpleArtistObject.fromJson(e.toJson())) + .toList(), + ); + } + + await history.addTrack(activeTrack); } catch (e, stack) { AppLogger.reportError(e, stack); } @@ -131,25 +135,30 @@ class AudioPlayerStreamListeners { StreamSubscription subscribeToPosition() { String lastTrack = ""; // used to prevent multiple calls to the same track return audioPlayer.positionStream.listen((event) async { + final percentProgress = + (event.inSeconds / max(audioPlayer.duration.inSeconds, 1)) * 100; try { - if (event < const Duration(seconds: 3) || - audioPlayerState.playlist.index == -1 || - audioPlayerState.playlist.index == + if (percentProgress < 80 || + audioPlayerState.currentIndex == -1 || + audioPlayerState.currentIndex == audioPlayerState.tracks.length - 1) { return; } - final nextTrack = SpotubeMedia.fromMedia(audioPlayerState - .playlist.medias - .elementAt(audioPlayerState.playlist.index + 1)); + final nextTrack = audioPlayerState.tracks + .elementAtOrNull(audioPlayerState.currentIndex + 1); - if (lastTrack == nextTrack.track.id || nextTrack.track is LocalTrack) { + if (nextTrack == null || + lastTrack == nextTrack.id || + nextTrack is SpotubeLocalTrackObject) { return; } try { - await ref.read(sourcedTrackProvider(nextTrack).future); + await ref.read( + sourcedTrackProvider(nextTrack as SpotubeFullTrackObject).future, + ); } finally { - lastTrack = nextTrack.track.id!; + lastTrack = nextTrack.id; } } catch (e, stack) { AppLogger.reportError(e, stack); diff --git a/lib/provider/audio_player/querying_track_info.dart b/lib/provider/audio_player/querying_track_info.dart index 55590d48..06e9653c 100644 --- a/lib/provider/audio_player/querying_track_info.dart +++ b/lib/provider/audio_player/querying_track_info.dart @@ -1,24 +1,23 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/provider/server/sourced_track.dart'; -import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; final queryingTrackInfoProvider = Provider((ref) { - final media = audioPlayer.playlist.index == -1 || - audioPlayer.playlist.medias.isEmpty - ? null - : audioPlayer.playlist.medias.elementAtOrNull(audioPlayer.playlist.index); - final audioPlayerActiveTrack = - media == null ? null : SpotubeMedia.fromMedia(media); + final audioPlayer = ref.watch(audioPlayerProvider); - final activeMedia = ref.watch(audioPlayerProvider.select( - (s) => s.activeMedia == null - ? null - : SpotubeMedia.fromMedia(s.activeMedia!), - )) ?? - audioPlayerActiveTrack; + if (audioPlayer.activeTrack == null) { + return false; + } - if (activeMedia == null) return false; + if (audioPlayer.activeTrack is! SpotubeFullTrackObject) { + return false; + } - return ref.watch(sourcedTrackProvider(activeMedia)).isLoading; + return ref + .watch( + sourcedTrackProvider( + audioPlayer.activeTrack! as SpotubeFullTrackObject), + ) + .isLoading; }); diff --git a/lib/provider/audio_player/sources/invidious_instances_provider.dart b/lib/provider/audio_player/sources/invidious_instances_provider.dart deleted file mode 100644 index c04ac765..00000000 --- a/lib/provider/audio_player/sources/invidious_instances_provider.dart +++ /dev/null @@ -1,12 +0,0 @@ -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotube/services/sourced_track/sources/invidious.dart'; - -final invidiousInstancesProvider = FutureProvider((ref) async { - final invidious = ref.watch(invidiousProvider); - - final instances = await invidious.instances(); - - return instances - .where((instance) => instance.details.type == "https") - .toList(); -}); diff --git a/lib/provider/audio_player/sources/piped_instances_provider.dart b/lib/provider/audio_player/sources/piped_instances_provider.dart deleted file mode 100644 index 3c5d5f04..00000000 --- a/lib/provider/audio_player/sources/piped_instances_provider.dart +++ /dev/null @@ -1,17 +0,0 @@ -import 'package:spotube/services/logger/logger.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:piped_client/piped_client.dart'; -import 'package:spotube/services/sourced_track/sources/piped.dart'; - -final pipedInstancesFutureProvider = FutureProvider>( - (ref) async { - try { - final pipedClient = ref.watch(pipedProvider); - - return await pipedClient.instanceList(); - } catch (e, stack) { - AppLogger.reportError(e, stack); - return []; - } - }, -); diff --git a/lib/provider/audio_player/state.dart b/lib/provider/audio_player/state.dart index 0e3004f5..d62155f3 100644 --- a/lib/provider/audio_player/state.dart +++ b/lib/provider/audio_player/state.dart @@ -1,105 +1,67 @@ +import 'package:freezed_annotation/freezed_annotation.dart'; import 'package:media_kit/media_kit.dart' hide Track; -import 'package:spotify/spotify.dart' hide Playlist; -import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/models/metadata/metadata.dart'; -class AudioPlayerState { - final bool playing; - final PlaylistMode loopMode; - final bool shuffled; - final Playlist playlist; +part 'state.freezed.dart'; +part 'state.g.dart'; - final List tracks; - final List collections; +@freezed +class AudioPlayerState with _$AudioPlayerState { + const AudioPlayerState._(); - AudioPlayerState({ - required this.playing, - required this.loopMode, - required this.shuffled, - required this.playlist, - required this.collections, - List? tracks, - }) : tracks = tracks ?? - playlist.medias - .map((media) => SpotubeMedia.fromMedia(media).track) - .toList(); + factory AudioPlayerState._inner({ + required bool playing, + required PlaylistMode loopMode, + required bool shuffled, + required List collections, + @Default(0) int currentIndex, + @Default([]) List tracks, + }) = _AudioPlayerState; - factory AudioPlayerState.fromJson(Map json) { - return AudioPlayerState( - playing: json['playing'], - loopMode: PlaylistMode.values.firstWhere( - (e) => e.name == json['loopMode'], - orElse: () => audioPlayer.loopMode, - ), - shuffled: json['shuffled'], - playlist: Playlist( - json['playlist']['medias'] - .map( - (media) => SpotubeMedia.fromMedia(Media( - media['uri'], - extras: media['extras'], - httpHeaders: media['httpHeaders'], - )), - ) - .cast() - .toList(), - index: json['playlist']['index'], - ), - collections: List.from(json['collections']), - ); - } - - Map toJson() { - return { - 'playing': playing, - 'loopMode': loopMode.name, - 'shuffled': shuffled, - 'playlist': { - 'medias': playlist.medias - .map((media) => { - 'uri': media.uri, - 'extras': media.extras, - 'httpHeaders': media.httpHeaders, - }) - .toList(), - 'index': playlist.index, - }, - 'collections': collections, - }; - } - - AudioPlayerState copyWith({ - bool? playing, - PlaylistMode? loopMode, - bool? shuffled, - Playlist? playlist, - List? collections, + factory AudioPlayerState({ + required bool playing, + required PlaylistMode loopMode, + required bool shuffled, + required List collections, + int currentIndex = 0, + List tracks = const [], }) { - return AudioPlayerState( - playing: playing ?? this.playing, - loopMode: loopMode ?? this.loopMode, - shuffled: shuffled ?? this.shuffled, - playlist: playlist ?? this.playlist, - collections: collections ?? this.collections, - tracks: playlist == null ? tracks : null, + assert( + tracks.every((track) => + track is SpotubeFullTrackObject || track is SpotubeLocalTrackObject), + 'All tracks must be either SpotubeFullTrackObject or SpotubeLocalTrackObject', + ); + + return AudioPlayerState._inner( + playing: playing, + loopMode: loopMode, + shuffled: shuffled, + currentIndex: currentIndex, + tracks: tracks, + collections: collections, ); } - Track? get activeTrack { - if (playlist.index == -1) return null; - return tracks.elementAtOrNull(playlist.index); + factory AudioPlayerState.fromJson(Map json) => + _$AudioPlayerStateFromJson(json); + + SpotubeTrackObject? get activeTrack { + if (currentIndex < 0 || currentIndex >= tracks.length) return null; + return tracks[currentIndex]; } - Media? get activeMedia { - if (playlist.index == -1 || playlist.medias.isEmpty) return null; - return playlist.medias.elementAt(playlist.index); + bool containsTrack(SpotubeTrackObject track) { + return tracks.isNotEmpty && + tracks.any( + (t) => + t is SpotubeLocalTrackObject && track is SpotubeLocalTrackObject + ? t.path == track.path + : t.id == track.id, + ); } - bool containsTrack(Track track) { - return tracks.any((t) => t.id == track.id); - } - - bool containsTracks(List tracks) { - return tracks.every(containsTrack); + bool containsTracks(List tracks) { + return this.tracks.isNotEmpty && tracks.every(containsTrack); } bool containsCollection(String collectionId) { diff --git a/lib/provider/audio_player/state.freezed.dart b/lib/provider/audio_player/state.freezed.dart new file mode 100644 index 00000000..0299cd2f --- /dev/null +++ b/lib/provider/audio_player/state.freezed.dart @@ -0,0 +1,297 @@ +// coverage:ignore-file +// GENERATED CODE - DO NOT MODIFY BY HAND +// ignore_for_file: type=lint +// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark + +part of 'state.dart'; + +// ************************************************************************** +// FreezedGenerator +// ************************************************************************** + +T _$identity(T value) => value; + +final _privateConstructorUsedError = UnsupportedError( + 'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); + +AudioPlayerState _$AudioPlayerStateFromJson(Map json) { + return _AudioPlayerState.fromJson(json); +} + +/// @nodoc +mixin _$AudioPlayerState { + bool get playing => throw _privateConstructorUsedError; + PlaylistMode get loopMode => throw _privateConstructorUsedError; + bool get shuffled => throw _privateConstructorUsedError; + List get collections => throw _privateConstructorUsedError; + int get currentIndex => throw _privateConstructorUsedError; + List get tracks => throw _privateConstructorUsedError; + + /// Serializes this AudioPlayerState to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of AudioPlayerState + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $AudioPlayerStateCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $AudioPlayerStateCopyWith<$Res> { + factory $AudioPlayerStateCopyWith( + AudioPlayerState value, $Res Function(AudioPlayerState) then) = + _$AudioPlayerStateCopyWithImpl<$Res, AudioPlayerState>; + @useResult + $Res call( + {bool playing, + PlaylistMode loopMode, + bool shuffled, + List collections, + int currentIndex, + List tracks}); +} + +/// @nodoc +class _$AudioPlayerStateCopyWithImpl<$Res, $Val extends AudioPlayerState> + implements $AudioPlayerStateCopyWith<$Res> { + _$AudioPlayerStateCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of AudioPlayerState + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? playing = null, + Object? loopMode = null, + Object? shuffled = null, + Object? collections = null, + Object? currentIndex = null, + Object? tracks = null, + }) { + return _then(_value.copyWith( + playing: null == playing + ? _value.playing + : playing // ignore: cast_nullable_to_non_nullable + as bool, + loopMode: null == loopMode + ? _value.loopMode + : loopMode // ignore: cast_nullable_to_non_nullable + as PlaylistMode, + shuffled: null == shuffled + ? _value.shuffled + : shuffled // ignore: cast_nullable_to_non_nullable + as bool, + collections: null == collections + ? _value.collections + : collections // ignore: cast_nullable_to_non_nullable + as List, + currentIndex: null == currentIndex + ? _value.currentIndex + : currentIndex // ignore: cast_nullable_to_non_nullable + as int, + tracks: null == tracks + ? _value.tracks + : tracks // ignore: cast_nullable_to_non_nullable + as List, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$AudioPlayerStateImplCopyWith<$Res> + implements $AudioPlayerStateCopyWith<$Res> { + factory _$$AudioPlayerStateImplCopyWith(_$AudioPlayerStateImpl value, + $Res Function(_$AudioPlayerStateImpl) then) = + __$$AudioPlayerStateImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {bool playing, + PlaylistMode loopMode, + bool shuffled, + List collections, + int currentIndex, + List tracks}); +} + +/// @nodoc +class __$$AudioPlayerStateImplCopyWithImpl<$Res> + extends _$AudioPlayerStateCopyWithImpl<$Res, _$AudioPlayerStateImpl> + implements _$$AudioPlayerStateImplCopyWith<$Res> { + __$$AudioPlayerStateImplCopyWithImpl(_$AudioPlayerStateImpl _value, + $Res Function(_$AudioPlayerStateImpl) _then) + : super(_value, _then); + + /// Create a copy of AudioPlayerState + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? playing = null, + Object? loopMode = null, + Object? shuffled = null, + Object? collections = null, + Object? currentIndex = null, + Object? tracks = null, + }) { + return _then(_$AudioPlayerStateImpl( + playing: null == playing + ? _value.playing + : playing // ignore: cast_nullable_to_non_nullable + as bool, + loopMode: null == loopMode + ? _value.loopMode + : loopMode // ignore: cast_nullable_to_non_nullable + as PlaylistMode, + shuffled: null == shuffled + ? _value.shuffled + : shuffled // ignore: cast_nullable_to_non_nullable + as bool, + collections: null == collections + ? _value._collections + : collections // ignore: cast_nullable_to_non_nullable + as List, + currentIndex: null == currentIndex + ? _value.currentIndex + : currentIndex // ignore: cast_nullable_to_non_nullable + as int, + tracks: null == tracks + ? _value._tracks + : tracks // ignore: cast_nullable_to_non_nullable + as List, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$AudioPlayerStateImpl extends _AudioPlayerState { + _$AudioPlayerStateImpl( + {required this.playing, + required this.loopMode, + required this.shuffled, + required final List collections, + this.currentIndex = 0, + final List tracks = const []}) + : _collections = collections, + _tracks = tracks, + super._(); + + factory _$AudioPlayerStateImpl.fromJson(Map json) => + _$$AudioPlayerStateImplFromJson(json); + + @override + final bool playing; + @override + final PlaylistMode loopMode; + @override + final bool shuffled; + final List _collections; + @override + List get collections { + if (_collections is EqualUnmodifiableListView) return _collections; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_collections); + } + + @override + @JsonKey() + final int currentIndex; + final List _tracks; + @override + @JsonKey() + List get tracks { + if (_tracks is EqualUnmodifiableListView) return _tracks; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_tracks); + } + + @override + String toString() { + return 'AudioPlayerState._inner(playing: $playing, loopMode: $loopMode, shuffled: $shuffled, collections: $collections, currentIndex: $currentIndex, tracks: $tracks)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$AudioPlayerStateImpl && + (identical(other.playing, playing) || other.playing == playing) && + (identical(other.loopMode, loopMode) || + other.loopMode == loopMode) && + (identical(other.shuffled, shuffled) || + other.shuffled == shuffled) && + const DeepCollectionEquality() + .equals(other._collections, _collections) && + (identical(other.currentIndex, currentIndex) || + other.currentIndex == currentIndex) && + const DeepCollectionEquality().equals(other._tracks, _tracks)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + playing, + loopMode, + shuffled, + const DeepCollectionEquality().hash(_collections), + currentIndex, + const DeepCollectionEquality().hash(_tracks)); + + /// Create a copy of AudioPlayerState + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$AudioPlayerStateImplCopyWith<_$AudioPlayerStateImpl> get copyWith => + __$$AudioPlayerStateImplCopyWithImpl<_$AudioPlayerStateImpl>( + this, _$identity); + + @override + Map toJson() { + return _$$AudioPlayerStateImplToJson( + this, + ); + } +} + +abstract class _AudioPlayerState extends AudioPlayerState { + factory _AudioPlayerState( + {required final bool playing, + required final PlaylistMode loopMode, + required final bool shuffled, + required final List collections, + final int currentIndex, + final List tracks}) = _$AudioPlayerStateImpl; + _AudioPlayerState._() : super._(); + + factory _AudioPlayerState.fromJson(Map json) = + _$AudioPlayerStateImpl.fromJson; + + @override + bool get playing; + @override + PlaylistMode get loopMode; + @override + bool get shuffled; + @override + List get collections; + @override + int get currentIndex; + @override + List get tracks; + + /// Create a copy of AudioPlayerState + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$AudioPlayerStateImplCopyWith<_$AudioPlayerStateImpl> get copyWith => + throw _privateConstructorUsedError; +} diff --git a/lib/provider/audio_player/state.g.dart b/lib/provider/audio_player/state.g.dart new file mode 100644 index 00000000..de5f6f1c --- /dev/null +++ b/lib/provider/audio_player/state.g.dart @@ -0,0 +1,40 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'state.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +_$AudioPlayerStateImpl _$$AudioPlayerStateImplFromJson(Map json) => + _$AudioPlayerStateImpl( + playing: json['playing'] as bool, + loopMode: $enumDecode(_$PlaylistModeEnumMap, json['loopMode']), + shuffled: json['shuffled'] as bool, + collections: (json['collections'] as List) + .map((e) => e as String) + .toList(), + currentIndex: (json['currentIndex'] as num?)?.toInt() ?? 0, + tracks: (json['tracks'] as List?) + ?.map((e) => SpotubeTrackObject.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + ); + +Map _$$AudioPlayerStateImplToJson( + _$AudioPlayerStateImpl instance) => + { + 'playing': instance.playing, + 'loopMode': _$PlaylistModeEnumMap[instance.loopMode]!, + 'shuffled': instance.shuffled, + 'collections': instance.collections, + 'currentIndex': instance.currentIndex, + 'tracks': instance.tracks.map((e) => e.toJson()).toList(), + }; + +const _$PlaylistModeEnumMap = { + PlaylistMode.none: 'none', + PlaylistMode.single: 'single', + PlaylistMode.loop: 'loop', +}; diff --git a/lib/provider/authentication/authentication.dart b/lib/provider/authentication/authentication.dart deleted file mode 100644 index 05a05972..00000000 --- a/lib/provider/authentication/authentication.dart +++ /dev/null @@ -1,173 +0,0 @@ -import 'dart:async'; -import 'dart:io'; - -import 'package:collection/collection.dart'; -import 'package:desktop_webview_window/desktop_webview_window.dart'; -import 'package:dio/dio.dart'; -import 'package:dio/io.dart'; -import 'package:drift/drift.dart'; -import 'package:flutter_inappwebview/flutter_inappwebview.dart' - hide X509Certificate; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/collections/routes.dart'; -import 'package:spotube/components/dialogs/prompt_dialog.dart'; -import 'package:spotube/extensions/context.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/utils/platform.dart'; - -extension ExpirationAuthenticationTableData on AuthenticationTableData { - bool get isExpired => DateTime.now().isAfter(expiration); - - String? getCookie(String key) => cookie.value - .split("; ") - .firstWhereOrNull((c) => c.trim().startsWith("$key=")) - ?.trim() - .split("=") - .last - .replaceAll(";", ""); -} - -class AuthenticationNotifier extends AsyncNotifier { - static final Dio dio = () { - final dio = Dio(); - - (dio.httpClientAdapter as IOHttpClientAdapter) - .createHttpClient = () => HttpClient() - ..badCertificateCallback = (X509Certificate cert, String host, int port) { - return host.endsWith("spotify.com") && port == 443; - }; - - return dio; - }(); - - @override - build() async { - final database = ref.watch(databaseProvider); - - final data = await (database.select(database.authenticationTable) - ..where((s) => s.id.equals(0))) - .getSingleOrNull(); - - Timer? refreshTimer; - - ref.listenSelf((prevData, newData) async { - if (newData.asData?.value == null) return; - - if (newData.asData!.value!.isExpired) { - await refreshCredentials(); - } - - // set the refresh timer - refreshTimer?.cancel(); - refreshTimer = Timer( - newData.asData!.value!.expiration.difference(DateTime.now()), - () => refreshCredentials(), - ); - }); - - final subscription = - database.select(database.authenticationTable).watch().listen( - (event) { - state = AsyncData(event.isEmpty ? null : event.first); - }, - ); - - ref.onDispose(() { - subscription.cancel(); - refreshTimer?.cancel(); - }); - - return data; - } - - Future refreshCredentials() async { - final database = ref.read(databaseProvider); - final refreshedCredentials = - await credentialsFromCookie(state.asData!.value!.cookie.value); - - await database - .update(database.authenticationTable) - .replace(refreshedCredentials); - } - - Future login(String cookie) async { - final database = ref.read(databaseProvider); - final refreshedCredentials = await credentialsFromCookie(cookie); - - await database - .into(database.authenticationTable) - .insert(refreshedCredentials, mode: InsertMode.replace); - } - - Future credentialsFromCookie( - String cookie, - ) async { - try { - final spDc = cookie - .split("; ") - .firstWhereOrNull((c) => c.trim().startsWith("sp_dc=")) - ?.trim(); - final res = await dio.getUri( - Uri.parse( - "https://open.spotify.com/get_access_token?reason=transport&productType=web_player", - ), - options: Options( - headers: { - "Cookie": spDc ?? "", - "User-Agent": - "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/124.0.0.0 Safari/537.36" - }, - validateStatus: (status) => true, - ), - ); - final body = res.data; - - if ((res.statusCode ?? 500) >= 400) { - throw Exception( - "Failed to get access token: ${body['error'] ?? res.statusMessage}", - ); - } - - return AuthenticationTableCompanion.insert( - id: const Value(0), - cookie: DecryptedText("${res.headers["set-cookie"]?.join(";")}; $spDc"), - accessToken: DecryptedText(body['accessToken']), - expiration: DateTime.fromMillisecondsSinceEpoch( - body['accessTokenExpirationTimestampMs'], - ), - ); - } catch (e) { - if (rootNavigatorKey.currentContext != null) { - showPromptDialog( - context: rootNavigatorKey.currentContext!, - title: rootNavigatorKey.currentContext!.l10n - .error("Authentication Failure"), - message: e.toString(), - cancelText: null, - ); - } - rethrow; - } - } - - Future logout() async { - state = const AsyncData(null); - final database = ref.read(databaseProvider); - await (database.delete(database.authenticationTable) - ..where((s) => s.id.equals(0))) - .go(); - if (kIsMobile) { - WebStorageManager.instance().deleteAllData(); - CookieManager.instance().deleteAllCookies(); - } - if (kIsDesktop) { - await WebviewWindow.clearAll(); - } - } -} - -final authenticationProvider = - AsyncNotifierProvider( - () => AuthenticationNotifier(), -); diff --git a/lib/provider/blacklist_provider.dart b/lib/provider/blacklist_provider.dart index a51d399f..f916c491 100644 --- a/lib/provider/blacklist_provider.dart +++ b/lib/provider/blacklist_provider.dart @@ -1,8 +1,7 @@ import 'package:collection/collection.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/models/current_playlist.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; class BlackListNotifier extends AsyncNotifier> { @@ -34,40 +33,29 @@ class BlackListNotifier extends AsyncNotifier> { .go(); } - bool contains(TrackSimple track) { + bool contains(SpotubeTrackObject track) { final containsTrack = state.asData?.value.any((element) => element.elementId == track.id) ?? false; - final containsTrackArtists = track.artists?.any( - (artist) => - state.asData?.value.any((el) => el.elementId == artist.id) ?? - false, - ) ?? - false; + final containsTrackArtists = track.artists.any( + (artist) => + state.asData?.value.any((el) => el.elementId == artist.id) ?? false, + ); return containsTrack || containsTrackArtists; } - bool containsArtist(ArtistSimple artist) { + bool containsArtist(String artistId) { return state.asData?.value - .any((element) => element.elementId == artist.id) ?? + .any((element) => element.elementId == artistId) ?? false; } /// Filters the non blacklisted tracks from the given [tracks] - Iterable filter(Iterable tracks) { + Iterable filter(Iterable tracks) { return tracks.whereNot(contains).toList(); } - - CurrentPlaylist filterPlaylist(CurrentPlaylist playlist) { - return CurrentPlaylist( - id: playlist.id, - name: playlist.name, - thumbnail: playlist.thumbnail, - tracks: playlist.tracks.where((track) => !contains(track)).toList(), - ); - } } final blacklistProvider = diff --git a/lib/provider/connect/connect.dart b/lib/provider/connect/connect.dart index 000a28af..268b6567 100644 --- a/lib/provider/connect/connect.dart +++ b/lib/provider/connect/connect.dart @@ -1,11 +1,15 @@ import 'dart:convert'; import 'package:media_kit/media_kit.dart' hide Track; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.dart'; +import 'package:spotube/collections/spotube_icons.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/state.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/logger/logger.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart' hide Playlist; import 'package:spotube/models/connect/connect.dart'; import 'package:spotube/provider/connect/clients.dart'; @@ -37,7 +41,8 @@ final queueProvider = StateProvider( playing: audioPlayer.isPlaying, loopMode: audioPlayer.loopMode, shuffled: audioPlayer.isShuffled, - playlist: audioPlayer.playlist, + tracks: [], + currentIndex: 0, collections: [], ), ); @@ -46,15 +51,17 @@ final volumeProvider = StateProvider( (ref) => 1.0, ); -class ConnectNotifier extends AsyncNotifier { +typedef ConnectState = ({WebSocketChannel channel, Stream stream}); + +class ConnectNotifier extends AsyncNotifier { @override build() async { try { - final connectClients = ref.watch(connectClientsProvider); + final connectClients = await ref.watch(connectClientsProvider.future); - if (connectClients.asData?.value.resolvedService == null) return null; + if (connectClients.resolvedService == null) return null; - final service = connectClients.asData!.value.resolvedService!; + final service = connectClients.resolvedService!; AppLogger.log.t( '♾️ Connecting to ${service.name}: ws://${service.host}:${service.port}/ws', @@ -70,7 +77,9 @@ class ConnectNotifier extends AsyncNotifier { '✅ Connected to ${service.name}: ws://${service.host}:${service.port}/ws', ); - final subscription = channel.stream.listen( + final stream = channel.stream.asBroadcastStream(); + + final subscription = stream.listen( (message) { final event = WebSocketEvent.fromJson(jsonDecode(message), (data) => data); @@ -102,6 +111,38 @@ class ConnectNotifier extends AsyncNotifier { event.onVolume((event) { ref.read(volumeProvider.notifier).state = event.data; }); + + event.onError((event) { + if (event.data == "Connection denied") { + ref.read(connectClientsProvider.notifier).clearResolvedService(); + + if (rootNavigatorKey.currentContext?.mounted == true) { + final theme = Theme.of(rootNavigatorKey.currentContext!); + + showToast( + context: rootNavigatorKey.currentContext!, + location: ToastLocation.topRight, + dismissible: true, + builder: (context, overlay) { + return SurfaceCard( + fillColor: theme.colorScheme.destructive, + filled: true, + child: Basic( + leading: const Icon(SpotubeIcons.error), + title: Text( + context.l10n.connection_request_denied, + style: theme.typography.normal.copyWith( + color: theme.colorScheme.destructiveForeground, + ), + ), + leadingAlignment: Alignment.center, + ), + ); + }, + ); + } + } + }); }, onError: (error) { AppLogger.reportError(error, StackTrace.current); @@ -113,7 +154,7 @@ class ConnectNotifier extends AsyncNotifier { channel.sink.close(status.goingAway); }); - return channel; + return (channel: channel, stream: stream); } catch (e, stack) { AppLogger.reportError(e, stack); rethrow; @@ -122,7 +163,7 @@ class ConnectNotifier extends AsyncNotifier { Future emit(Object message) async { if (state.value == null) return; - state.value?.sink.add( + state.value?.channel.sink.add( message is String ? message : (message as dynamic).toJson(), ); } @@ -167,7 +208,7 @@ class ConnectNotifier extends AsyncNotifier { emit(WebSocketLoopEvent(value)); } - Future addTrack(Track data) async { + Future addTrack(SpotubeFullTrackObject data) async { emit(WebSocketAddTrackEvent(data)); } @@ -184,7 +225,6 @@ class ConnectNotifier extends AsyncNotifier { } } -final connectProvider = - AsyncNotifierProvider( +final connectProvider = AsyncNotifierProvider( () => ConnectNotifier(), ); diff --git a/lib/provider/custom_spotify_endpoint_provider.dart b/lib/provider/custom_spotify_endpoint_provider.dart deleted file mode 100644 index ad0c389a..00000000 --- a/lib/provider/custom_spotify_endpoint_provider.dart +++ /dev/null @@ -1,10 +0,0 @@ -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/spotify_provider.dart'; -import 'package:spotube/services/custom_spotify_endpoints/spotify_endpoints.dart'; - -final customSpotifyEndpointProvider = Provider((ref) { - ref.watch(spotifyProvider); - final auth = ref.watch(authenticationProvider); - return CustomSpotifyEndpoints(auth.asData?.value?.accessToken.value ?? ""); -}); diff --git a/lib/provider/discord_provider.dart b/lib/provider/discord_provider.dart index 8f81fc51..fb1c41b1 100644 --- a/lib/provider/discord_provider.dart +++ b/lib/provider/discord_provider.dart @@ -2,8 +2,7 @@ import 'dart:async'; import 'package:flutter_discord_rpc/flutter_discord_rpc.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/extensions/artist_simple.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; @@ -74,29 +73,28 @@ class DiscordNotifier extends AsyncNotifier { } } - Future updatePresence(Track track) async { + Future updatePresence(SpotubeTrackObject track) async { if (!kIsDesktop) return; if (FlutterDiscordRPC.instance.isConnected == false) return; - final artistNames = track.artists?.asString(); + final artistNames = track.artists.asString(); final isPlaying = audioPlayer.isPlaying; final position = audioPlayer.position; await FlutterDiscordRPC.instance.setActivity( activity: RPCActivity( details: track.name, - state: artistNames != null ? "by $artistNames" : null, + state: artistNames, assets: RPCAssets( largeImage: - track.album?.images?.first.url ?? "spotube-logo-foreground", - largeText: track.album?.name ?? "Unknown album", + track.album.images.firstOrNull?.url ?? "spotube-logo-foreground", + largeText: track.album.name, smallImage: "spotube-logo-foreground", smallText: "Spotube", ), buttons: [ RPCButton( - label: "Listen on Spotify", - url: track.externalUrls?.spotify ?? - "https://open.spotify.com/tracks/${track.id}", + label: "Listen on Spotube", + url: track.externalUri, ), ], timestamps: RPCTimestamps( diff --git a/lib/provider/download_manager_provider.dart b/lib/provider/download_manager_provider.dart index 5e9eda20..0ca99ec1 100644 --- a/lib/provider/download_manager_provider.dart +++ b/lib/provider/download_manager_provider.dart @@ -1,237 +1,285 @@ import 'dart:async'; import 'dart:io'; -import 'package:spotube/extensions/track.dart'; -import 'package:spotube/services/logger/logger.dart'; import 'package:collection/collection.dart'; -import 'package:flutter/foundation.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:dio/dio.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:metadata_god/metadata_god.dart'; import 'package:path/path.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/extensions/artist_simple.dart'; -import 'package:spotube/extensions/image.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide join; +import 'package:spotube/collections/routes.dart'; +import 'package:spotube/components/dialogs/replace_downloaded_dialog.dart'; +import 'package:spotube/extensions/dio.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/audio_source/quality_presets.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:spotube/services/download_manager/download_manager.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:spotube/utils/primitive_utils.dart'; +import 'package:spotube/services/logger/logger.dart'; import 'package:spotube/utils/service_utils.dart'; -class DownloadManagerProvider extends ChangeNotifier { - DownloadManagerProvider({required this.ref}) - : $history = {}, - $backHistory = {}, - dl = DownloadManager() { - dl.statusStream.listen((event) async { - try { - final (:request, :status) = event; +enum DownloadStatus { + queued, + downloading, + completed, + failed, + canceled, +} - final track = $history.firstWhereOrNull( - (element) => element.getUrlOfCodec(downloadCodec) == request.url, - ); - if (track == null) return; +class DownloadTask { + final SpotubeFullTrackObject track; + final DownloadStatus status; + final CancelToken cancelToken; + final int? totalSizeBytes; + final StreamController _downloadedBytesStreamController; - final savePath = getTrackFileUrl(track); - // related to onFileExists - final oldFile = File("$savePath.old"); + Stream get downloadedBytesStream => + _downloadedBytesStreamController.stream; - // if download failed and old file exists, rename it back - if ((status == DownloadStatus.failed || - status == DownloadStatus.canceled) && - await oldFile.exists()) { - await oldFile.rename(savePath); - } - if (status != DownloadStatus.completed || - //? WebA audiotagging is not supported yet - //? Although in future by converting weba to opus & then tagging it - //? is possible using vorbis comments - downloadCodec == SourceCodecs.weba) return; + DownloadTask({ + required this.track, + required this.status, + required this.cancelToken, + this.totalSizeBytes, + StreamController? downloadedBytesStreamController, + }) : _downloadedBytesStreamController = + downloadedBytesStreamController ?? StreamController.broadcast(); - final file = File(request.path); - - if (await oldFile.exists()) { - await oldFile.delete(); - } - - final imageBytes = await ServiceUtils.downloadImage( - (track.album?.images).asUrlString( - placeholder: ImagePlaceholder.albumArt, - index: 1, - ), - ); - - final metadata = track.toMetadata( - fileLength: await file.length(), - imageBytes: imageBytes, - ); - - await MetadataGod.writeMetadata( - file: file.path, - metadata: metadata, - ); - } catch (e, stack) { - AppLogger.reportError(e, stack); - } - }); - } - - Future Function(Track track) onFileExists = (Track track) async => true; - - final Ref ref; - - String get downloadDirectory => - ref.read(userPreferencesProvider.select((s) => s.downloadLocation)); - SourceCodecs get downloadCodec => - ref.read(userPreferencesProvider.select((s) => s.downloadMusicCodec)); - - int get $downloadCount => dl - .getAllDownloads() - .where( - (download) => - download.status.value == DownloadStatus.downloading || - download.status.value == DownloadStatus.paused || - download.status.value == DownloadStatus.queued, - ) - .length; - - final Set $history; - // these are the tracks which metadata hasn't been fetched yet - final Set $backHistory; - final DownloadManager dl; - - String getTrackFileUrl(Track track) { - final name = - "${track.name} - ${track.artists?.asString() ?? ""}.${downloadCodec.name}"; - return join(downloadDirectory, PrimitiveUtils.toSafeFileName(name)); - } - - bool isActive(Track track) { - if ($backHistory.contains(track)) return true; - - final sourcedTrack = mapToSourcedTrack(track); - - if (sourcedTrack == null) return false; - - return dl - .getAllDownloads() - .where( - (download) => - download.status.value == DownloadStatus.downloading || - download.status.value == DownloadStatus.paused || - download.status.value == DownloadStatus.queued, - ) - .map((e) => e.request.url) - .contains(sourcedTrack.getUrlOfCodec(downloadCodec)); - } - - /// For singular downloads - Future addToQueue(Track track) async { - final savePath = getTrackFileUrl(track); - - final oldFile = File(savePath); - if (await oldFile.exists() && !await onFileExists(track)) { - return; - } - - if (await oldFile.exists()) { - await oldFile.rename("$savePath.old"); - } - - if (track is SourcedTrack && track.codec == downloadCodec) { - final downloadTask = - await dl.addDownload(track.getUrlOfCodec(downloadCodec), savePath); - if (downloadTask != null) { - $history.add(track); - } - } else { - $backHistory.add(track); - final sourcedTrack = await SourcedTrack.fetchFromTrack( - ref: ref, - track: track, - ).then((d) { - $backHistory.remove(track); - return d; - }); - final downloadTask = await dl.addDownload( - sourcedTrack.getUrlOfCodec(downloadCodec), - savePath, - ); - if (downloadTask != null) { - $history.add(sourcedTrack); - } - } - - notifyListeners(); - } - - Future batchAddToQueue(List tracks) async { - $backHistory.addAll( - tracks.where((element) => element is! SourcedTrack), + DownloadTask copyWith({ + SpotubeFullTrackObject? track, + DownloadStatus? status, + CancelToken? cancelToken, + int? totalSizeBytes, + StreamController? downloadedBytesStreamController, + }) { + return DownloadTask( + track: track ?? this.track, + status: status ?? this.status, + cancelToken: cancelToken ?? this.cancelToken, + totalSizeBytes: totalSizeBytes ?? this.totalSizeBytes, + downloadedBytesStreamController: + downloadedBytesStreamController ?? _downloadedBytesStreamController, ); - notifyListeners(); - for (final track in tracks) { - try { - if (track == tracks.first) { - await addToQueue(track); - } else { - await Future.delayed( - const Duration(seconds: 1), - () => addToQueue(track), - ); - } - } catch (e) { - AppLogger.reportError(e, StackTrace.current); - continue; - } - } - } - - Future removeFromQueue(SourcedTrack track) async { - await dl.removeDownload(track.getUrlOfCodec(downloadCodec)); - $history.remove(track); - } - - Future pause(SourcedTrack track) { - return dl.pauseDownload(track.getUrlOfCodec(downloadCodec)); - } - - Future resume(SourcedTrack track) { - return dl.resumeDownload(track.getUrlOfCodec(downloadCodec)); - } - - Future retry(SourcedTrack track) { - return addToQueue(track); - } - - void cancel(SourcedTrack track) { - dl.cancelDownload(track.getUrlOfCodec(downloadCodec)); - } - - void cancelAll() { - for (final download in dl.getAllDownloads()) { - if (download.status.value == DownloadStatus.completed) continue; - dl.cancelDownload(download.request.url); - } - } - - SourcedTrack? mapToSourcedTrack(Track track) { - if (track is SourcedTrack) { - return track; - } else { - return $history.firstWhereOrNull((element) => element.id == track.id); - } - } - - ValueNotifier? getStatusNotifier(SourcedTrack track) { - return dl.getDownload(track.getUrlOfCodec(downloadCodec))?.status; - } - - ValueNotifier? getProgressNotifier(SourcedTrack track) { - return dl.getDownload(track.getUrlOfCodec(downloadCodec))?.progress; } } -final downloadManagerProvider = ChangeNotifierProvider( - (ref) => DownloadManagerProvider(ref: ref), +class DownloadManagerNotifier extends Notifier> { + final Dio dio; + DownloadManagerNotifier() + : dio = Dio(), + super(); + + @override + build() { + ref.onDispose(() { + for (final task in state) { + if (task.status == DownloadStatus.downloading) { + task.cancelToken.cancel(); + } + task._downloadedBytesStreamController.close(); + } + }); + + return []; + } + + DownloadTask? getTaskByTrackId(String trackId) { + return state.firstWhereOrNull((element) => element.track.id == trackId); + } + + void addToQueue(SpotubeFullTrackObject track) { + if (state.any((element) => element.track.id == track.id)) return; + state = [ + ...state, + DownloadTask( + track: track, + status: DownloadStatus.queued, + cancelToken: CancelToken(), + ), + ]; + + ref.read(sourcedTrackProvider(track)); + + _startDownloading(); // No await should be invoked to avoid stuck UI + } + + void addAllToQueue(List tracks) { + state = [ + ...state, + ...tracks.map((e) => DownloadTask( + track: e, + status: DownloadStatus.queued, + cancelToken: CancelToken(), + )), + ]; + + ref.read(sourcedTrackProvider(tracks.first)); + _startDownloading(); // No await should be invoked to avoid stuck UI + } + + void retry(SpotubeFullTrackObject track) { + if (state.firstWhereOrNull((e) => e.track.id == track.id)?.status + case DownloadStatus.canceled || DownloadStatus.failed) { + _setStatus(track, DownloadStatus.queued); + _startDownloading(); // No await should be invoked to avoid stuck UI + } + } + + void cancel(SpotubeFullTrackObject track) { + if (state.firstWhereOrNull((e) => e.track.id == track.id)?.status == + DownloadStatus.failed) { + return; + } + _setStatus(track, DownloadStatus.canceled); + } + + void clearAll() { + for (final task in state) { + if (task.status == DownloadStatus.downloading) { + task.cancelToken.cancel(); + } + } + state = []; + } + + void _setStatus(SpotubeFullTrackObject track, DownloadStatus status) { + state = state.map((e) { + if (e.track.id == track.id) { + if ((status == DownloadStatus.canceled) && e.cancelToken.isCancelled) { + e.cancelToken.cancel(); + } + + return e.copyWith(status: status); + } + return e; + }).toList(); + } + + bool _isShowingDialog = false; + + Future _shouldReplaceFileOnExist(DownloadTask task) async { + if (rootNavigatorKey.currentContext == null || _isShowingDialog) { + return false; + } + final replaceAll = ref.read(replaceDownloadedFileState); + if (replaceAll != null) return replaceAll; + _isShowingDialog = true; + try { + return await showDialog( + context: rootNavigatorKey.currentContext!, + builder: (context) => ReplaceDownloadedDialog( + track: task.track, + ), + ) ?? + false; + } finally { + _isShowingDialog = false; + } + } + + Future _downloadTrack(DownloadTask task) async { + try { + _setStatus(task.track, DownloadStatus.downloading); + final track = await ref.read(sourcedTrackProvider(task.track).future); + if (task.cancelToken.isCancelled) { + _setStatus(task.track, DownloadStatus.canceled); + } + final presets = ref.read(audioSourcePresetsProvider); + final container = + presets.presets[presets.selectedDownloadingContainerIndex]; + final downloadLocation = ref.read( + userPreferencesProvider.select((value) => value.downloadLocation)); + + final url = track.getUrlOfQuality( + container, + presets.selectedDownloadingQualityIndex, + ); + + if (url == null) { + throw Exception("No download URL found for selected codec"); + } + + final savePath = join( + downloadLocation, + ServiceUtils.sanitizeFilename( + "${track.query.name} - ${track.query.artists.map((e) => e.name).join(", ")}.${container.getFileExtension()}", + ), + ); + + final savePathFile = File(savePath); + if (await savePathFile.exists()) { + // dio automatically replaces the file if it exists so no deletion required + if (!await _shouldReplaceFileOnExist(task)) { + _setStatus(track.query, DownloadStatus.completed); + return; + } + } + + final response = await dio.chunkDownload( + url, + savePath, + cancelToken: task.cancelToken, + onReceiveProgress: (count, total) { + if (task.totalSizeBytes == null) { + state = state.map((e) { + if (e.track.id == track.query.id) { + return e.copyWith(totalSizeBytes: total); + } + return e; + }).toList(); + } + task._downloadedBytesStreamController.add(count); + }, + deleteOnError: true, + fileAccessMode: FileAccessMode.write, + ); + if (response.statusCode != null && response.statusCode! < 400) { + _setStatus(track.query, DownloadStatus.completed); + } else { + _setStatus(track.query, DownloadStatus.failed); + return; + } + + if (container.getFileExtension() == "weba") return; + + final imageBytes = await ServiceUtils.downloadImage( + (task.track.album.images).asUrlString( + placeholder: ImagePlaceholder.albumArt, + index: 1, + ), + ); + await MetadataGod.writeMetadata( + file: savePath, + metadata: task.track.toMetadata( + fileLength: await savePathFile.length(), + imageBytes: imageBytes, + ), + ); + } catch (e, stack) { + if (e is! DioException || e.type != DioExceptionType.cancel) { + _setStatus(task.track, DownloadStatus.failed); + AppLogger.reportError(e, stack); + } + } + } + + Future _startDownloading() async { + for (final task in state) { + if (task.status == DownloadStatus.downloading) return; + + if (task.status == DownloadStatus.queued) { + try { + await _downloadTrack(task); + } finally { + // After completion, check for more queued tasks + // Ignore errors of the prior task to allow next task to complete + await _startDownloading(); + } + } + } + } +} + +final downloadManagerProvider = + NotifierProvider>( + DownloadManagerNotifier.new, ); diff --git a/lib/provider/glance/glance.dart b/lib/provider/glance/glance.dart new file mode 100644 index 00000000..00b6bc38 --- /dev/null +++ b/lib/provider/glance/glance.dart @@ -0,0 +1,174 @@ +import 'dart:convert'; + +import 'package:flutter_cache_manager/flutter_cache_manager.dart'; +import 'package:home_widget/home_widget.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:http/http.dart'; +import 'package:logger/logger.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/server/server.dart'; +import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/services/logger/logger.dart'; +import 'package:spotube/utils/platform.dart'; + +@pragma("vm:entry-point") +Future glanceBackgroundCallback(Uri? data) async { + final logger = Logger(); + try { + if (data == null || + data.host != "playback" || + data.pathSegments.isEmpty || + data.queryParameters["serverAddress"] == null) { + return; + } + + final command = data.pathSegments.first; + final res = await get( + Uri.parse( + "http://${data.queryParameters["serverAddress"]}/playback/$command", + ), + ); + + if (res.statusCode != 200) { + throw Exception("Failed to execute command: $command\nBody: ${res.body}"); + } + } catch (e) { + logger.e("[GlanceBackgroundCallback] $e"); + } +} + +Future _saveWidgetData(String key, T? value) async { + try { + if (!kIsMobile) return null; + + return await HomeWidget.saveWidgetData(key, value); + } catch (e, stack) { + AppLogger.reportError(e, stack); + return null; + } +} + +Future _updateWidget() async { + try { + if (!kIsMobile) return; + + if (kIsAndroid) { + await HomeWidget.updateWidget( + androidName: 'HomePlayerWidgetReceiver', + qualifiedAndroidName: + 'oss.krtirtho.spotube.glance.HomePlayerWidgetReceiver', + ); + } + if (kIsIOS) { + await HomeWidget.updateWidget( + name: 'HomePlayerWidget', + iOSName: 'HomePlayerWidget', + ); + } + } on Exception catch (e, stack) { + AppLogger.reportError(e, stack); + } +} + +Future _sendActiveTrack(SpotubeTrackObject? track) async { + if (track == null) { + await _saveWidgetData("activeTrack", null); + await _updateWidget(); + return; + } + + final jsonTrack = track.toJson(); + + final image = track.album.images.firstOrNull; + final cachedImage = image == null + ? null + : image.url.startsWith("http") + ? (await DefaultCacheManager().getSingleFile(image.url)).path + : image.url; + final data = { + ...jsonTrack, + "album": { + ...jsonTrack["album"], + "images": [ + if (cachedImage != null && image != null) + { + ...image.toJson(), + "path": cachedImage, + } + ] + } + }; + + await _saveWidgetData("activeTrack", jsonEncode(data)); + + await _updateWidget(); +} + +final glanceProvider = Provider((ref) { + final server = ref.read(serverProvider); + final activeTrack = ref.read(audioPlayerProvider).activeTrack; + + server.whenData( + (value) async { + final (:server, :port) = value; + + await _saveWidgetData( + "playbackServerAddress", + "${server.address.host}:$port", + ); + await _updateWidget(); + }, + ); + + _sendActiveTrack(activeTrack); + + ref.listen(serverProvider, (prev, next) async { + next.whenData( + (value) async { + final (:server, :port) = value; + + await _saveWidgetData( + "playbackServerAddress", + "${server.address.host}:$port", + ); + await _updateWidget(); + }, + ); + }); + + ref.listen( + audioPlayerProvider, + (previous, next) async { + try { + if (previous?.activeTrack != next.activeTrack && + next.activeTrack != null) { + await _sendActiveTrack(next.activeTrack); + } + } catch (e, stack) { + AppLogger.reportError(e, stack); + } + }, + ); + + final subscriptions = [ + audioPlayer.playingStream.listen((playing) async { + await _saveWidgetData("isPlaying", playing); + await _updateWidget(); + }), + audioPlayer.positionStream.listen((position) async { + await _saveWidgetData("position", position.inSeconds); + await _updateWidget(); + }), + audioPlayer.durationStream.listen((duration) async { + await _saveWidgetData("duration", duration.inSeconds); + await _updateWidget(); + }), + ]; + + ref.onDispose(() { + for (final subscription in subscriptions) { + subscription.cancel(); + } + }); +}); diff --git a/lib/provider/history/history.dart b/lib/provider/history/history.dart index 0c20a9e5..b83e5db1 100644 --- a/lib/provider/history/history.dart +++ b/lib/provider/history/history.dart @@ -1,6 +1,6 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; class PlaybackHistoryActions { @@ -16,44 +16,54 @@ class PlaybackHistoryActions { }); } - Future addPlaylists(List playlists) async { + Future addPlaylists(List playlists) async { await _batchInsertHistoryEntries([ for (final playlist in playlists) HistoryTableCompanion.insert( type: HistoryEntryType.playlist, - itemId: playlist.id!, + itemId: playlist.id, data: playlist.toJson(), ), ]); } - Future addAlbums(List albums) async { + Future addAlbums(List albums) async { await _batchInsertHistoryEntries([ for (final albums in albums) HistoryTableCompanion.insert( type: HistoryEntryType.album, - itemId: albums.id!, + itemId: albums.id, data: albums.toJson(), ), ]); } - Future addTracks(List tracks) async { + Future addTracks(List tracks) async { + assert( + tracks.every((t) => t.artists.every((a) => a.images != null)), + 'Track artists must have images', + ); + await _batchInsertHistoryEntries([ for (final track in tracks) HistoryTableCompanion.insert( type: HistoryEntryType.track, - itemId: track.id!, + itemId: track.id, data: track.toJson(), ), ]); } - Future addTrack(Track track) async { + Future addTrack(SpotubeTrackObject track) async { + assert( + track.artists.every((a) => a.images != null), + 'Track artists must have images', + ); + await _db.into(_db.historyTable).insert( HistoryTableCompanion.insert( type: HistoryEntryType.track, - itemId: track.id!, + itemId: track.id, data: track.toJson(), ), ); diff --git a/lib/provider/history/recent.dart b/lib/provider/history/recent.dart index ef393a17..1ee2a5d6 100644 --- a/lib/provider/history/recent.dart +++ b/lib/provider/history/recent.dart @@ -1,6 +1,7 @@ -import 'package:collection/collection.dart'; -import 'package:drift/drift.dart'; +import 'dart:convert'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/models/database/database.dart'; import 'package:spotube/provider/database/database.dart'; @@ -9,48 +10,40 @@ class RecentlyPlayedItemNotifier extends AsyncNotifier> { build() async { final database = ref.watch(databaseProvider); - final uniqueItemIds = await (database.selectOnly( - database.historyTable, - distinct: true, - ) - ..addColumns([database.historyTable.itemId, database.historyTable.id]) - ..where( - database.historyTable.type.isInValues([ - HistoryEntryType.playlist, - HistoryEntryType.album, - ]), - ) - ..limit(10) - ..orderBy([ - OrderingTerm( - expression: database.historyTable.createdAt, - mode: OrderingMode.desc, - ), - ])) - .map( - (row) => row.read(database.historyTable.id), - ) - .get() - .then((value) => value.whereNotNull().toList()); - - final query = database.select(database.historyTable) - ..where( - (tbl) => tbl.id.isIn(uniqueItemIds), + final query = database.customSelect( + """ + WITH RankedHistory AS ( + SELECT *, ROW_NUMBER() OVER (PARTITION BY item_id ORDER BY created_at DESC) AS rn + FROM history_table + WHERE type in ('playlist', 'album') ) - ..orderBy([ - (tbl) => OrderingTerm( - expression: tbl.createdAt, - mode: OrderingMode.desc, - ), - ]); + SELECT * + FROM RankedHistory + WHERE rn = 1 + ORDER BY created_at DESC + LIMIT 10 + """, + readsFrom: {database.historyTable}, + ).map((rows) async { + return await rows.map((row) { + final type = row.read('type'); + return HistoryTableData( + id: row.read('id'), + itemId: row.read('item_id'), + type: HistoryEntryType.values.firstWhere((e) => e.name == type), + createdAt: row.read('created_at'), + data: jsonDecode(row.read('data')) as Map, + ); + }); + }); - final subscription = query.watch().listen((event) { - state = AsyncData(event); + final subscription = query.watch().listen((event) async { + state = AsyncData(await Future.wait(event)); }); ref.onDispose(() => subscription.cancel()); - final items = await query.get(); + final items = await Future.wait(await query.get()); return items; } diff --git a/lib/provider/history/summary.dart b/lib/provider/history/summary.dart index 99df4c11..5ced7559 100644 --- a/lib/provider/history/summary.dart +++ b/lib/provider/history/summary.dart @@ -53,7 +53,7 @@ class PlaybackHistorySummaryNotifier database.historyTable.itemId.count(distinct: true); final itemIdCountingCol = database.historyTable.itemId.count(); final durationSumJsonColumn = - database.historyTable.data.jsonExtract(r"$.duration_ms").sum(); + database.historyTable.data.jsonExtract(r"$.durationMs").sum(); final artistCountingCol = database.historyTable.data.jsonExtract(r"$.artists"); diff --git a/lib/provider/history/top/albums.dart b/lib/provider/history/top/albums.dart index b11e62d2..1caad5cd 100644 --- a/lib/provider/history/top/albums.dart +++ b/lib/provider/history/top/albums.dart @@ -3,42 +3,19 @@ import 'dart:convert'; import 'package:collection/collection.dart'; import 'package:drift/drift.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; import 'package:spotube/provider/history/top.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; -typedef PlaybackHistoryAlbum = ({int count, AlbumSimple album}); - -class HistoryTopAlbumsState extends PaginatedState { - HistoryTopAlbumsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - HistoryTopAlbumsState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return HistoryTopAlbumsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} +typedef PlaybackHistoryAlbum = ({int count, SpotubeSimpleAlbumObject album}); class HistoryTopAlbumsNotifier extends FamilyPaginatedAsyncNotifier< - PlaybackHistoryAlbum, HistoryTopAlbumsState, HistoryDuration> { + PlaybackHistoryAlbum, HistoryDuration> { HistoryTopAlbumsNotifier() : super(); - Selectable createAlbumsQuery({int? limit, int? offset}) { + Selectable createAlbumsQuery( + {int? limit, int? offset}) { final database = ref.read(databaseProvider); final duration = switch (arg) { @@ -81,28 +58,28 @@ class HistoryTopAlbumsNotifier extends FamilyPaginatedAsyncNotifier< readsFrom: {database.historyTable}, ).map((row) { final data = row.read('data'); - final album = AlbumSimple.fromJson(jsonDecode(data)); + final album = SpotubeSimpleAlbumObject.fromJson(jsonDecode(data)); return album; }); } @override - fetch(arg, offset, limit) async { + fetch(offset, limit) async { final albumsQuery = createAlbumsQuery(limit: limit, offset: offset); final items = getAlbumsWithCount(await albumsQuery.get()); - return ( + return SpotubePaginationResponseObject( items: items, + limit: limit, hasMore: items.length == limit, - nextOffset: offset + limit, + nextOffset: (offset + limit).toInt(), + total: items.length, ); } @override build(arg) async { - final (items: albums, :hasMore, :nextOffset) = await fetch(arg, 0, 20); - final subscription = createAlbumsQuery().watch().listen((event) { if (state.asData == null) return; state = AsyncData(state.asData!.value.copyWith( @@ -115,18 +92,13 @@ class HistoryTopAlbumsNotifier extends FamilyPaginatedAsyncNotifier< subscription.cancel(); }); - return HistoryTopAlbumsState( - items: albums, - offset: nextOffset, - limit: 20, - hasMore: hasMore, - ); + return await fetch(0, 20); } List getAlbumsWithCount( - List albumsWithTrackAlbums, + List albumsWithTrackAlbums, ) { - return groupBy(albumsWithTrackAlbums, (album) => album.id!) + return groupBy(albumsWithTrackAlbums, (album) => album.id) .entries .map((entry) { return (count: entry.value.length, album: entry.value.first); @@ -137,6 +109,8 @@ class HistoryTopAlbumsNotifier extends FamilyPaginatedAsyncNotifier< } final historyTopAlbumsProvider = AsyncNotifierProviderFamily< - HistoryTopAlbumsNotifier, HistoryTopAlbumsState, HistoryDuration>( + HistoryTopAlbumsNotifier, + SpotubePaginationResponseObject, + HistoryDuration>( () => HistoryTopAlbumsNotifier(), ); diff --git a/lib/provider/history/top/playlists.dart b/lib/provider/history/top/playlists.dart index 19eb3622..1beabb80 100644 --- a/lib/provider/history/top/playlists.dart +++ b/lib/provider/history/top/playlists.dart @@ -1,40 +1,19 @@ import 'package:collection/collection.dart'; import 'package:drift/drift.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; import 'package:spotube/provider/history/top.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; -typedef PlaybackHistoryPlaylist = ({int count, PlaylistSimple playlist}); - -class HistoryTopPlaylistsState extends PaginatedState { - HistoryTopPlaylistsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - HistoryTopPlaylistsState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return HistoryTopPlaylistsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} +typedef PlaybackHistoryPlaylist = ({ + int count, + SpotubeSimplePlaylistObject playlist +}); class HistoryTopPlaylistsNotifier extends FamilyPaginatedAsyncNotifier< - PlaybackHistoryPlaylist, HistoryTopPlaylistsState, HistoryDuration> { + PlaybackHistoryPlaylist, HistoryDuration> { HistoryTopPlaylistsNotifier() : super(); SimpleSelectStatement<$HistoryTableTable, HistoryTableData> @@ -52,22 +31,22 @@ class HistoryTopPlaylistsNotifier extends FamilyPaginatedAsyncNotifier< } @override - fetch(arg, offset, limit) async { + fetch(offset, limit) async { final playlistsQuery = createPlaylistsQuery()..limit(limit, offset: offset); final items = getPlaylistsWithCount(await playlistsQuery.get()); - return ( + return SpotubePaginationResponseObject( items: items, - hasMore: items.length == limit, nextOffset: offset + limit, + total: items.length, + limit: limit, + hasMore: items.length == limit, ); } @override build(arg) async { - final (items: playlists, :hasMore, :nextOffset) = await fetch(arg, 0, 20); - final subscription = createPlaylistsQuery().watch().listen((event) { if (state.asData == null) return; state = AsyncData(state.asData!.value.copyWith( @@ -80,18 +59,13 @@ class HistoryTopPlaylistsNotifier extends FamilyPaginatedAsyncNotifier< subscription.cancel(); }); - return HistoryTopPlaylistsState( - items: playlists, - offset: nextOffset, - limit: 20, - hasMore: hasMore, - ); + return await fetch(0, 20); } List getPlaylistsWithCount( List playlists, ) { - return groupBy(playlists, (playlist) => playlist.playlist!.id!) + return groupBy(playlists, (playlist) => playlist.playlist!.id) .entries .map((entry) { return ( @@ -105,6 +79,8 @@ class HistoryTopPlaylistsNotifier extends FamilyPaginatedAsyncNotifier< } final historyTopPlaylistsProvider = AsyncNotifierProviderFamily< - HistoryTopPlaylistsNotifier, HistoryTopPlaylistsState, HistoryDuration>( + HistoryTopPlaylistsNotifier, + SpotubePaginationResponseObject, + HistoryDuration>( () => HistoryTopPlaylistsNotifier(), ); diff --git a/lib/provider/history/top/tracks.dart b/lib/provider/history/top/tracks.dart index b737d148..5c1dbdbf 100644 --- a/lib/provider/history/top/tracks.dart +++ b/lib/provider/history/top/tracks.dart @@ -1,57 +1,18 @@ import 'package:collection/collection.dart'; import 'package:drift/drift.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; import 'package:spotube/provider/history/top.dart'; -import 'package:spotube/provider/spotify/spotify.dart'; +import 'package:spotube/provider/metadata_plugin/artist/artist.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; -typedef PlaybackHistoryTrack = ({int count, Track track}); -typedef PlaybackHistoryArtist = ({int count, Artist artist}); - -class HistoryTopTracksState extends PaginatedState { - HistoryTopTracksState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - List get artists { - return getArtistsWithCount( - items.expand((e) => e.track.artists ?? []), - ); - } - - List getArtistsWithCount(Iterable artists) { - return groupBy(artists, (artist) => artist.id!) - .entries - .map((entry) { - return (count: entry.value.length, artist: entry.value.first); - }) - .sorted((a, b) => b.count.compareTo(a.count)) - .toList(); - } - - @override - HistoryTopTracksState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return HistoryTopTracksState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} +typedef PlaybackHistoryTrack = ({int count, SpotubeTrackObject track}); +typedef PlaybackHistoryArtist = ({int count, SpotubeSimpleArtistObject artist}); class HistoryTopTracksNotifier extends FamilyPaginatedAsyncNotifier< - PlaybackHistoryTrack, HistoryTopTracksState, HistoryDuration> { + PlaybackHistoryTrack, HistoryDuration> { HistoryTopTracksNotifier() : super(); SimpleSelectStatement<$HistoryTableTable, HistoryTableData> @@ -85,23 +46,80 @@ class HistoryTopTracksNotifier extends FamilyPaginatedAsyncNotifier< ); } + Future fixImageNotLoadingForArtistIssue( + List entries, + ) async { + final nonImageArtistTracks = + entries.where((e) => e.track!.artists.any((a) => a.images == null)); + + if (nonImageArtistTracks.isEmpty) return; + + final artistIds = nonImageArtistTracks + .map((e) => e.track!.artists.map((a) => a.id)) + .expand((e) => e) + .toSet() + .toList(); + + if (artistIds.isEmpty) return; + + final artists = await Future.wait([ + for (final id in artistIds) + ref.read(metadataPluginArtistProvider(id).future), + ]); + + final imagedArtistTracks = nonImageArtistTracks.map((e) { + var track = e.track!; + final includedArtists = track.artists + .map((a) { + final fullArtist = + artists.firstWhereOrNull((artist) => artist.id == a.id); + + return fullArtist != null + ? a.copyWith(images: fullArtist.images) + : a; + }) + .nonNulls + .toList(); + + track = track.copyWith(artists: includedArtists); + + return e.copyWith(data: track.toJson()); + }); + + assert( + imagedArtistTracks + .every((e) => e.track!.artists.every((a) => a.images != null)), + 'Tracks artists should have images', + ); + + final database = ref.read(databaseProvider); + await database.batch((batch) { + batch.insertAllOnConflictUpdate( + database.historyTable, + imagedArtistTracks, + ); + }); + } + @override - fetch(arg, offset, limit) async { + fetch(offset, limit) async { final tracksQuery = createTracksQuery()..limit(limit, offset: offset); - final items = getTracksWithCount(await tracksQuery.get()); + final entries = await tracksQuery.get(); - return ( + final items = getTracksWithCount(entries); + + return SpotubePaginationResponseObject( items: items, - hasMore: items.length == limit, nextOffset: offset + limit, + total: items.length, + limit: limit, + hasMore: items.length == limit, ); } @override build(arg) async { - final (items: tracks, :hasMore, :nextOffset) = await fetch(arg, 0, 20); - final subscription = createTracksQuery().watch().listen((event) { if (state.asData == null) return; state = AsyncData(state.asData!.value.copyWith( @@ -114,22 +132,56 @@ class HistoryTopTracksNotifier extends FamilyPaginatedAsyncNotifier< subscription.cancel(); }); - return HistoryTopTracksState( - items: tracks, - offset: nextOffset, - limit: 20, - hasMore: hasMore, + return await fetch(0, 20); + } + + List get artists { + return getArtistsWithCount( + state.asData?.value.items.expand((e) => e.track.artists) ?? [], ); } + List getArtistsWithCount( + Iterable artists, + ) { + return groupBy(artists, (artist) => artist.id) + .entries + .map((entry) { + return ( + count: entry.value.length, + + /// Previously, due to a bug, artist images were not being saved. + /// Now it's fixed, but we need to handle the case where images are null. + /// So we take the first artist with images if available, otherwise the first one. + artist: entry.value.firstWhereOrNull((a) => a.images != null) ?? + entry.value.first, + ); + }) + .sorted((a, b) => b.count.compareTo(a.count)) + .toList(); + } + List getTracksWithCount(List tracks) { + fixImageNotLoadingForArtistIssue(tracks); + return groupBy( tracks, - (track) => track.track!.id!, + (track) => track.track!.id, ) .entries .map((entry) { - return (count: entry.value.length, track: entry.value.first.track!); + return ( + count: entry.value.length, + + /// Previously, due to a bug, artist images were not being saved. + /// Now it's fixed, but we need to handle the case where images are null. + /// So we take the first artist with images if available, otherwise the first one. + track: entry.value + .firstWhereOrNull( + (t) => t.track!.artists.every((a) => a.images != null)) + ?.track! ?? + entry.value.first.track!, + ); }) .sorted((a, b) => b.count.compareTo(a.count)) .toList(); @@ -137,6 +189,8 @@ class HistoryTopTracksNotifier extends FamilyPaginatedAsyncNotifier< } final historyTopTracksProvider = AsyncNotifierProviderFamily< - HistoryTopTracksNotifier, HistoryTopTracksState, HistoryDuration>( + HistoryTopTracksNotifier, + SpotubePaginationResponseObject, + HistoryDuration>( () => HistoryTopTracksNotifier(), ); diff --git a/lib/provider/local_tracks/local_tracks_provider.dart b/lib/provider/local_tracks/local_tracks_provider.dart index 3245ff2d..8d44b607 100644 --- a/lib/provider/local_tracks/local_tracks_provider.dart +++ b/lib/provider/local_tracks/local_tracks_provider.dart @@ -1,7 +1,7 @@ import 'dart:async'; import 'dart:io'; -import 'package:collection/collection.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/services/logger/logger.dart'; import 'package:flutter/foundation.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; @@ -10,12 +10,10 @@ import 'package:mime/mime.dart'; import 'package:path/path.dart'; import 'package:path_provider/path_provider.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/extensions/track.dart'; -import 'package:spotube/models/local_track.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; // ignore: depend_on_referenced_packages import 'package:flutter_rust_bridge/flutter_rust_bridge.dart' show FrbException; +import 'package:spotube/utils/service_utils.dart'; const supportedAudioTypes = [ "audio/webm", @@ -25,6 +23,9 @@ const supportedAudioTypes = [ "audio/opus", "audio/wav", "audio/aac", + "audio/flac", + "audio/x-flac", + "audio/x-wav", ]; const imgMimeToExt = { @@ -34,15 +35,26 @@ const imgMimeToExt = { "image/gif": ".gif", }; +typedef MetadataFile = ({ + Metadata? metadata, + File file, + String? art, +}); + final localTracksProvider = - FutureProvider>>((ref) async { + FutureProvider>>((ref) async { try { if (kIsWeb) return {}; - final Map> libraryToTracks = {}; + final Map> libraryToTracks = {}; final downloadLocation = ref.watch( userPreferencesProvider.select((s) => s.downloadLocation), ); + + if (downloadLocation.isEmpty) { + return {}; + } + final downloadDir = Directory(downloadLocation); final cacheDir = Directory(await UserPreferencesNotifier.getMusicCacheDir()); @@ -69,30 +81,34 @@ final localTracksProvider = await Directory(location).list(recursive: true).toList(); entities.addAll( - dirEntities - .where( - (e) => - e is File && - supportedAudioTypes.contains(lookupMimeType(e.path)), - ) - .cast(), + dirEntities.where( + (e) { + final mime = lookupMimeType(e.path) ?? + (extension(e.path) == ".opus" ? "audio/opus" : null); + + return e is File && supportedAudioTypes.contains(mime); + }, + ).cast(), ); } catch (e, stack) { AppLogger.reportError(e, stack); } } - final List> filesWithMetadata = await Future.wait( + final List filesWithMetadata = await Future.wait( entities.map((file) async { try { final metadata = await MetadataGod.readMetadata(file: file.path); - final imageFile = File(join( - (await getTemporaryDirectory()).path, - "spotube", - basenameWithoutExtension(file.path) + - imgMimeToExt[metadata.picture?.mimeType ?? "image/jpeg"]!, - )); + final imageFile = File( + join( + (await getTemporaryDirectory()).path, + "spotube", + ServiceUtils.sanitizeFilename( + basenameWithoutExtension(file.path)) + + imgMimeToExt[metadata.picture?.mimeType ?? "image/jpeg"]!, + ), + ); if (!await imageFile.exists() && metadata.picture != null) { await imageFile.create(recursive: true); await imageFile.writeAsBytes( @@ -101,27 +117,24 @@ final localTracksProvider = ); } - return {"metadata": metadata, "file": file, "art": imageFile.path}; + return (metadata: metadata, file: file, art: imageFile.path); } catch (e, stack) { if (e case FrbException() || TimeoutException()) { - return {"file": file}; + return (file: file, metadata: null, art: null); } AppLogger.reportError(e, stack); return null; } }), - ).then((value) => value.whereNotNull().toList()); + ).then((value) => value.nonNulls.toList()); final tracksFromMetadata = filesWithMetadata .map( - (fileWithMetadata) => LocalTrack.fromTrack( - track: Track().fromFile( - fileWithMetadata["file"], - metadata: fileWithMetadata["metadata"], - art: fileWithMetadata["art"], - ), - path: fileWithMetadata["file"].path, - ), + (fileWithMetadata) => SpotubeTrackObject.localTrackFromFile( + fileWithMetadata.file, + metadata: fileWithMetadata.metadata, + art: fileWithMetadata.art, + ) as SpotubeLocalTrackObject, ) .toList(); diff --git a/lib/provider/logs/logs_provider.dart b/lib/provider/logs/logs_provider.dart index b0e95cae..d39059ac 100644 --- a/lib/provider/logs/logs_provider.dart +++ b/lib/provider/logs/logs_provider.dart @@ -5,7 +5,14 @@ import 'package:spotube/services/logger/logger.dart'; final logsProvider = StreamProvider.autoDispose((ref) async* { final file = await AppLogger.getLogsPath(); + // Check if file is empty or non-existent + + if (await file.length() == 0) { + throw StateError("Logs file is empty or non-existent"); + } + final stream = file.openRead().transform(utf8.decoder); + await for (final line in stream) { yield line; } diff --git a/lib/provider/spotify/lyrics/synced.dart b/lib/provider/lyrics/synced.dart similarity index 58% rename from lib/provider/spotify/lyrics/synced.dart rename to lib/provider/lyrics/synced.dart index c6c0d6e3..de34005a 100644 --- a/lib/provider/spotify/lyrics/synced.dart +++ b/lib/provider/lyrics/synced.dart @@ -1,54 +1,20 @@ -part of '../spotify.dart'; +import 'dart:async'; -class SyncedLyricsNotifier extends FamilyAsyncNotifier { - Track get _track => arg!; +import 'package:dio/dio.dart'; +import 'package:drift/drift.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:lrc/lrc.dart'; +import 'package:package_info_plus/package_info_plus.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/lyrics.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/database/database.dart'; +import 'package:spotube/services/dio/dio.dart'; +import 'package:spotube/services/logger/logger.dart'; - Future getSpotifyLyrics(String? token) async { - final res = await globalDio.getUri( - Uri.parse( - "https://spclient.wg.spotify.com/color-lyrics/v2/track/${_track.id}?format=json&market=from_token", - ), - options: Options( - headers: { - "User-Agent": - "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.0.0 Safari/537.36", - "App-platform": "WebPlayer", - "authorization": "Bearer $token" - }, - responseType: ResponseType.json, - validateStatus: (status) => true, - ), - ); - - if (res.statusCode != 200) { - return SubtitleSimple( - lyrics: [], - name: _track.name!, - uri: res.realUri, - rating: 0, - provider: "Spotify", - ); - } - final linesRaw = - Map.castFrom(res.data)["lyrics"] - ?["lines"] as List?; - - final lines = linesRaw?.map((line) { - return LyricSlice( - time: Duration(milliseconds: int.parse(line["startTimeMs"])), - text: line["words"] as String, - ); - }).toList() ?? - []; - - return SubtitleSimple( - lyrics: lines, - name: _track.name!, - uri: res.realUri, - rating: 100, - provider: "Spotify", - ); - } +class SyncedLyricsNotifier + extends FamilyAsyncNotifier { + SpotubeTrackObject get _track => arg!; /// Lyrics credits: [lrclib.net](https://lrclib.net) and their contributors /// Thanks for their generous public API @@ -61,10 +27,10 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { host: "lrclib.net", path: "/api/get", queryParameters: { - "artist_name": _track.artists?.first.name, + "artist_name": _track.artists.first.name, "track_name": _track.name, - "album_name": _track.album?.name, - "duration": _track.duration?.inSeconds.toString(), + "album_name": _track.album.name, + "duration": (_track.durationMs / 1000).toInt().toString(), }, ), options: Options( @@ -79,7 +45,7 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { if (res.statusCode != 200) { return SubtitleSimple( lyrics: [], - name: _track.name!, + name: _track.name, uri: res.realUri, rating: 0, provider: "LRCLib", @@ -99,7 +65,7 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { if (syncedLyrics?.isNotEmpty == true) { return SubtitleSimple( lyrics: syncedLyrics!, - name: _track.name!, + name: _track.name, uri: res.realUri, rating: 100, provider: "LRCLib", @@ -113,7 +79,7 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { return SubtitleSimple( lyrics: plainLyrics, - name: _track.name!, + name: _track.name, uri: res.realUri, rating: 0, provider: "LRCLib", @@ -124,26 +90,18 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { FutureOr build(track) async { try { final database = ref.watch(databaseProvider); - final spotify = ref.watch(spotifyProvider); - final auth = await ref.watch(authenticationProvider.future); if (track == null) { throw "No track currently"; } final cachedLyrics = await (database.select(database.lyricsTable) - ..where((tbl) => tbl.trackId.equals(track.id!))) + ..where((tbl) => tbl.trackId.equals(track.id))) .map((row) => row.data) .getSingleOrNull(); SubtitleSimple? lyrics = cachedLyrics; - final token = await spotify.getCredentials(); - - if ((lyrics == null || lyrics.lyrics.isEmpty) && auth != null) { - lyrics = await getSpotifyLyrics(token.accessToken); - } - if (lyrics == null || lyrics.lyrics.isEmpty || lyrics.lyrics.length <= 5) { @@ -157,7 +115,7 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { if (cachedLyrics == null || cachedLyrics.lyrics.isEmpty) { await database.into(database.lyricsTable).insert( LyricsTableCompanion.insert( - trackId: track.id!, + trackId: track.id, data: lyrics, ), mode: InsertMode.replace, @@ -174,13 +132,13 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier { final syncedLyricsDelayProvider = StateProvider((ref) => 0); -final syncedLyricsProvider = - AsyncNotifierProviderFamily( +final syncedLyricsProvider = AsyncNotifierProviderFamily( () => SyncedLyricsNotifier(), ); final syncedLyricsMapProvider = - FutureProvider.family((ref, Track? track) async { + FutureProvider.family((ref, SpotubeTrackObject? track) async { final syncedLyrics = await ref.watch(syncedLyricsProvider(track).future); final isStaticLyrics = diff --git a/lib/provider/metadata_plugin/album/album.dart b/lib/provider/metadata_plugin/album/album.dart new file mode 100644 index 00000000..394f6eb0 --- /dev/null +++ b/lib/provider/metadata_plugin/album/album.dart @@ -0,0 +1,20 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +final metadataPluginAlbumProvider = + FutureProvider.autoDispose.family( + (ref, id) async { + ref.cacheFor(); + + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + return metadataPlugin.album.getAlbum(id); + }, +); diff --git a/lib/provider/metadata_plugin/album/releases.dart b/lib/provider/metadata_plugin/album/releases.dart new file mode 100644 index 00000000..e6e88baf --- /dev/null +++ b/lib/provider/metadata_plugin/album/releases.dart @@ -0,0 +1,29 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; + +class MetadataPluginAlbumReleasesNotifier + extends PaginatedAsyncNotifier { + @override + Future> fetch( + int offset, + int limit, + ) async { + return await (await metadataPlugin) + .album + .releases(limit: limit, offset: offset); + } + + @override + build() async { + ref.watch(metadataPluginAuthenticatedProvider); + return await fetch(0, 20); + } +} + +final metadataPluginAlbumReleasesProvider = AsyncNotifierProvider< + MetadataPluginAlbumReleasesNotifier, + SpotubePaginationResponseObject>( + () => MetadataPluginAlbumReleasesNotifier(), +); diff --git a/lib/provider/metadata_plugin/artist/albums.dart b/lib/provider/metadata_plugin/artist/albums.dart new file mode 100644 index 00000000..0f582bf9 --- /dev/null +++ b/lib/provider/metadata_plugin/artist/albums.dart @@ -0,0 +1,32 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginArtistAlbumNotifier + extends FamilyPaginatedAsyncNotifier { + @override + Future> fetch( + int offset, + int limit, + ) async { + return await (await metadataPlugin).artist.albums( + arg, + limit: limit, + offset: offset, + ); + } + + @override + build(arg) async { + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginArtistAlbumsProvider = AsyncNotifierProviderFamily< + MetadataPluginArtistAlbumNotifier, + SpotubePaginationResponseObject, + String>( + () => MetadataPluginArtistAlbumNotifier(), +); diff --git a/lib/provider/metadata_plugin/artist/artist.dart b/lib/provider/metadata_plugin/artist/artist.dart new file mode 100644 index 00000000..e66309d4 --- /dev/null +++ b/lib/provider/metadata_plugin/artist/artist.dart @@ -0,0 +1,20 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +final metadataPluginArtistProvider = + FutureProvider.autoDispose.family( + (ref, artistId) async { + ref.cacheFor(); + + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + return metadataPlugin.artist.getArtist(artistId); + }, +); diff --git a/lib/provider/metadata_plugin/artist/related.dart b/lib/provider/metadata_plugin/artist/related.dart new file mode 100644 index 00000000..c6a80f75 --- /dev/null +++ b/lib/provider/metadata_plugin/artist/related.dart @@ -0,0 +1,32 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginArtistRelatedArtistsNotifier + extends FamilyPaginatedAsyncNotifier { + @override + Future> fetch( + int offset, + int limit, + ) async { + return await (await metadataPlugin).artist.related( + arg, + limit: limit, + offset: offset, + ); + } + + @override + build(arg) async { + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginArtistRelatedArtistsProvider = AsyncNotifierProviderFamily< + MetadataPluginArtistRelatedArtistsNotifier, + SpotubePaginationResponseObject, + String>( + () => MetadataPluginArtistRelatedArtistsNotifier(), +); diff --git a/lib/provider/metadata_plugin/artist/top_tracks.dart b/lib/provider/metadata_plugin/artist/top_tracks.dart new file mode 100644 index 00000000..c622a738 --- /dev/null +++ b/lib/provider/metadata_plugin/artist/top_tracks.dart @@ -0,0 +1,38 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; + +class MetadataPluginArtistTopTracksNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginArtistTopTracksNotifier() : super(); + + @override + fetch(offset, limit) async { + final tracks = await (await metadataPlugin).artist.topTracks( + arg, + offset: offset, + limit: limit, + ); + + return tracks; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginArtistTopTracksProvider = + AutoDisposeAsyncNotifierProviderFamily< + MetadataPluginArtistTopTracksNotifier, + SpotubePaginationResponseObject, + String>( + () => MetadataPluginArtistTopTracksNotifier(), +); diff --git a/lib/provider/metadata_plugin/artist/wikipedia.dart b/lib/provider/metadata_plugin/artist/wikipedia.dart new file mode 100644 index 00000000..81fcc77c --- /dev/null +++ b/lib/provider/metadata_plugin/artist/wikipedia.dart @@ -0,0 +1,18 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/services/wikipedia/wikipedia.dart'; +import 'package:wikipedia_api/wikipedia_api.dart'; + +final artistWikipediaSummaryProvider = + FutureProvider.autoDispose.family( + (ref, artist) async { + final query = artist.name.replaceAll(" ", "_"); + final res = await wikipedia.pageContent.pageSummaryTitleGet(query); + + if (res?.type != "standard") { + return await wikipedia.pageContent + .pageSummaryTitleGet("${query}_(singer)"); + } + return res; + }, +); diff --git a/lib/provider/metadata_plugin/audio_source/quality_label.dart b/lib/provider/metadata_plugin/audio_source/quality_label.dart new file mode 100644 index 00000000..113ed54e --- /dev/null +++ b/lib/provider/metadata_plugin/audio_source/quality_label.dart @@ -0,0 +1,12 @@ +import 'package:riverpod/riverpod.dart'; +import 'package:spotube/provider/metadata_plugin/audio_source/quality_presets.dart'; + +final audioSourceQualityLabelProvider = Provider((ref) { + final sourceQuality = ref.watch(audioSourcePresetsProvider); + final sourceContainer = sourceQuality.presets + .elementAtOrNull(sourceQuality.selectedStreamingContainerIndex); + final quality = sourceContainer?.qualities + .elementAtOrNull(sourceQuality.selectedStreamingQualityIndex); + + return "${sourceContainer?.name ?? "Unknown"} • ${quality?.toString() ?? "Unknown"}"; +}); diff --git a/lib/provider/metadata_plugin/audio_source/quality_presets.dart b/lib/provider/metadata_plugin/audio_source/quality_presets.dart new file mode 100644 index 00000000..ba88fed6 --- /dev/null +++ b/lib/provider/metadata_plugin/audio_source/quality_presets.dart @@ -0,0 +1,132 @@ +import 'dart:convert'; + +import 'package:freezed_annotation/freezed_annotation.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/audio_player/audio_player.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; +import 'package:spotube/services/metadata/metadata.dart'; + +part 'quality_presets.g.dart'; +part 'quality_presets.freezed.dart'; + +@freezed +class AudioSourcePresetsState with _$AudioSourcePresetsState { + factory AudioSourcePresetsState({ + @Default([]) final List presets, + @Default(0) final int selectedStreamingQualityIndex, + @Default(0) final int selectedStreamingContainerIndex, + @Default(0) final int selectedDownloadingQualityIndex, + @Default(0) final int selectedDownloadingContainerIndex, + }) = _AudioSourcePresetsState; + + factory AudioSourcePresetsState.fromJson(Map json) => + _$AudioSourcePresetsStateFromJson(json); +} + +class AudioSourceAvailableQualityPresetsNotifier + extends Notifier { + @override + build() { + final audioSourceSnapshot = ref.watch(audioSourcePluginProvider); + final audioSourceConfigSnapshot = ref.watch( + metadataPluginsProvider.select((data) => + data.whenData((value) => value.defaultAudioSourcePluginConfig)), + ); + + _initialize(audioSourceSnapshot, audioSourceConfigSnapshot); + + listenSelf((previous, next) { + final isNewLossless = + next.presets.elementAtOrNull(next.selectedStreamingContainerIndex) + is SpotubeAudioSourceContainerPresetLossless; + final isOldLossless = previous?.presets + .elementAtOrNull(previous.selectedStreamingContainerIndex) + is SpotubeAudioSourceContainerPresetLossless; + if (!isOldLossless && isNewLossless) { + audioPlayer.setDemuxerBufferSize(6 * 1024 * 1024); // 6MB + } else if (isOldLossless && !isNewLossless) { + audioPlayer.setDemuxerBufferSize(4 * 1024 * 1024); // 4MB + } + }); + + return AudioSourcePresetsState(); + } + + void _initialize( + AsyncValue audioSourceSnapshot, + AsyncValue audioSourceConfigSnapshot, + ) async { + audioSourceConfigSnapshot.whenData((audioSourceConfig) { + audioSourceSnapshot.whenData((audioSource) async { + if (audioSource == null || audioSourceConfig == null) { + throw MetadataPluginException.noDefaultAudioSourcePlugin(); + } + final preferences = await SharedPreferences.getInstance(); + final persistedStateStr = + preferences.getString("audioSourceState-${audioSourceConfig.slug}"); + + if (persistedStateStr != null) { + state = + AudioSourcePresetsState.fromJson(jsonDecode(persistedStateStr)) + .copyWith( + presets: audioSource.audioSource.supportedPresets, + ); + } else { + state = AudioSourcePresetsState( + presets: audioSource.audioSource.supportedPresets, + ); + } + }); + }); + } + + void setSelectedStreamingContainerIndex(int index) { + state = state.copyWith( + selectedStreamingContainerIndex: index, + selectedStreamingQualityIndex: + 0, // Resetting both because it's a different quality + ); + _updatePreferences(); + } + + void setSelectedStreamingQualityIndex(int index) { + state = state.copyWith(selectedStreamingQualityIndex: index); + _updatePreferences(); + } + + void setSelectedDownloadingContainerIndex(int index) { + state = state.copyWith( + selectedDownloadingContainerIndex: index, + selectedDownloadingQualityIndex: + 0, // Resetting both because it's a different quality + ); + _updatePreferences(); + } + + void setSelectedDownloadingQualityIndex(int index) { + state = state.copyWith(selectedDownloadingQualityIndex: index); + _updatePreferences(); + } + + void _updatePreferences() async { + final audioSourceConfig = await ref.read(metadataPluginsProvider + .selectAsync((data) => data.defaultAudioSourcePluginConfig)); + if (audioSourceConfig == null) { + throw MetadataPluginException.noDefaultAudioSourcePlugin(); + } + + final preferences = await SharedPreferences.getInstance(); + await preferences.setString( + "audioSourceState-${audioSourceConfig.slug}", + jsonEncode(state), + ); + } +} + +final audioSourcePresetsProvider = NotifierProvider< + AudioSourceAvailableQualityPresetsNotifier, AudioSourcePresetsState>( + () => AudioSourceAvailableQualityPresetsNotifier(), +); diff --git a/lib/provider/metadata_plugin/audio_source/quality_presets.freezed.dart b/lib/provider/metadata_plugin/audio_source/quality_presets.freezed.dart new file mode 100644 index 00000000..a8e0c9f7 --- /dev/null +++ b/lib/provider/metadata_plugin/audio_source/quality_presets.freezed.dart @@ -0,0 +1,289 @@ +// coverage:ignore-file +// GENERATED CODE - DO NOT MODIFY BY HAND +// ignore_for_file: type=lint +// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark + +part of 'quality_presets.dart'; + +// ************************************************************************** +// FreezedGenerator +// ************************************************************************** + +T _$identity(T value) => value; + +final _privateConstructorUsedError = UnsupportedError( + 'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); + +AudioSourcePresetsState _$AudioSourcePresetsStateFromJson( + Map json) { + return _AudioSourcePresetsState.fromJson(json); +} + +/// @nodoc +mixin _$AudioSourcePresetsState { + List get presets => + throw _privateConstructorUsedError; + int get selectedStreamingQualityIndex => throw _privateConstructorUsedError; + int get selectedStreamingContainerIndex => throw _privateConstructorUsedError; + int get selectedDownloadingQualityIndex => throw _privateConstructorUsedError; + int get selectedDownloadingContainerIndex => + throw _privateConstructorUsedError; + + /// Serializes this AudioSourcePresetsState to a JSON map. + Map toJson() => throw _privateConstructorUsedError; + + /// Create a copy of AudioSourcePresetsState + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + $AudioSourcePresetsStateCopyWith get copyWith => + throw _privateConstructorUsedError; +} + +/// @nodoc +abstract class $AudioSourcePresetsStateCopyWith<$Res> { + factory $AudioSourcePresetsStateCopyWith(AudioSourcePresetsState value, + $Res Function(AudioSourcePresetsState) then) = + _$AudioSourcePresetsStateCopyWithImpl<$Res, AudioSourcePresetsState>; + @useResult + $Res call( + {List presets, + int selectedStreamingQualityIndex, + int selectedStreamingContainerIndex, + int selectedDownloadingQualityIndex, + int selectedDownloadingContainerIndex}); +} + +/// @nodoc +class _$AudioSourcePresetsStateCopyWithImpl<$Res, + $Val extends AudioSourcePresetsState> + implements $AudioSourcePresetsStateCopyWith<$Res> { + _$AudioSourcePresetsStateCopyWithImpl(this._value, this._then); + + // ignore: unused_field + final $Val _value; + // ignore: unused_field + final $Res Function($Val) _then; + + /// Create a copy of AudioSourcePresetsState + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? presets = null, + Object? selectedStreamingQualityIndex = null, + Object? selectedStreamingContainerIndex = null, + Object? selectedDownloadingQualityIndex = null, + Object? selectedDownloadingContainerIndex = null, + }) { + return _then(_value.copyWith( + presets: null == presets + ? _value.presets + : presets // ignore: cast_nullable_to_non_nullable + as List, + selectedStreamingQualityIndex: null == selectedStreamingQualityIndex + ? _value.selectedStreamingQualityIndex + : selectedStreamingQualityIndex // ignore: cast_nullable_to_non_nullable + as int, + selectedStreamingContainerIndex: null == selectedStreamingContainerIndex + ? _value.selectedStreamingContainerIndex + : selectedStreamingContainerIndex // ignore: cast_nullable_to_non_nullable + as int, + selectedDownloadingQualityIndex: null == selectedDownloadingQualityIndex + ? _value.selectedDownloadingQualityIndex + : selectedDownloadingQualityIndex // ignore: cast_nullable_to_non_nullable + as int, + selectedDownloadingContainerIndex: null == + selectedDownloadingContainerIndex + ? _value.selectedDownloadingContainerIndex + : selectedDownloadingContainerIndex // ignore: cast_nullable_to_non_nullable + as int, + ) as $Val); + } +} + +/// @nodoc +abstract class _$$AudioSourcePresetsStateImplCopyWith<$Res> + implements $AudioSourcePresetsStateCopyWith<$Res> { + factory _$$AudioSourcePresetsStateImplCopyWith( + _$AudioSourcePresetsStateImpl value, + $Res Function(_$AudioSourcePresetsStateImpl) then) = + __$$AudioSourcePresetsStateImplCopyWithImpl<$Res>; + @override + @useResult + $Res call( + {List presets, + int selectedStreamingQualityIndex, + int selectedStreamingContainerIndex, + int selectedDownloadingQualityIndex, + int selectedDownloadingContainerIndex}); +} + +/// @nodoc +class __$$AudioSourcePresetsStateImplCopyWithImpl<$Res> + extends _$AudioSourcePresetsStateCopyWithImpl<$Res, + _$AudioSourcePresetsStateImpl> + implements _$$AudioSourcePresetsStateImplCopyWith<$Res> { + __$$AudioSourcePresetsStateImplCopyWithImpl( + _$AudioSourcePresetsStateImpl _value, + $Res Function(_$AudioSourcePresetsStateImpl) _then) + : super(_value, _then); + + /// Create a copy of AudioSourcePresetsState + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({ + Object? presets = null, + Object? selectedStreamingQualityIndex = null, + Object? selectedStreamingContainerIndex = null, + Object? selectedDownloadingQualityIndex = null, + Object? selectedDownloadingContainerIndex = null, + }) { + return _then(_$AudioSourcePresetsStateImpl( + presets: null == presets + ? _value._presets + : presets // ignore: cast_nullable_to_non_nullable + as List, + selectedStreamingQualityIndex: null == selectedStreamingQualityIndex + ? _value.selectedStreamingQualityIndex + : selectedStreamingQualityIndex // ignore: cast_nullable_to_non_nullable + as int, + selectedStreamingContainerIndex: null == selectedStreamingContainerIndex + ? _value.selectedStreamingContainerIndex + : selectedStreamingContainerIndex // ignore: cast_nullable_to_non_nullable + as int, + selectedDownloadingQualityIndex: null == selectedDownloadingQualityIndex + ? _value.selectedDownloadingQualityIndex + : selectedDownloadingQualityIndex // ignore: cast_nullable_to_non_nullable + as int, + selectedDownloadingContainerIndex: null == + selectedDownloadingContainerIndex + ? _value.selectedDownloadingContainerIndex + : selectedDownloadingContainerIndex // ignore: cast_nullable_to_non_nullable + as int, + )); + } +} + +/// @nodoc +@JsonSerializable() +class _$AudioSourcePresetsStateImpl implements _AudioSourcePresetsState { + _$AudioSourcePresetsStateImpl( + {final List presets = const [], + this.selectedStreamingQualityIndex = 0, + this.selectedStreamingContainerIndex = 0, + this.selectedDownloadingQualityIndex = 0, + this.selectedDownloadingContainerIndex = 0}) + : _presets = presets; + + factory _$AudioSourcePresetsStateImpl.fromJson(Map json) => + _$$AudioSourcePresetsStateImplFromJson(json); + + final List _presets; + @override + @JsonKey() + List get presets { + if (_presets is EqualUnmodifiableListView) return _presets; + // ignore: implicit_dynamic_type + return EqualUnmodifiableListView(_presets); + } + + @override + @JsonKey() + final int selectedStreamingQualityIndex; + @override + @JsonKey() + final int selectedStreamingContainerIndex; + @override + @JsonKey() + final int selectedDownloadingQualityIndex; + @override + @JsonKey() + final int selectedDownloadingContainerIndex; + + @override + String toString() { + return 'AudioSourcePresetsState(presets: $presets, selectedStreamingQualityIndex: $selectedStreamingQualityIndex, selectedStreamingContainerIndex: $selectedStreamingContainerIndex, selectedDownloadingQualityIndex: $selectedDownloadingQualityIndex, selectedDownloadingContainerIndex: $selectedDownloadingContainerIndex)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$AudioSourcePresetsStateImpl && + const DeepCollectionEquality().equals(other._presets, _presets) && + (identical(other.selectedStreamingQualityIndex, + selectedStreamingQualityIndex) || + other.selectedStreamingQualityIndex == + selectedStreamingQualityIndex) && + (identical(other.selectedStreamingContainerIndex, + selectedStreamingContainerIndex) || + other.selectedStreamingContainerIndex == + selectedStreamingContainerIndex) && + (identical(other.selectedDownloadingQualityIndex, + selectedDownloadingQualityIndex) || + other.selectedDownloadingQualityIndex == + selectedDownloadingQualityIndex) && + (identical(other.selectedDownloadingContainerIndex, + selectedDownloadingContainerIndex) || + other.selectedDownloadingContainerIndex == + selectedDownloadingContainerIndex)); + } + + @JsonKey(includeFromJson: false, includeToJson: false) + @override + int get hashCode => Object.hash( + runtimeType, + const DeepCollectionEquality().hash(_presets), + selectedStreamingQualityIndex, + selectedStreamingContainerIndex, + selectedDownloadingQualityIndex, + selectedDownloadingContainerIndex); + + /// Create a copy of AudioSourcePresetsState + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$AudioSourcePresetsStateImplCopyWith<_$AudioSourcePresetsStateImpl> + get copyWith => __$$AudioSourcePresetsStateImplCopyWithImpl< + _$AudioSourcePresetsStateImpl>(this, _$identity); + + @override + Map toJson() { + return _$$AudioSourcePresetsStateImplToJson( + this, + ); + } +} + +abstract class _AudioSourcePresetsState implements AudioSourcePresetsState { + factory _AudioSourcePresetsState( + {final List presets, + final int selectedStreamingQualityIndex, + final int selectedStreamingContainerIndex, + final int selectedDownloadingQualityIndex, + final int selectedDownloadingContainerIndex}) = + _$AudioSourcePresetsStateImpl; + + factory _AudioSourcePresetsState.fromJson(Map json) = + _$AudioSourcePresetsStateImpl.fromJson; + + @override + List get presets; + @override + int get selectedStreamingQualityIndex; + @override + int get selectedStreamingContainerIndex; + @override + int get selectedDownloadingQualityIndex; + @override + int get selectedDownloadingContainerIndex; + + /// Create a copy of AudioSourcePresetsState + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$AudioSourcePresetsStateImplCopyWith<_$AudioSourcePresetsStateImpl> + get copyWith => throw _privateConstructorUsedError; +} diff --git a/lib/provider/metadata_plugin/audio_source/quality_presets.g.dart b/lib/provider/metadata_plugin/audio_source/quality_presets.g.dart new file mode 100644 index 00000000..f3d8fd41 --- /dev/null +++ b/lib/provider/metadata_plugin/audio_source/quality_presets.g.dart @@ -0,0 +1,38 @@ +// GENERATED CODE - DO NOT MODIFY BY HAND + +part of 'quality_presets.dart'; + +// ************************************************************************** +// JsonSerializableGenerator +// ************************************************************************** + +_$AudioSourcePresetsStateImpl _$$AudioSourcePresetsStateImplFromJson( + Map json) => + _$AudioSourcePresetsStateImpl( + presets: (json['presets'] as List?) + ?.map((e) => SpotubeAudioSourceContainerPreset.fromJson( + Map.from(e as Map))) + .toList() ?? + const [], + selectedStreamingQualityIndex: + (json['selectedStreamingQualityIndex'] as num?)?.toInt() ?? 0, + selectedStreamingContainerIndex: + (json['selectedStreamingContainerIndex'] as num?)?.toInt() ?? 0, + selectedDownloadingQualityIndex: + (json['selectedDownloadingQualityIndex'] as num?)?.toInt() ?? 0, + selectedDownloadingContainerIndex: + (json['selectedDownloadingContainerIndex'] as num?)?.toInt() ?? 0, + ); + +Map _$$AudioSourcePresetsStateImplToJson( + _$AudioSourcePresetsStateImpl instance) => + { + 'presets': instance.presets.map((e) => e.toJson()).toList(), + 'selectedStreamingQualityIndex': instance.selectedStreamingQualityIndex, + 'selectedStreamingContainerIndex': + instance.selectedStreamingContainerIndex, + 'selectedDownloadingQualityIndex': + instance.selectedDownloadingQualityIndex, + 'selectedDownloadingContainerIndex': + instance.selectedDownloadingContainerIndex, + }; diff --git a/lib/provider/metadata_plugin/browse/section_items.dart b/lib/provider/metadata_plugin/browse/section_items.dart new file mode 100644 index 00000000..5c03ec2c --- /dev/null +++ b/lib/provider/metadata_plugin/browse/section_items.dart @@ -0,0 +1,32 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginBrowseSectionItemsNotifier + extends FamilyPaginatedAsyncNotifier { + @override + Future> fetch( + int offset, + int limit, + ) async { + return await (await metadataPlugin).browse.sectionItems( + arg, + limit: limit, + offset: offset, + ); + } + + @override + build(arg) async { + ref.watch(metadataPluginAuthenticatedProvider); + return await fetch(0, 20); + } +} + +final metadataPluginBrowseSectionItemsProvider = AsyncNotifierProviderFamily< + MetadataPluginBrowseSectionItemsNotifier, + SpotubePaginationResponseObject, + String>( + () => MetadataPluginBrowseSectionItemsNotifier(), +); diff --git a/lib/provider/metadata_plugin/browse/sections.dart b/lib/provider/metadata_plugin/browse/sections.dart new file mode 100644 index 00000000..1f73e10c --- /dev/null +++ b/lib/provider/metadata_plugin/browse/sections.dart @@ -0,0 +1,31 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; + +class MetadataPluginBrowseSectionsNotifier + extends PaginatedAsyncNotifier> { + @override + Future>> + fetch( + int offset, + int limit, + ) async { + return await (await metadataPlugin).browse.sections( + limit: limit, + offset: offset, + ); + } + + @override + build() async { + ref.watch(metadataPluginAuthenticatedProvider); + return await fetch(0, 20); + } +} + +final metadataPluginBrowseSectionsProvider = AsyncNotifierProvider< + MetadataPluginBrowseSectionsNotifier, + SpotubePaginationResponseObject>>( + () => MetadataPluginBrowseSectionsNotifier(), +); diff --git a/lib/provider/metadata_plugin/core/auth.dart b/lib/provider/metadata_plugin/core/auth.dart new file mode 100644 index 00000000..dc5e7eb6 --- /dev/null +++ b/lib/provider/metadata_plugin/core/auth.dart @@ -0,0 +1,70 @@ +import 'dart:async'; + +import 'package:riverpod/riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; + +class MetadataPluginAuthenticatedNotifier extends AsyncNotifier { + @override + FutureOr build() async { + final defaultPluginConfig = ref.watch(metadataPluginsProvider); + if (defaultPluginConfig.asData?.value.defaultMetadataPluginConfig?.abilities + .contains(PluginAbilities.authentication) != + true) { + return false; + } + + final defaultPlugin = await ref.watch(metadataPluginProvider.future); + if (defaultPlugin == null) { + return false; + } + + final sub = defaultPlugin.auth.authStateStream.listen((event) { + state = AsyncData(defaultPlugin.auth.isAuthenticated()); + }); + + ref.onDispose(() { + sub.cancel(); + }); + + return defaultPlugin.auth.isAuthenticated(); + } +} + +final metadataPluginAuthenticatedProvider = + AsyncNotifierProvider( + MetadataPluginAuthenticatedNotifier.new, +); + +class AudioSourcePluginAuthenticatedNotifier extends AsyncNotifier { + @override + FutureOr build() async { + final defaultPluginConfig = ref.watch(metadataPluginsProvider); + if (defaultPluginConfig + .asData?.value.defaultAudioSourcePluginConfig?.abilities + .contains(PluginAbilities.authentication) != + true) { + return false; + } + + final defaultPlugin = await ref.watch(audioSourcePluginProvider.future); + if (defaultPlugin == null) { + return false; + } + + final sub = defaultPlugin.auth.authStateStream.listen((event) { + state = AsyncData(defaultPlugin.auth.isAuthenticated()); + }); + + ref.onDispose(() { + sub.cancel(); + }); + + return defaultPlugin.auth.isAuthenticated(); + } +} + +final audioSourcePluginAuthenticatedProvider = + AsyncNotifierProvider( + AudioSourcePluginAuthenticatedNotifier.new, +); diff --git a/lib/provider/metadata_plugin/core/repositories.dart b/lib/provider/metadata_plugin/core/repositories.dart new file mode 100644 index 00000000..a78f63d9 --- /dev/null +++ b/lib/provider/metadata_plugin/core/repositories.dart @@ -0,0 +1,90 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; +import 'package:spotube/services/dio/dio.dart'; + +class MetadataPluginRepositoriesNotifier + extends PaginatedAsyncNotifier { + MetadataPluginRepositoriesNotifier() : super(); + + Map _hasMore = {}; + + @override + fetch(int offset, int limit) async { + final gitubSearch = globalDio.get( + "https://api.github.com/search/repositories", + queryParameters: { + "q": "topic:spotube-plugin", + "sort": "stars", + "order": "desc", + "page": offset, + "per_page": limit, + }, + ); + + final codebergSearch = globalDio.get( + "https://codeberg.org/api/v1/repos/search", + queryParameters: { + "q": "spotube-plugin", + "topic": "true", + "sort": "stars", + "order": "desc", + "page": offset, + "limit": limit, + }, + ); + + final responses = await Future.wait([ + if (_hasMore["github.com"] ?? true) gitubSearch, + if (_hasMore["codeberg.org"] ?? true) codebergSearch, + ]); + + final repos = responses + .expand( + (response) => response.data["data"] ?? response.data["items"] ?? [], + ) + .map((repo) { + return MetadataPluginRepository( + name: repo["name"] ?? "", + owner: repo["owner"]["login"] ?? "", + description: repo["description"] ?? "", + repoUrl: repo["html_url"] ?? "", + topics: repo["topics"].cast() ?? [], + ); + }).toList(); + + final hasMore = responses.any((response) { + final items = + (response.data["data"] ?? response.data["items"] ?? []) as List; + _hasMore[response.requestOptions.uri.host] = + items.length >= limit && items.isNotEmpty; + + return _hasMore[response.requestOptions.uri.host] ?? false; + }); + + return SpotubePaginationResponseObject( + items: repos, + total: responses.fold( + 0, + (previousValue, response) => previousValue + + (response.data["total_count"] ?? + int.tryParse(response.headers["x-total-count"]?[0] ?? "") ?? + 0) as int, + ), + hasMore: hasMore, + nextOffset: hasMore ? offset + 1 : null, + limit: limit, + ); + } + + @override + build() async { + return await fetch(0, 10); + } +} + +final metadataPluginRepositoriesProvider = AsyncNotifierProvider< + MetadataPluginRepositoriesNotifier, + SpotubePaginationResponseObject>( + () => MetadataPluginRepositoriesNotifier(), +); diff --git a/lib/provider/metadata_plugin/core/scrobble.dart b/lib/provider/metadata_plugin/core/scrobble.dart new file mode 100644 index 00000000..0f8fcc19 --- /dev/null +++ b/lib/provider/metadata_plugin/core/scrobble.dart @@ -0,0 +1,66 @@ +import 'dart:async'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/logger/logger.dart'; + +class MetadataPluginScrobbleNotifier + extends Notifier?> { + @override + build() { + final metadataPlugin = ref.watch(metadataPluginProvider); + final pluginConfig = ref + .watch(metadataPluginsProvider) + .valueOrNull + ?.defaultMetadataPluginConfig; + + if (metadataPlugin.valueOrNull == null || + pluginConfig == null || + !pluginConfig.abilities.contains(PluginAbilities.scrobbling)) { + return null; + } + + final controller = StreamController.broadcast(); + + final subscription = controller.stream.listen((event) async { + try { + await metadataPlugin.valueOrNull?.core.scrobble({ + "id": event.id, + "title": event.name, + "artists": event.artists + .map((artist) => { + "id": artist.id, + "name": artist.name, + }) + .toList(), + "album": { + "id": event.album.id, + "name": event.album.name, + }, + "timestamp": DateTime.now().millisecondsSinceEpoch ~/ 1000, + "duration_ms": event.durationMs, + "isrc": event is SpotubeFullTrackObject ? event.isrc : null, + }); + } catch (e, stack) { + AppLogger.reportError(e, stack); + } + }); + + ref.onDispose(() { + subscription.cancel(); + controller.close(); + }); + + return controller; + } + + void scrobble(SpotubeTrackObject track) { + state?.add(track); + } +} + +final metadataPluginScrobbleProvider = NotifierProvider< + MetadataPluginScrobbleNotifier, StreamController?>( + MetadataPluginScrobbleNotifier.new, +); diff --git a/lib/provider/metadata_plugin/core/support.dart b/lib/provider/metadata_plugin/core/support.dart new file mode 100644 index 00000000..8864f1b1 --- /dev/null +++ b/lib/provider/metadata_plugin/core/support.dart @@ -0,0 +1,21 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; + +final metadataPluginSupportTextProvider = FutureProvider((ref) async { + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw 'No metadata plugin available'; + } + return await metadataPlugin.core.support; +}); + +final audioSourcePluginSupportTextProvider = + FutureProvider((ref) async { + final audioSourcePlugin = await ref.watch(audioSourcePluginProvider.future); + + if (audioSourcePlugin == null) { + throw 'No metadata plugin available'; + } + return await audioSourcePlugin.core.support; +}); diff --git a/lib/provider/metadata_plugin/core/user.dart b/lib/provider/metadata_plugin/core/user.dart new file mode 100644 index 00000000..3ad46d63 --- /dev/null +++ b/lib/provider/metadata_plugin/core/user.dart @@ -0,0 +1,17 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; + +final metadataPluginUserProvider = FutureProvider( + (ref) async { + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + final authenticated = + await ref.watch(metadataPluginAuthenticatedProvider.future); + + if (!authenticated || metadataPlugin == null) { + return null; + } + return metadataPlugin.user.me(); + }, +); diff --git a/lib/provider/metadata_plugin/library/albums.dart b/lib/provider/metadata_plugin/library/albums.dart new file mode 100644 index 00000000..10438025 --- /dev/null +++ b/lib/provider/metadata_plugin/library/albums.dart @@ -0,0 +1,88 @@ +import 'package:riverpod/riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; + +class MetadataPluginSavedAlbumNotifier + extends PaginatedAsyncNotifier { + @override + Future> fetch( + int offset, + int limit, + ) async { + return await (await metadataPlugin).user.savedAlbums( + limit: limit, + offset: offset, + ); + } + + @override + build() async { + await ref.watch(metadataPluginAuthenticatedProvider.future); + return await fetch(0, 20); + } + + Future addFavorite(List albums) async { + if (albums.isEmpty || state.value == null) return; + final oldState = state.value; + + state = AsyncData( + state.value!.copyWith( + items: [ + ...albums, + ...state.value!.items, + ], + ), + ); + try { + await (await metadataPlugin).album.save(albums.map((e) => e.id).toList()); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } + + Future removeFavorite(List albums) async { + if (albums.isEmpty || state.value == null) return; + + final oldState = state.value; + + final albumIds = albums.map((e) => e.id).toList(); + state = AsyncData( + state.value!.copyWith( + items: state.value!.items + .where( + (e) => albumIds.contains((e).id) == false, + ) + .toList(), + ), + ); + try { + await (await metadataPlugin).album.unsave(albumIds); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } +} + +final metadataPluginSavedAlbumsProvider = AsyncNotifierProvider< + MetadataPluginSavedAlbumNotifier, + SpotubePaginationResponseObject>( + () => MetadataPluginSavedAlbumNotifier(), +); + +final metadataPluginIsSavedAlbumProvider = + FutureProvider.autoDispose.family( + (ref, albumId) async { + final savedAlbums = + await ref.watch(metadataPluginSavedAlbumsProvider.future); + final savedAlbumsNotifier = + ref.read(metadataPluginSavedAlbumsProvider.notifier); + final allSavedAlbums = savedAlbums.hasMore + ? await savedAlbumsNotifier.fetchAll() + : savedAlbums.items; + + return allSavedAlbums.any((element) => element.id == albumId); + }, +); diff --git a/lib/provider/metadata_plugin/library/artists.dart b/lib/provider/metadata_plugin/library/artists.dart new file mode 100644 index 00000000..31f976e0 --- /dev/null +++ b/lib/provider/metadata_plugin/library/artists.dart @@ -0,0 +1,94 @@ +import 'package:riverpod/riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; + +class MetadataPluginSavedArtistNotifier + extends PaginatedAsyncNotifier { + @override + Future> fetch( + int offset, + int limit, + ) async { + final artists = await (await metadataPlugin).user.savedArtists( + limit: limit, + offset: offset, + ); + + return artists; + } + + @override + build() async { + await ref.watch(metadataPluginAuthenticatedProvider.future); + return await fetch(0, 20); + } + + Future addFavorite(List artists) async { + if (artists.isEmpty || state.value == null) return; + final oldState = state.value; + + state = AsyncData( + state.value!.copyWith( + items: [ + ...artists, + ...state.value!.items, + ], + ), + ); + try { + await (await metadataPlugin) + .artist + .save(artists.map((e) => e.id).toList()); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } + + Future removeFavorite(List artists) async { + if (artists.isEmpty || state.value == null) return; + + final oldState = state.value; + + final artistIds = artists.map((e) => e.id).toList(); + state = AsyncData( + state.value!.copyWith( + items: state.value!.items + .where( + (e) => artistIds.contains((e).id) == false, + ) + .toList(), + ), + ); + + try { + await (await metadataPlugin).artist.unsave(artistIds); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } +} + +final metadataPluginSavedArtistsProvider = AsyncNotifierProvider< + MetadataPluginSavedArtistNotifier, + SpotubePaginationResponseObject>( + () => MetadataPluginSavedArtistNotifier(), +); + +final metadataPluginIsSavedArtistProvider = + FutureProvider.autoDispose.family( + (ref, artistId) async { + final savedArtists = + await ref.watch(metadataPluginSavedArtistsProvider.future); + final savedArtistsNotifier = + ref.read(metadataPluginSavedArtistsProvider.notifier); + + final allSavedArtists = savedArtists.hasMore + ? await savedArtistsNotifier.fetchAll() + : savedArtists.items; + + return allSavedArtists.any((element) => element.id == artistId); + }, +); diff --git a/lib/provider/metadata_plugin/library/playlists.dart b/lib/provider/metadata_plugin/library/playlists.dart new file mode 100644 index 00000000..5793eb57 --- /dev/null +++ b/lib/provider/metadata_plugin/library/playlists.dart @@ -0,0 +1,149 @@ +import 'package:collection/collection.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/tracks/playlist.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +class MetadataPluginSavedPlaylistsNotifier + extends PaginatedAsyncNotifier { + MetadataPluginSavedPlaylistsNotifier() : super(); + + @override + fetch(int offset, int limit) async { + final playlists = await (await metadataPlugin) + .user + .savedPlaylists(limit: limit, offset: offset); + + return playlists; + } + + @override + build() async { + await ref.watch(metadataPluginAuthenticatedProvider.future); + + final playlists = await fetch(0, 20); + + return playlists; + } + + void updatePlaylist(SpotubeSimplePlaylistObject playlist) { + if (state.value == null) return; + + if (state.value!.items.none((e) => e.id == playlist.id)) return; + + state = AsyncData( + state.value!.copyWith( + items: state.value!.items + .map((element) => element.id == playlist.id ? playlist : element) + .toList(), + ), + ); + } + + Future addFavorite(SpotubeSimplePlaylistObject playlist) async { + if (state.value == null) return; + + final oldState = state.value; + + state = AsyncData( + state.value!.copyWith( + items: [ + playlist, + ...state.value!.items, + ], + ), + ); + + try { + await (await metadataPlugin).playlist.save(playlist.id); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } + + Future removeFavorite(SpotubeSimplePlaylistObject playlist) async { + if (state.value == null) return; + + final oldState = state.value; + state = AsyncData( + state.value!.copyWith( + items: state.value!.items.where((e) => (e).id != playlist.id).toList(), + ), + ); + + try { + await (await metadataPlugin).playlist.unsave(playlist.id); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } + + Future delete(String playlistId) async { + if (state.value == null) return; + final oldState = state; + try { + state = const AsyncLoading(); + await (await metadataPlugin).playlist.deletePlaylist(playlistId); + ref.invalidateSelf(); + ref.invalidate(metadataPluginIsSavedPlaylistProvider(playlistId)); + ref.invalidate(metadataPluginPlaylistTracksProvider(playlistId)); + } catch (e) { + state = oldState; + rethrow; + } + } + + Future addTracks(String playlistId, List trackIds) async { + if (state.value == null) return; + + await (await metadataPlugin) + .playlist + .addTracks(playlistId, trackIds: trackIds); + + ref.invalidate(metadataPluginPlaylistTracksProvider(playlistId)); + } + + Future removeTracks(String playlistId, List trackIds) async { + if (state.value == null) return; + + await (await metadataPlugin) + .playlist + .removeTracks(playlistId, trackIds: trackIds); + + ref.invalidate(metadataPluginPlaylistTracksProvider(playlistId)); + } +} + +final metadataPluginSavedPlaylistsProvider = AsyncNotifierProvider< + MetadataPluginSavedPlaylistsNotifier, + SpotubePaginationResponseObject>( + () => MetadataPluginSavedPlaylistsNotifier(), +); + +final metadataPluginIsSavedPlaylistProvider = + FutureProvider.family( + (ref, id) async { + final plugin = await ref.watch(metadataPluginProvider.future); + + if (plugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + final savedPlaylists = + await ref.watch(metadataPluginSavedPlaylistsProvider.future); + + final savedPlaylistsNotifier = + ref.read(metadataPluginSavedPlaylistsProvider.notifier); + + final allSavedPlaylists = savedPlaylists.hasMore + ? await savedPlaylistsNotifier.fetchAll() + : savedPlaylists.items; + + return allSavedPlaylists.any((element) => element.id == id); + }, +); diff --git a/lib/provider/metadata_plugin/library/tracks.dart b/lib/provider/metadata_plugin/library/tracks.dart new file mode 100644 index 00000000..d19865dd --- /dev/null +++ b/lib/provider/metadata_plugin/library/tracks.dart @@ -0,0 +1,96 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/provider/metadata_plugin/utils/paginated.dart'; + +class MetadataPluginSavedTracksNotifier + extends AutoDisposePaginatedAsyncNotifier { + MetadataPluginSavedTracksNotifier() : super(); + + @override + fetch(offset, limit) async { + final tracks = await (await metadataPlugin).user.savedTracks( + offset: offset, + limit: limit, + ); + + return tracks; + } + + @override + build() async { + ref.cacheFor(); + + await ref.watch(metadataPluginAuthenticatedProvider.future); + return await fetch(0, 20); + } + + Future addFavorite(List tracks) async { + if (state.value == null) { + return; + } + + final oldState = state.value; + state = AsyncData( + state.value!.copyWith( + items: [ + ...tracks.whereType(), + ...state.value!.items + ], + ), + ); + + try { + await (await metadataPlugin).track.save(tracks.map((e) => e.id).toList()); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } + + Future removeFavorite(List tracks) async { + if (state.value == null) { + return; + } + + final oldState = state.value; + state = AsyncData( + state.value!.copyWith( + items: state.value!.items + .where( + (savedTrack) => !tracks.any((track) => track.id == savedTrack.id), + ) + .toList(), + ), + ); + + try { + await (await metadataPlugin) + .track + .unsave(tracks.map((e) => e.id).toList()); + } catch (e) { + state = AsyncData(oldState!); + rethrow; + } + } +} + +final metadataPluginSavedTracksProvider = AutoDisposeAsyncNotifierProvider< + MetadataPluginSavedTracksNotifier, + SpotubePaginationResponseObject>( + () => MetadataPluginSavedTracksNotifier(), +); + +final metadataPluginIsSavedTrackProvider = + FutureProvider.autoDispose.family( + (ref, trackId) async { + final savedTracks = + await ref.watch(metadataPluginSavedTracksProvider.future); + final allSavedTracks = savedTracks.hasMore + ? await ref.read(metadataPluginSavedTracksProvider.notifier).fetchAll() + : savedTracks.items; + + return allSavedTracks.any((track) => track.id == trackId); + }, +); diff --git a/lib/provider/metadata_plugin/metadata_plugin_provider.dart b/lib/provider/metadata_plugin/metadata_plugin_provider.dart new file mode 100644 index 00000000..cdc96c41 --- /dev/null +++ b/lib/provider/metadata_plugin/metadata_plugin_provider.dart @@ -0,0 +1,635 @@ +import 'dart:convert'; +import 'dart:io'; + +import 'package:collection/collection.dart'; +import 'package:dio/dio.dart'; +import 'package:drift/drift.dart'; +import 'package:flutter/services.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:path/path.dart'; +import 'package:path_provider/path_provider.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/database/database.dart'; +import 'package:spotube/provider/youtube_engine/youtube_engine.dart'; +import 'package:spotube/services/dio/dio.dart'; +import 'package:spotube/services/logger/logger.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; +import 'package:spotube/services/metadata/metadata.dart'; +import 'package:spotube/utils/service_utils.dart'; +import 'package:archive/archive.dart'; +import 'package:pub_semver/pub_semver.dart'; + +final allowedDomainsRegex = RegExp( + r"^(https?:\/\/)?(www\.)?(github\.com|codeberg\.org)\/.+", +); + +class MetadataPluginState { + final List plugins; + final int defaultMetadataPlugin; + final int defaultAudioSourcePlugin; + + const MetadataPluginState({ + this.plugins = const [], + this.defaultMetadataPlugin = -1, + this.defaultAudioSourcePlugin = -1, + }); + + PluginConfiguration? get defaultMetadataPluginConfig { + if (defaultMetadataPlugin < 0 || defaultMetadataPlugin >= plugins.length) { + return null; + } + return plugins[defaultMetadataPlugin]; + } + + PluginConfiguration? get defaultAudioSourcePluginConfig { + if (defaultAudioSourcePlugin < 0 || + defaultAudioSourcePlugin >= plugins.length) { + return null; + } + return plugins[defaultAudioSourcePlugin]; + } + + factory MetadataPluginState.fromJson(Map json) { + return MetadataPluginState( + plugins: (json["plugins"] as List) + .map((e) => PluginConfiguration.fromJson(e)) + .toList(), + defaultMetadataPlugin: json["default_metadata_plugin"] ?? -1, + defaultAudioSourcePlugin: json['default_audio_source_plugin'], + ); + } + + Map toJson() { + return { + "plugins": plugins.map((e) => e.toJson()).toList(), + "default_metadata_plugin": defaultMetadataPlugin, + "default_audio_source_plugin": defaultAudioSourcePlugin + }; + } + + MetadataPluginState copyWith({ + List? plugins, + int? defaultMetadataPlugin, + int? defaultAudioSourcePlugin, + }) { + return MetadataPluginState( + plugins: plugins ?? this.plugins, + defaultMetadataPlugin: + defaultMetadataPlugin ?? this.defaultMetadataPlugin, + defaultAudioSourcePlugin: + defaultAudioSourcePlugin ?? this.defaultAudioSourcePlugin, + ); + } +} + +class MetadataPluginNotifier extends AsyncNotifier { + AppDatabase get database => ref.read(databaseProvider); + + @override + build() async { + final database = ref.watch(databaseProvider); + + final subscription = database.pluginsTable.select().watch().listen( + (event) async { + state = AsyncValue.data(await toStatePlugins(event)); + }, + ); + + ref.onDispose(() { + subscription.cancel(); + }); + + final plugins = await database.pluginsTable.select().get(); + + final pluginState = await toStatePlugins(plugins); + + await _loadDefaultPlugins(pluginState); + + return pluginState; + } + + Future toStatePlugins( + List plugins, + ) async { + int defaultMetadataPlugin = -1; + int defaultAudioSourcePlugin = -1; + final pluginConfigs = []; + + for (int i = 0; i < plugins.length; i++) { + final plugin = plugins[i]; + + final pluginConfig = PluginConfiguration( + name: plugin.name, + author: plugin.author, + description: plugin.description, + version: plugin.version, + entryPoint: plugin.entryPoint, + pluginApiVersion: plugin.pluginApiVersion, + repository: plugin.repository, + apis: plugin.apis + .map( + (e) => PluginApis.values.firstWhereOrNull( + (api) => api.name == e, + ), + ) + .nonNulls + .toList(), + abilities: plugin.abilities + .map( + (e) => PluginAbilities.values.firstWhereOrNull( + (ability) => ability.name == e, + ), + ) + .nonNulls + .toList(), + ); + + final pluginExtractionDir = await _getPluginExtractionDir(pluginConfig); + final pluginJsonFile = + File(join(pluginExtractionDir.path, "plugin.json")); + final pluginBinaryFile = + File(join(pluginExtractionDir.path, "plugin.out")); + + if (!await pluginExtractionDir.exists() || + !await pluginJsonFile.exists() || + !await pluginBinaryFile.exists()) { + // Delete the plugin entry from DB if the plugin files are not there. + await database.pluginsTable.deleteOne(plugin); + continue; + } + + pluginConfigs.add(pluginConfig); + + if (plugin.selectedForMetadata) { + defaultMetadataPlugin = pluginConfigs.length - 1; + } + if (plugin.selectedForAudioSource) { + defaultAudioSourcePlugin = pluginConfigs.length - 1; + } + } + + return MetadataPluginState( + plugins: pluginConfigs, + defaultMetadataPlugin: defaultMetadataPlugin, + defaultAudioSourcePlugin: defaultAudioSourcePlugin, + ); + } + + Future _loadDefaultPlugins(MetadataPluginState pluginState) async { + const plugins = [ + "spotube-plugin-musicbrainz-listenbrainz", + "spotube-plugin-youtube-audio", + ]; + + for (final plugin in plugins) { + final byteData = await rootBundle.load( + "assets/plugins/$plugin/plugin.smplug", + ); + final pluginConfig = + await extractPluginArchive(byteData.buffer.asUint8List()); + try { + await addPlugin(pluginConfig); + } on MetadataPluginException catch (e) { + if (e.errorCode == MetadataPluginErrorCode.duplicatePlugin && + await isPluginUpdate(pluginConfig)) { + final oldConfig = pluginState.plugins + .firstWhereOrNull((p) => p.slug == pluginConfig.slug); + if (oldConfig == null) continue; + final isDefaultMetadata = + oldConfig == pluginState.defaultMetadataPluginConfig; + final isDefaultAudioSource = + oldConfig == pluginState.defaultAudioSourcePluginConfig; + + await removePlugin(pluginConfig); + await addPlugin(pluginConfig); + + if (isDefaultMetadata) { + await setDefaultMetadataPlugin(pluginConfig); + } + if (isDefaultAudioSource) { + await setDefaultAudioSourcePlugin(pluginConfig); + } + } + } + } + } + + Uri _getGithubReleasesUrl(String repoUrl) { + final parsedUri = Uri.parse(repoUrl); + final uri = parsedUri.replace( + host: "api.github.com", + pathSegments: [ + "repos", + ...parsedUri.pathSegments, + "releases", + ], + queryParameters: { + "per_page": "1", + "page": "1", + }, + ); + + return uri; + } + + Uri _getCodebergeReleasesUrl(String repoUrl) { + final parsedUri = Uri.parse(repoUrl); + final uri = parsedUri.replace( + pathSegments: [ + "api", + "v1", + "repos", + ...parsedUri.pathSegments, + "releases", + ], + queryParameters: { + "limit": "1", + "page": "1", + }, + ); + + return uri; + } + + Future _getPluginDownloadUrl(Uri uri) async { + AppLogger.log.i("Getting plugin download URL from: $uri"); + final res = await globalDio.getUri( + uri, + options: Options(responseType: ResponseType.json), + ); + + if (res.statusCode != 200) { + throw MetadataPluginException.failedToGetRelease(); + } + final releases = res.data as List; + if (releases.isEmpty) { + throw MetadataPluginException.noReleasesFound(); + } + final latestRelease = releases.first; + final downloadUrl = (latestRelease["assets"] as List).firstWhere( + (asset) => (asset["name"] as String).endsWith(".smplug"), + )["browser_download_url"]; + if (downloadUrl == null) { + throw MetadataPluginException.assetUrlNotFound(); + } + return downloadUrl; + } + + /// Root directory where all metadata plugins are stored. + Future _getPluginRootDir() async => Directory( + join( + (await getApplicationSupportDirectory()).path, + "metadata-plugins", + ), + ); + + /// Directory where the plugin will be extracted. + /// This is a unique directory for each plugin version. + /// It is used to avoid conflicts when multiple versions of the same plugin are installed + Future _getPluginExtractionDir(PluginConfiguration plugin) async { + final pluginDir = await _getPluginRootDir(); + final pluginExtractionDirPath = join( + pluginDir.path, + "${ServiceUtils.sanitizeFilename(plugin.author)}-${ServiceUtils.sanitizeFilename(plugin.name)}-${plugin.version}", + ); + return Directory(pluginExtractionDirPath); + } + + Future extractPluginArchive(List bytes) async { + final archive = ZipDecoder().decodeBytes(bytes); + final pluginJson = archive + .firstWhereOrNull((file) => file.isFile && file.name == "plugin.json"); + + if (pluginJson == null) { + throw MetadataPluginException.pluginConfigJsonNotFound(); + } + final pluginConfig = PluginConfiguration.fromJson( + jsonDecode( + utf8.decode(pluginJson.content as List), + ) as Map, + ); + + final pluginDir = await _getPluginRootDir(); + await pluginDir.create(recursive: true); + + final pluginExtractionDir = await _getPluginExtractionDir(pluginConfig); + + for (final file in archive) { + if (file.isFile) { + final filename = file.name; + final data = file.content as List; + final extractedFile = File(join( + pluginExtractionDir.path, + filename, + )); + await extractedFile.create(recursive: true); + await extractedFile.writeAsBytes(data); + } + } + + return pluginConfig; + } + + /// Downloads, extracts & caches the plugin from the given URL and returns the plugin config. + /// If only a text/html URL is provided, it will try to get the latest release from + /// the URL for supported websites (github.com, codeberg.org). + Future downloadAndCachePlugin(String url) async { + final res = await globalDio.head(url); + final isSupportedWebsite = + (res.headers["Content-Type"]?.first)?.startsWith("text/html") == true && + allowedDomainsRegex.hasMatch(url); + String pluginDownloadUrl = url; + if (isSupportedWebsite) { + if (url.contains("github.com")) { + final uri = _getGithubReleasesUrl(url); + pluginDownloadUrl = await _getPluginDownloadUrl(uri); + } else if (url.contains("codeberg.org")) { + final uri = _getCodebergeReleasesUrl(url); + pluginDownloadUrl = await _getPluginDownloadUrl(uri); + } else { + throw MetadataPluginException.unsupportedPluginDownloadWebsite(); + } + } + + // Now let's download, extract and cache the plugin + final pluginDir = await _getPluginRootDir(); + await pluginDir.create(recursive: true); + + final pluginRes = await globalDio.get( + pluginDownloadUrl, + options: Options( + responseType: ResponseType.bytes, + followRedirects: true, + receiveTimeout: const Duration(seconds: 30), + ), + ); + + if ((pluginRes.statusCode ?? 500) > 299) { + throw MetadataPluginException.pluginDownloadFailed(); + } + + return await extractPluginArchive(pluginRes.data); + } + + bool validatePluginApiCompatibility(PluginConfiguration plugin) { + final configPluginApiVersion = Version.parse(plugin.pluginApiVersion); + final appPluginApiVersion = MetadataPlugin.pluginApiVersion; + + // Plugin API's major version must match the app's major version + if (configPluginApiVersion.major != appPluginApiVersion.major) { + return false; + } + return configPluginApiVersion >= appPluginApiVersion; + } + + void _assertPluginApiCompatibility(PluginConfiguration plugin) { + if (!validatePluginApiCompatibility(plugin)) { + throw MetadataPluginException.pluginApiVersionMismatch(); + } + } + + Future addPlugin(PluginConfiguration plugin) async { + _assertPluginApiCompatibility(plugin); + + final pluginRes = await (database.pluginsTable.select() + ..where( + (tbl) => + tbl.name.equals(plugin.name) & tbl.author.equals(plugin.author), + ) + ..limit(1)) + .get(); + + if (pluginRes.isNotEmpty) { + throw MetadataPluginException.duplicatePlugin(); + } + + await database.pluginsTable.insertOne( + PluginsTableCompanion.insert( + name: plugin.name, + author: plugin.author, + description: plugin.description, + version: plugin.version, + entryPoint: plugin.entryPoint, + apis: plugin.apis.map((e) => e.name).toList(), + abilities: plugin.abilities.map((e) => e.name).toList(), + pluginApiVersion: Value(plugin.pluginApiVersion), + repository: Value(plugin.repository), + // Setting the very first plugin as the default plugin + selectedForMetadata: Value( + (state.valueOrNull?.plugins + .where( + (d) => d.abilities.contains(PluginAbilities.metadata)) + .isEmpty ?? + true) && + plugin.abilities.contains(PluginAbilities.metadata), + ), + selectedForAudioSource: Value( + (state.valueOrNull?.plugins + .where((d) => + d.abilities.contains(PluginAbilities.audioSource)) + .isEmpty ?? + true) && + plugin.abilities.contains(PluginAbilities.audioSource), + ), + ), + ); + } + + Future removePlugin(PluginConfiguration plugin) async { + final pluginExtractionDir = await _getPluginExtractionDir(plugin); + + if (pluginExtractionDir.existsSync()) { + await pluginExtractionDir.delete(recursive: true); + } + await database.pluginsTable.deleteWhere((tbl) => + tbl.name.equals(plugin.name) & tbl.author.equals(plugin.author)); + + // Same here, if the removed plugin is the default plugin + // set the first available plugin as the default plugin + // only when there is 1 remaining plugin + if (state.valueOrNull?.defaultMetadataPluginConfig == plugin) { + final remainingPlugins = state.valueOrNull?.plugins.where( + (p) => + p != plugin && p.abilities.contains(PluginAbilities.metadata), + ) ?? + []; + if (remainingPlugins.length == 1) { + await setDefaultMetadataPlugin(remainingPlugins.first); + } + } + + if (state.valueOrNull?.defaultAudioSourcePluginConfig == plugin) { + final remainingPlugins = state.valueOrNull?.plugins.where( + (p) => + p != plugin && + p.abilities.contains(PluginAbilities.audioSource), + ) ?? + []; + if (remainingPlugins.length == 1) { + await setDefaultAudioSourcePlugin(remainingPlugins.first); + } + } + } + + Future isPluginUpdate(PluginConfiguration newPlugin) async { + final pluginRes = await (database.pluginsTable.select() + ..where( + (tbl) => + tbl.name.equals(newPlugin.name) & + tbl.author.equals(newPlugin.author), + ) + ..limit(1)) + .get(); + + if (pluginRes.isEmpty) { + return false; + } + + final oldPlugin = pluginRes.first; + final oldPluginApiVersion = Version.parse(oldPlugin.pluginApiVersion); + final newPluginApiVersion = Version.parse(newPlugin.pluginApiVersion); + + return newPluginApiVersion > oldPluginApiVersion; + } + + Future updatePlugin( + PluginConfiguration plugin, + PluginUpdateAvailable update, + ) async { + final isDefaultMetadata = + plugin == state.valueOrNull?.defaultMetadataPluginConfig; + final isDefaultAudioSource = + plugin == state.valueOrNull?.defaultAudioSourcePluginConfig; + final pluginUpdatedConfig = + await downloadAndCachePlugin(update.downloadUrl); + + if (pluginUpdatedConfig.name != plugin.name && + pluginUpdatedConfig.author != plugin.author) { + throw MetadataPluginException.invalidPluginConfiguration(); + } + _assertPluginApiCompatibility(pluginUpdatedConfig); + + await removePlugin(plugin); + await addPlugin(pluginUpdatedConfig); + + if (isDefaultMetadata) { + await setDefaultMetadataPlugin(pluginUpdatedConfig); + } + if (isDefaultAudioSource) { + await setDefaultAudioSourcePlugin(pluginUpdatedConfig); + } + } + + Future setDefaultMetadataPlugin(PluginConfiguration plugin) async { + assert( + plugin.abilities.contains(PluginAbilities.metadata), + "Must be a metadata plugin", + ); + + await database.pluginsTable + .update() + .write(const PluginsTableCompanion(selectedForMetadata: Value(false))); + + await (database.pluginsTable.update() + ..where((tbl) => + tbl.name.equals(plugin.name) & tbl.author.equals(plugin.author))) + .write( + const PluginsTableCompanion(selectedForMetadata: Value(true)), + ); + } + + Future setDefaultAudioSourcePlugin(PluginConfiguration plugin) async { + assert( + plugin.abilities.contains(PluginAbilities.audioSource), + "Must be an audio-source plugin", + ); + + await database.pluginsTable.update().write( + const PluginsTableCompanion(selectedForAudioSource: Value(false))); + + await (database.pluginsTable.update() + ..where((tbl) => + tbl.name.equals(plugin.name) & tbl.author.equals(plugin.author))) + .write( + const PluginsTableCompanion(selectedForAudioSource: Value(true)), + ); + } + + Future getPluginByteCode(PluginConfiguration plugin) async { + final pluginExtractionDirPath = await _getPluginExtractionDir(plugin); + + final libraryFile = File(join(pluginExtractionDirPath.path, "plugin.out")); + + if (!libraryFile.existsSync()) { + throw MetadataPluginException.pluginByteCodeFileNotFound(); + } + + return await libraryFile.readAsBytes(); + } + + Future getLogoPath(PluginConfiguration plugin) async { + final pluginExtractionDirPath = await _getPluginExtractionDir(plugin); + + final logoFile = File(join(pluginExtractionDirPath.path, "logo.png")); + + if (!logoFile.existsSync()) { + return null; + } + + return logoFile; + } +} + +final metadataPluginsProvider = + AsyncNotifierProvider( + MetadataPluginNotifier.new, +); + +final metadataPluginProvider = FutureProvider( + (ref) async { + final defaultPlugin = await ref.watch( + metadataPluginsProvider + .selectAsync((data) => data.defaultMetadataPluginConfig), + ); + final youtubeEngine = ref.read(youtubeEngineProvider); + + if (defaultPlugin == null) { + return null; + } + + final pluginsNotifier = ref.read(metadataPluginsProvider.notifier); + final pluginByteCode = + await pluginsNotifier.getPluginByteCode(defaultPlugin); + + return await MetadataPlugin.create( + youtubeEngine, + defaultPlugin, + pluginByteCode, + ); + }, +); + +final audioSourcePluginProvider = FutureProvider( + (ref) async { + final defaultPlugin = await ref.watch( + metadataPluginsProvider + .selectAsync((data) => data.defaultAudioSourcePluginConfig), + ); + final youtubeEngine = ref.watch(youtubeEngineProvider); + + if (defaultPlugin == null) { + return null; + } + + final pluginsNotifier = ref.read(metadataPluginsProvider.notifier); + final pluginByteCode = + await pluginsNotifier.getPluginByteCode(defaultPlugin); + + return await MetadataPlugin.create( + youtubeEngine, + defaultPlugin, + pluginByteCode, + ); + }, +); diff --git a/lib/provider/metadata_plugin/playlist/playlist.dart b/lib/provider/metadata_plugin/playlist/playlist.dart new file mode 100644 index 00000000..9a41340d --- /dev/null +++ b/lib/provider/metadata_plugin/playlist/playlist.dart @@ -0,0 +1,131 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/core/user.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; +import 'package:spotube/services/metadata/metadata.dart'; + +class MetadataPluginPlaylistNotifier + extends AutoDisposeFamilyAsyncNotifier { + Future get metadataPlugin async { + final metadataPlugin = await ref.read(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + return metadataPlugin; + } + + @override + build(playlistId) async { + ref.cacheFor(); + + return (await metadataPlugin).playlist.getPlaylist(playlistId); + } + + Future create({ + required String name, + String? description, + bool? public, + bool? collaborative, + void Function(dynamic error)? onError, + }) async { + final userId = await ref + .read(metadataPluginUserProvider.selectAsync((data) => data?.id)); + if (userId == null) { + throw Exception('User ID is not available. Please log in first.'); + } + state = const AsyncValue.loading(); + try { + final playlist = await (await metadataPlugin).playlist.create( + userId, + name: name, + description: description, + public: public, + collaborative: collaborative, + ); + if (playlist != null) { + state = AsyncValue.data(playlist); + } + ref.invalidate(metadataPluginSavedPlaylistsProvider); + } catch (e) { + onError?.call(e); + rethrow; + } + } + + Future modify({ + String? name, + String? description, + bool? public, + bool? collaborative, + void Function(dynamic error)? onError, + }) async { + try { + if (name == null && + description == null && + public == null && + collaborative == null) { + throw Exception('No modifications provided.'); + } + await (await metadataPlugin).playlist.update( + arg, + name: name, + description: description, + public: public, + collaborative: collaborative, + ); + ref.invalidateSelf(); + } on Exception catch (e) { + onError?.call(e); + rethrow; + } + } + + Future addTracks(List trackIds, + [void Function(dynamic error)? onError]) async { + if (state.value == null) return; + + try { + await ref + .read(metadataPluginSavedPlaylistsProvider.notifier) + .addTracks(arg, trackIds); + } catch (e) { + onError?.call(e); + rethrow; + } + } + + Future removeTracks(List trackIds, + [void Function(dynamic error)? onError]) async { + try { + if (state.value == null) return; + + await ref + .read(metadataPluginSavedPlaylistsProvider.notifier) + .removeTracks(arg, trackIds); + } catch (e) { + onError?.call(e); + rethrow; + } + } + + Future delete() async { + if (state.value == null) return; + final userId = await ref + .read(metadataPluginUserProvider.selectAsync((data) => data?.id)); + if (userId == null || userId != state.value!.owner.id) { + throw Exception('You can only delete your own playlists.'); + } + + await ref.read(metadataPluginSavedPlaylistsProvider.notifier).delete(arg); + } +} + +final metadataPluginPlaylistProvider = AutoDisposeAsyncNotifierProviderFamily< + MetadataPluginPlaylistNotifier, SpotubeFullPlaylistObject, String>( + () => MetadataPluginPlaylistNotifier(), +); diff --git a/lib/provider/metadata_plugin/search/albums.dart b/lib/provider/metadata_plugin/search/albums.dart new file mode 100644 index 00000000..40bb62e6 --- /dev/null +++ b/lib/provider/metadata_plugin/search/albums.dart @@ -0,0 +1,46 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginSearchAlbumsNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginSearchAlbumsNotifier() : super(); + + @override + fetch(offset, limit) async { + if (arg.isEmpty) { + return SpotubePaginationResponseObject( + limit: limit, + nextOffset: null, + total: 0, + items: [], + hasMore: false, + ); + } + + final res = await (await metadataPlugin).search.albums( + arg, + offset: offset, + limit: limit, + ); + + return res; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginSearchAlbumsProvider = + AutoDisposeAsyncNotifierProviderFamily, String>( + () => MetadataPluginSearchAlbumsNotifier(), +); diff --git a/lib/provider/metadata_plugin/search/all.dart b/lib/provider/metadata_plugin/search/all.dart new file mode 100644 index 00000000..4b051e58 --- /dev/null +++ b/lib/provider/metadata_plugin/search/all.dart @@ -0,0 +1,26 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +final metadataPluginSearchAllProvider = + FutureProvider.autoDispose.family( + (ref, query) async { + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + return metadataPlugin.search.all(query); + }, +); + +final metadataPluginSearchChipsProvider = FutureProvider((ref) async { + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + return metadataPlugin.search.chips; +}); diff --git a/lib/provider/metadata_plugin/search/artists.dart b/lib/provider/metadata_plugin/search/artists.dart new file mode 100644 index 00000000..b4d619f7 --- /dev/null +++ b/lib/provider/metadata_plugin/search/artists.dart @@ -0,0 +1,46 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginSearchArtistsNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginSearchArtistsNotifier() : super(); + + @override + fetch(offset, limit) async { + if (arg.isEmpty) { + return SpotubePaginationResponseObject( + limit: limit, + nextOffset: null, + total: 0, + items: [], + hasMore: false, + ); + } + + final res = await (await metadataPlugin).search.artists( + arg, + offset: offset, + limit: limit, + ); + + return res; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginSearchArtistsProvider = + AutoDisposeAsyncNotifierProviderFamily, String>( + () => MetadataPluginSearchArtistsNotifier(), +); diff --git a/lib/provider/metadata_plugin/search/playlists.dart b/lib/provider/metadata_plugin/search/playlists.dart new file mode 100644 index 00000000..dbf54250 --- /dev/null +++ b/lib/provider/metadata_plugin/search/playlists.dart @@ -0,0 +1,48 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginSearchPlaylistsNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginSearchPlaylistsNotifier() : super(); + + @override + fetch(offset, limit) async { + if (arg.isEmpty) { + return SpotubePaginationResponseObject( + limit: limit, + nextOffset: null, + total: 0, + items: [], + hasMore: false, + ); + } + + final res = await (await metadataPlugin).search.playlists( + arg, + offset: offset, + limit: limit, + ); + + return res; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginSearchPlaylistsProvider = + AutoDisposeAsyncNotifierProviderFamily< + MetadataPluginSearchPlaylistsNotifier, + SpotubePaginationResponseObject, + String>( + () => MetadataPluginSearchPlaylistsNotifier(), +); diff --git a/lib/provider/metadata_plugin/search/tracks.dart b/lib/provider/metadata_plugin/search/tracks.dart new file mode 100644 index 00000000..0b6ac141 --- /dev/null +++ b/lib/provider/metadata_plugin/search/tracks.dart @@ -0,0 +1,46 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; + +class MetadataPluginSearchTracksNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginSearchTracksNotifier() : super(); + + @override + fetch(offset, limit) async { + if (arg.isEmpty) { + return SpotubePaginationResponseObject( + limit: limit, + nextOffset: null, + total: 0, + items: [], + hasMore: false, + ); + } + + final tracks = await (await metadataPlugin).search.tracks( + arg, + offset: offset, + limit: limit, + ); + + return tracks; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginSearchTracksProvider = + AutoDisposeAsyncNotifierProviderFamily, String>( + () => MetadataPluginSearchTracksNotifier(), +); diff --git a/lib/provider/metadata_plugin/tracks/album.dart b/lib/provider/metadata_plugin/tracks/album.dart new file mode 100644 index 00000000..5491bdd0 --- /dev/null +++ b/lib/provider/metadata_plugin/tracks/album.dart @@ -0,0 +1,36 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; + +class MetadataPluginAlbumTracksNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginAlbumTracksNotifier() : super(); + + @override + fetch(offset, limit) async { + final tracks = await (await metadataPlugin).album.tracks( + arg, + offset: offset, + limit: limit, + ); + + return tracks; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginAlbumTracksProvider = + AutoDisposeAsyncNotifierProviderFamily, String>( + () => MetadataPluginAlbumTracksNotifier(), +); diff --git a/lib/provider/metadata_plugin/tracks/playlist.dart b/lib/provider/metadata_plugin/tracks/playlist.dart new file mode 100644 index 00000000..7fdd47db --- /dev/null +++ b/lib/provider/metadata_plugin/tracks/playlist.dart @@ -0,0 +1,36 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/provider/metadata_plugin/utils/family_paginated.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; + +class MetadataPluginPlaylistTracksNotifier + extends AutoDisposeFamilyPaginatedAsyncNotifier { + MetadataPluginPlaylistTracksNotifier() : super(); + + @override + fetch(offset, limit) async { + final tracks = await (await metadataPlugin).playlist.tracks( + arg, + offset: offset, + limit: limit, + ); + + return tracks; + } + + @override + build(arg) async { + ref.cacheFor(); + + ref.watch(metadataPluginProvider); + return await fetch(0, 20); + } +} + +final metadataPluginPlaylistTracksProvider = + AutoDisposeAsyncNotifierProviderFamily, String>( + () => MetadataPluginPlaylistTracksNotifier(), +); diff --git a/lib/provider/metadata_plugin/tracks/track.dart b/lib/provider/metadata_plugin/tracks/track.dart new file mode 100644 index 00000000..1beac43a --- /dev/null +++ b/lib/provider/metadata_plugin/tracks/track.dart @@ -0,0 +1,15 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +final metadataPluginTrackProvider = + FutureProvider.family((ref, trackId) async { + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + return metadataPlugin.track.getTrack(trackId); +}); diff --git a/lib/provider/metadata_plugin/updater/update_checker.dart b/lib/provider/metadata_plugin/updater/update_checker.dart new file mode 100644 index 00000000..6a7dc589 --- /dev/null +++ b/lib/provider/metadata_plugin/updater/update_checker.dart @@ -0,0 +1,32 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; + +final metadataPluginUpdateCheckerProvider = + FutureProvider((ref) async { + final metadataPluginConfigs = await ref.watch(metadataPluginsProvider.future); + final metadataPlugin = await ref.watch(metadataPluginProvider.future); + + if (metadataPlugin == null || + metadataPluginConfigs.defaultMetadataPluginConfig == null) { + return null; + } + + return metadataPlugin.core + .checkUpdate(metadataPluginConfigs.defaultMetadataPluginConfig!); +}); + +final audioSourcePluginUpdateCheckerProvider = + FutureProvider((ref) async { + final audioSourcePluginConfigs = + await ref.watch(metadataPluginsProvider.future); + final audioSourcePlugin = await ref.watch(audioSourcePluginProvider.future); + + if (audioSourcePlugin == null || + audioSourcePluginConfigs.defaultAudioSourcePluginConfig == null) { + return null; + } + + return audioSourcePlugin.core + .checkUpdate(audioSourcePluginConfigs.defaultAudioSourcePluginConfig!); +}); diff --git a/lib/provider/metadata_plugin/utils/common.dart b/lib/provider/metadata_plugin/utils/common.dart new file mode 100644 index 00000000..dc56e494 --- /dev/null +++ b/lib/provider/metadata_plugin/utils/common.dart @@ -0,0 +1,56 @@ +// ignore: implementation_imports +import 'package:riverpod/src/async_notifier.dart'; +import 'dart:async'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; +import 'package:spotube/services/metadata/metadata.dart'; + +extension PaginationExtension on AsyncValue { + bool get isLoadingNextPage => this is AsyncData && this is AsyncLoadingNext; +} + +mixin MetadataPluginMixin +// ignore: invalid_use_of_internal_member + on AsyncNotifierBase> { + Future get metadataPlugin async { + final plugin = await ref.read(metadataPluginProvider.future); + + if (plugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + return plugin; + } +} + +extension AutoDisposeAsyncNotifierCacheFor +// ignore: deprecated_member_use + on AutoDisposeAsyncNotifierProviderRef { + // When invoked keeps your provider alive for [duration] + // ignore: unused_element + void cacheFor([Duration duration = const Duration(minutes: 5)]) { + final link = keepAlive(); + final timer = Timer(duration, () => link.close()); + onDispose(() => timer.cancel()); + } +} + +// ignore: deprecated_member_use +extension AutoDisposeCacheFor on AutoDisposeRef { + // When invoked keeps your provider alive for [duration] + // ignore: unused_element + void cacheFor([Duration duration = const Duration(minutes: 5)]) { + final link = keepAlive(); + final timer = Timer(duration, () => link.close()); + onDispose(() => timer.cancel()); + } +} + +// ignore: subtype_of_sealed_class +class AsyncLoadingNext extends AsyncData { + const AsyncLoadingNext(super.value); +} diff --git a/lib/provider/metadata_plugin/utils/family_paginated.dart b/lib/provider/metadata_plugin/utils/family_paginated.dart new file mode 100644 index 00000000..b798dc8e --- /dev/null +++ b/lib/provider/metadata_plugin/utils/family_paginated.dart @@ -0,0 +1,141 @@ +import 'dart:async'; +import 'dart:math'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/logger/logger.dart'; + +abstract class FamilyPaginatedAsyncNotifier + extends FamilyAsyncNotifier, A> + with MetadataPluginMixin { + Future> fetch(int offset, int limit); + + Future fetchMore() async { + if (state.value == null || !state.value!.hasMore) return; + + final oldState = state.value; + + try { + state = AsyncLoadingNext(state.asData!.value); + + final newState = await fetch( + state.value!.nextOffset!, + state.value!.limit, + ); + + final oldItems = + state.value!.items.isEmpty ? [] : state.value!.items.cast(); + final items = newState.items.isEmpty ? [] : newState.items.cast(); + + state = AsyncData(newState.copyWith(items: [...oldItems, ...items])); + } catch (e, stack) { + AppLogger.reportError(e, stack); + state = AsyncData(oldState!); + } + } + + Future> fetchAll() async { + if (state.value == null) return []; + if (!state.value!.hasMore) return state.value!.items.cast(); + + bool hasMore = true; + while (hasMore) { + final newState = await fetch( + state.value!.nextOffset!, + max(state.value!.limit, 100), + ) + .catchError( + (e) => fetch(state.value!.nextOffset!, max(state.value!.limit, 50)), + ) + .catchError( + (e) => fetch(state.value!.nextOffset!, state.value!.limit), + ) + .catchError( + (e) async { + await Future.delayed(const Duration(milliseconds: 500)); + return fetch(state.value!.nextOffset!, state.value!.limit); + }, + ); + + hasMore = newState.hasMore; + + final oldItems = + state.value!.items.isEmpty ? [] : state.value!.items.cast(); + final items = newState.items.isEmpty ? [] : newState.items.cast(); + + state = AsyncData( + newState.copyWith(items: [...oldItems, ...items]), + ); + } + + return state.value!.items.cast(); + } +} + +abstract class AutoDisposeFamilyPaginatedAsyncNotifier + extends AutoDisposeFamilyAsyncNotifier, + A> with MetadataPluginMixin { + Future> fetch(int offset, int limit); + + Future fetchMore() async { + if (state.value == null || !state.value!.hasMore) return; + final oldState = state.value; + + try { + state = AsyncLoadingNext(state.value!); + + final newState = await fetch( + state.value!.nextOffset!, + state.value!.limit, + ); + + state = AsyncData( + newState.copyWith(items: [ + ...state.value!.items.cast(), + ...newState.items.cast(), + ]), + ); + } catch (e, stack) { + AppLogger.reportError(e, stack); + state = AsyncData(oldState!); + } + } + + Future> fetchAll() async { + if (state.value == null) return []; + if (!state.value!.hasMore) return state.value!.items.cast(); + + bool hasMore = true; + while (hasMore) { + final newState = await fetch( + state.value!.nextOffset!, + max(state.value!.limit, 100), + ) + .catchError( + (e) => fetch(state.value!.nextOffset!, max(state.value!.limit, 50)), + ) + .catchError( + (e) => fetch(state.value!.nextOffset!, state.value!.limit), + ) + .catchError( + (e) async { + await Future.delayed(const Duration(milliseconds: 500)); + return fetch(state.value!.nextOffset!, state.value!.limit); + }, + ); + + hasMore = newState.hasMore; + + final oldItems = + state.value!.items.isEmpty ? [] : state.value!.items.cast(); + final items = newState.items.isEmpty ? [] : newState.items.cast(); + + state = AsyncData( + newState.copyWith(items: [...oldItems, ...items]), + ); + } + + return state.value!.items.cast(); + } +} diff --git a/lib/provider/metadata_plugin/utils/paginated.dart b/lib/provider/metadata_plugin/utils/paginated.dart new file mode 100644 index 00000000..4c77441a --- /dev/null +++ b/lib/provider/metadata_plugin/utils/paginated.dart @@ -0,0 +1,83 @@ +import 'dart:async'; +import 'dart:math'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +// ignore: implementation_imports +import 'package:riverpod/src/async_notifier.dart'; +import 'package:spotube/provider/metadata_plugin/utils/common.dart'; +import 'package:spotube/services/logger/logger.dart'; + +mixin PaginatedAsyncNotifierMixin + // ignore: invalid_use_of_internal_member + on AsyncNotifierBase> { + Future> fetch(int offset, int limit); + + Future fetchMore() async { + if (state.value == null || !state.value!.hasMore) return; + + final oldState = state.value; + try { + state = AsyncLoadingNext(state.asData!.value); + + final newState = await fetch( + state.value!.nextOffset!, + state.value!.limit, + ); + + final oldItems = + state.value!.items.isEmpty ? [] : state.value!.items.cast(); + final items = newState.items.isEmpty ? [] : newState.items.cast(); + + state = AsyncData(newState.copyWith(items: [...oldItems, ...items])); + } catch (e, stack) { + AppLogger.reportError(e, stack); + state = AsyncData(oldState!); + } + } + + Future> fetchAll() async { + if (state.value == null) return []; + if (!state.value!.hasMore) return state.value!.items.cast(); + + bool hasMore = true; + while (hasMore) { + final newState = await fetch( + state.value!.nextOffset!, + max(state.value!.limit, 100), + ) + .catchError( + (e) => fetch(state.value!.nextOffset!, max(state.value!.limit, 50)), + ) + .catchError( + (e) => fetch(state.value!.nextOffset!, state.value!.limit), + ) + .catchError( + (e) async { + await Future.delayed(const Duration(milliseconds: 500)); + return fetch(state.value!.nextOffset!, state.value!.limit); + }, + ); + + hasMore = newState.hasMore; + + final oldItems = + state.value!.items.isEmpty ? [] : state.value!.items.cast(); + final items = newState.items.isEmpty ? [] : newState.items.cast(); + + state = AsyncData( + newState.copyWith(items: [...oldItems, ...items]), + ); + } + + return state.value!.items.cast(); + } +} + +abstract class PaginatedAsyncNotifier + extends AsyncNotifier> + with PaginatedAsyncNotifierMixin, MetadataPluginMixin {} + +abstract class AutoDisposePaginatedAsyncNotifier + extends AutoDisposeAsyncNotifier> + with PaginatedAsyncNotifierMixin, MetadataPluginMixin {} diff --git a/lib/provider/palette_provider.dart b/lib/provider/palette_provider.dart deleted file mode 100644 index 8f0e9e29..00000000 --- a/lib/provider/palette_provider.dart +++ /dev/null @@ -1,4 +0,0 @@ -import 'package:palette_generator/palette_generator.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; - -final paletteProvider = StateProvider((ref) => null); diff --git a/lib/provider/scrobbler/scrobbler.dart b/lib/provider/scrobbler/scrobbler.dart index 8aff0438..f5e5556d 100644 --- a/lib/provider/scrobbler/scrobbler.dart +++ b/lib/provider/scrobbler/scrobbler.dart @@ -3,16 +3,15 @@ import 'dart:async'; import 'package:drift/drift.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:scrobblenaut/scrobblenaut.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/collections/env.dart'; -import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; import 'package:spotube/services/logger/logger.dart'; class ScrobblerNotifier extends AsyncNotifier { - final StreamController _scrobbleController = - StreamController.broadcast(); + final StreamController _scrobbleController = + StreamController.broadcast(); @override build() async { final database = ref.watch(databaseProvider); @@ -47,13 +46,12 @@ class ScrobblerNotifier extends AsyncNotifier { _scrobbleController.stream.listen((track) async { try { await state.asData?.value?.track.scrobble( - artist: track.artists!.first.name!, - track: track.name!, - album: track.album!.name!, + artist: track.artists.first.name, + track: track.name, + album: track.album.name, chosenByUser: true, - duration: track.duration, + duration: Duration(milliseconds: track.durationMs), timestamp: DateTime.now().toUtc(), - trackNumber: track.trackNumber, ); } catch (e, stackTrace) { AppLogger.reportError(e, stackTrace); @@ -109,21 +107,21 @@ class ScrobblerNotifier extends AsyncNotifier { await database.delete(database.scrobblerTable).go(); } - void scrobble(Track track) { + void scrobble(SpotubeTrackObject track) { _scrobbleController.add(track); } - Future love(Track track) async { + Future love(SpotubeTrackObject track) async { await state.asData?.value?.track.love( - artist: track.artists!.asString(), - track: track.name!, + artist: track.artists.asString(), + track: track.name, ); } - Future unlove(Track track) async { + Future unlove(SpotubeTrackObject track) async { await state.asData?.value?.track.unLove( - artist: track.artists!.asString(), - track: track.name!, + artist: track.artists.asString(), + track: track.name, ); } } diff --git a/lib/provider/server/active_sourced_track.dart b/lib/provider/server/active_sourced_track.dart deleted file mode 100644 index 37d0dec8..00000000 --- a/lib/provider/server/active_sourced_track.dart +++ /dev/null @@ -1,47 +0,0 @@ -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/services/audio_player/audio_player.dart'; -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; - -class ActiveSourcedTrackNotifier extends Notifier { - @override - build() { - return null; - } - - void update(SourcedTrack? sourcedTrack) { - state = sourcedTrack; - } - - Future populateSibling() async { - if (state == null) return; - state = await state!.copyWithSibling(); - } - - Future swapSibling(SourceInfo sibling) async { - if (state == null) return; - await populateSibling(); - final newTrack = await state!.swapWithSibling(sibling); - if (newTrack == null) return; - - state = newTrack; - await audioPlayer.pause(); - - final playbackNotifier = ref.read(audioPlayerProvider.notifier); - final oldActiveIndex = audioPlayer.currentIndex; - - await playbackNotifier.addTracksAtFirst([newTrack], allowDuplicates: true); - await Future.delayed(const Duration(milliseconds: 50)); - await playbackNotifier.jumpToTrack(newTrack); - - await audioPlayer.removeTrack(oldActiveIndex); - - await audioPlayer.resume(); - } -} - -final activeSourcedTrackProvider = - NotifierProvider( - () => ActiveSourcedTrackNotifier(), -); diff --git a/lib/provider/server/active_track_sources.dart b/lib/provider/server/active_track_sources.dart new file mode 100644 index 00000000..603ca0e4 --- /dev/null +++ b/lib/provider/server/active_track_sources.dart @@ -0,0 +1,43 @@ +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; +import 'package:spotube/services/sourced_track/sourced_track.dart'; + +final activeTrackSourcesProvider = FutureProvider< + ({ + SourcedTrack? source, + SourcedTrackNotifier? notifier, + SpotubeTrackObject track, + })?>((ref) async { + final audioPlayerState = ref.watch(audioPlayerProvider); + + if (audioPlayerState.activeTrack == null) { + return null; + } + + if (audioPlayerState.activeTrack is SpotubeLocalTrackObject) { + return ( + source: null, + notifier: null, + track: audioPlayerState.activeTrack!, + ); + } + + final sourcedTrack = await ref.watch( + sourcedTrackProvider( + audioPlayerState.activeTrack! as SpotubeFullTrackObject, + ).future, + ); + final sourcedTrackNotifier = ref.watch( + sourcedTrackProvider( + audioPlayerState.activeTrack! as SpotubeFullTrackObject, + ).notifier, + ); + + return ( + source: sourcedTrack, + track: audioPlayerState.activeTrack!, + notifier: sourcedTrackNotifier, + ); +}); diff --git a/lib/provider/server/router.dart b/lib/provider/server/router.dart index e2a579cc..f103ea8c 100644 --- a/lib/provider/server/router.dart +++ b/lib/provider/server/router.dart @@ -12,8 +12,13 @@ final serverRouterProvider = Provider((ref) { router.get("/ping", (Request request) => Response.ok("pong")); + router.head("/stream/", playbackRoutes.headStreamTrackId); router.get("/stream/", playbackRoutes.getStreamTrackId); + router.get("/playback/toggle-playback", playbackRoutes.togglePlayback); + router.get("/playback/previous", playbackRoutes.previousTrack); + router.get("/playback/next", playbackRoutes.nextTrack); + router.all("/ws", connectRoutes.websocket); return router; diff --git a/lib/provider/server/routes/connect.dart b/lib/provider/server/routes/connect.dart index 0d35b473..257c4cb4 100644 --- a/lib/provider/server/routes/connect.dart +++ b/lib/provider/server/routes/connect.dart @@ -3,10 +3,13 @@ import 'dart:convert'; import 'dart:io'; import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:shelf/shelf.dart'; import 'package:shelf_web_socket/shelf_web_socket.dart'; -import 'package:spotify/spotify.dart'; +import 'package:spotube/collections/routes.dart'; +import 'package:spotube/extensions/context.dart'; import 'package:spotube/models/connect/connect.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/history/history.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; @@ -43,6 +46,8 @@ class ServerConnectRoutes { Stream get connectClientStream => _connectClientStreamController.stream; + final List _allowedConnections = []; + FutureOr websocket(Request req) { return webSocketHandler( ( @@ -54,6 +59,47 @@ class ServerConnectRoutes { final origin = "${context?.remoteAddress.host}:${context?.remotePort}"; _connectClientStreamController.add(origin); + // Confirm whether user allows to connect + if (rootNavigatorKey.currentContext?.mounted == true && + _allowedConnections.contains(origin) == false) { + final confirmed = await showDialog( + context: rootNavigatorKey.currentContext!, + builder: (context) { + return AlertDialog( + title: Text(context.l10n.connect), + content: Text( + context.l10n.connect_request(origin), + ), + actions: [ + Button.secondary( + onPressed: () { + Navigator.of(context).pop(false); + }, + child: Text(context.l10n.decline), + ), + Button.primary( + onPressed: () { + Navigator.of(context).pop(true); + }, + child: Text(context.l10n.accept), + ), + ], + ); + }, + ) ?? + false; + + if (confirmed) { + _allowedConnections.add(origin); + } else { + channel.sink.addEvent( + WebSocketErrorEvent("Connection denied"), + ); + await channel.sink.close(); + return; + } + } + ref.listen( audioPlayerProvider, (previous, next) { @@ -106,7 +152,7 @@ class ServerConnectRoutes { }, ), channel.stream.listen( - (message) { + (message) async { try { final event = WebSocketEvent.fromJson( jsonDecode(message), @@ -115,19 +161,19 @@ class ServerConnectRoutes { event.onLoad((event) async { await audioPlayerNotifier.load( - event.data.tracks, + event.data.tracks.cast().toList(), autoPlay: true, initialIndex: event.data.initialIndex ?? 0, ); if (event.data.collectionId == null) return; audioPlayerNotifier.addCollection(event.data.collectionId!); - if (event.data.collection is AlbumSimple) { - historyNotifier - .addAlbums([event.data.collection as AlbumSimple]); + if (event.data.collection is SpotubeSimpleAlbumObject) { + historyNotifier.addAlbums( + [event.data.collection as SpotubeSimpleAlbumObject]); } else { historyNotifier.addPlaylists( - [event.data.collection as PlaylistSimple]); + [event.data.collection as SpotubeSimplePlaylistObject]); } }); @@ -140,7 +186,7 @@ class ServerConnectRoutes { }); event.onStop((event) async { - await audioPlayer.stop(); + await ref.read(audioPlayerProvider.notifier).stop(); }); event.onNext((event) async { diff --git a/lib/provider/server/routes/playback.dart b/lib/provider/server/routes/playback.dart index 34317aa1..db6bf8f5 100644 --- a/lib/provider/server/routes/playback.dart +++ b/lib/provider/server/routes/playback.dart @@ -1,4 +1,6 @@ +import 'dart:async'; import 'dart:io'; +import 'dart:math'; import 'package:dio/dio.dart' hide Response; import 'package:dio/dio.dart' as dio_lib; @@ -7,21 +9,32 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:metadata_god/metadata_god.dart'; import 'package:path/path.dart'; import 'package:shelf/shelf.dart'; -import 'package:spotube/extensions/artist_simple.dart'; -import 'package:spotube/extensions/image.dart'; -import 'package:spotube/extensions/track.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/models/parser/range_headers.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/audio_player/state.dart'; -import 'package:spotube/provider/server/active_sourced_track.dart'; -import 'package:spotube/provider/server/sourced_track.dart'; +import 'package:spotube/provider/server/active_track_sources.dart'; +import 'package:spotube/provider/server/sourced_track_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/logger/logger.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; import 'package:spotube/services/sourced_track/sourced_track.dart'; import 'package:spotube/utils/service_utils.dart'; +import 'package:youtube_explode_dart/youtube_explode_dart.dart'; + +final _deviceClients = Set.unmodifiable({ + YoutubeApiClient.ios, + YoutubeApiClient.android, + YoutubeApiClient.mweb, + YoutubeApiClient.safari, +}); + +String? get _randomUserAgent => _deviceClients + .elementAt( + Random().nextInt(_deviceClients.length), + ) + .payload["context"]["client"]["userAgent"]; class ServerPlaybackRoutes { final Ref ref; @@ -31,156 +44,252 @@ class ServerPlaybackRoutes { ServerPlaybackRoutes(this.ref) : dio = Dio(); - Future<({dio_lib.Response response, Uint8List? bytes})> - streamTrack( - SourcedTrack track, - Map headers, - ) async { - final trackCacheFile = File( - join( - await UserPreferencesNotifier.getMusicCacheDir(), - '${track.name} - ${track.artists?.asString()} (${track.sourceInfo.id}).${track.codec.name}', + Future _getTrackCacheFilePath(SourcedTrack track) async { + return join( + await UserPreferencesNotifier.getMusicCacheDir(), + ServiceUtils.sanitizeFilename( + '${track.query.name} - ${track.query.artists.map((d) => d.name).join(",")} (${track.info.id}).${track.qualityPreset!.getFileExtension()}', ), ); - final trackPartialCacheFile = File("${trackCacheFile.path}.part"); + } - var options = Options( + Future _getSourcedTrack( + Request request, + String trackId, + ) async { + final track = + playlist.tracks.firstWhere((element) => element.id == trackId); + + final activeSourcedTrack = + await ref.read(activeTrackSourcesProvider.future); + + final media = audioPlayer.playlist.medias + .firstWhere((e) => e.uri == request.requestedUri.toString()); + final spotubeMedia = + media is SpotubeMedia ? media : SpotubeMedia.media(media); + final sourcedTrack = activeSourcedTrack?.track.id == track.id + ? activeSourcedTrack?.source + : await ref.read( + sourcedTrackProvider(spotubeMedia.track as SpotubeFullTrackObject) + .future, + ); + + return sourcedTrack; + } + + Future streamTrackInformation( + Request request, + SourcedTrack track, + ) async { + AppLogger.log.i( + "HEAD request for track: ${track.query.name}\n" + "Headers: ${request.headers}", + ); + + final trackCacheFile = File(await _getTrackCacheFilePath(track)); + + if (await trackCacheFile.exists() && userPreferences.cacheMusic) { + final fileLength = await trackCacheFile.length(); + + return dio_lib.Response( + statusCode: 200, + headers: Headers.fromMap({ + "content-type": ["audio/${track.qualityPreset!.name}"], + "content-length": ["$fileLength"], + "accept-ranges": ["bytes"], + "content-range": ["bytes 0-$fileLength/$fileLength"], + }), + requestOptions: RequestOptions(path: request.requestedUri.toString()), + ); + } + + String url = track.url ?? + await ref + .read(sourcedTrackProvider(track.query).notifier) + .swapWithNextSibling() + .then((track) => track.url!); + + final options = Options( headers: { - ...headers, - "User-Agent": - "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36", - "Cache-Control": "max-age=0", + "user-agent": _randomUserAgent, + "Cache-Control": "max-age=3600", "Connection": "keep-alive", - "host": Uri.parse(track.url).host, + "host": Uri.parse(url).host, }, - responseType: ResponseType.bytes, validateStatus: (status) => status! < 400, ); - final headersRes = await Future.value( - dio.head( - track.url, - options: options, - ), - ).catchError((_) async => null); + final res = await dio.head(url, options: options); - final contentLength = headersRes?.headers.value("content-length"); + return res; + } + + Future streamTrack( + Request request, + SourcedTrack track, + Map headers, + ) async { + AppLogger.log.i( + "GET request for track: ${track.query.name}\n" + "Headers: ${request.headers}", + ); + + final trackCacheFile = File(await _getTrackCacheFilePath(track)); if (await trackCacheFile.exists() && userPreferences.cacheMusic) { final bytes = await trackCacheFile.readAsBytes(); final cachedFileLength = bytes.length; - return ( - response: dio_lib.Response( - statusCode: 200, - headers: Headers.fromMap({ - "content-type": ["audio/${track.codec.name}"], - "content-length": ["$cachedFileLength"], - "accept-ranges": ["bytes"], - "content-range": ["bytes 0-$cachedFileLength/$cachedFileLength"], - }), - requestOptions: RequestOptions(path: track.url), - ), - bytes: bytes, + return dio_lib.Response( + statusCode: 200, + headers: Headers.fromMap({ + "content-type": ["audio/${track.qualityPreset!.name}"], + "content-length": ["${cachedFileLength - 1}"], + "accept-ranges": ["bytes"], + "content-range": [ + "bytes 0-${cachedFileLength - 1}/$cachedFileLength" + ], + "connection": ["close"], + }), + requestOptions: RequestOptions(path: request.requestedUri.toString()), + data: bytes, ); } - /// Forcing partial content range as mpv sometimes greedily wants - /// everything at one go. Slows down overall streaming. - final range = RangeHeader.parse(headers["range"] ?? ""); - final contentPartialLength = int.tryParse(contentLength ?? ""); - if ((range.end == null) && - contentPartialLength != null && - range.start == 0) { - options = options.copyWith( - headers: { - ...?options.headers, - "range": "$range${(contentPartialLength * 0.3).ceil()}", - }, - ); - } + String url = track.url ?? + await ref + .read(sourcedTrackProvider(track.query).notifier) + .swapWithNextSibling() + .then((track) => track.url!); - final res = - await dio.get(track.url, options: options).catchError( - (e, stack) async { - final sourcedTrack = await ref - .read(sourcedTrackProvider(SpotubeMedia(track)).notifier) - .switchToAlternativeSources(); - - ref.read(activeSourcedTrackProvider.notifier).update(sourcedTrack); - - return await dio.get(sourcedTrack!.url, options: options); + final options = Options( + headers: { + ...headers, + "user-agent": _randomUserAgent, + "Cache-Control": "max-age=3600", + "Connection": "keep-alive", + "host": Uri.parse(url).host, }, + responseType: ResponseType.stream, + validateStatus: (status) => status! < 400, ); - final bytes = res.data; + final contentLengthRes = await Future.value( + dio.head( + url, + options: options.copyWith(responseType: ResponseType.bytes), + ), + ).catchError((e, stack) async { + AppLogger.reportError(e, stack); - if (bytes == null || !userPreferences.cacheMusic) { - return (response: res, bytes: bytes); + final sourcedTrack = await ref + .read(sourcedTrackProvider(track.query).notifier) + .refreshStreamingUrl(); + + url = sourcedTrack.url!; + + return dio.head(url, options: options); + }); + + // Redirect to m3u8 link directly as it handles range requests internally + if (contentLengthRes?.headers.value("content-type") == + "application/vnd.apple.mpegurl") { + return dio_lib.Response( + statusCode: 301, + statusMessage: "M3U8 Redirect", + headers: Headers.fromMap({ + "location": [url], + "content-type": ["application/vnd.apple.mpegurl"], + }), + requestOptions: RequestOptions(path: request.requestedUri.toString()), + isRedirect: true, + ); } - final contentRange = - ContentRangeHeader.parse(res.headers.value("content-range") ?? ""); + final res = await dio.get(url, options: options); + AppLogger.log.i( + "Response for track: ${track.query.name}\n" + "Status Code: ${res.statusCode}\n" + "Headers: ${res.headers.map}", + ); + + if (!userPreferences.cacheMusic) { + return res; + } + + final resStream = res.data!.stream.asBroadcastStream(); + + final trackPartialCacheFile = File("${trackCacheFile.path}.part"); if (!await trackPartialCacheFile.exists()) { await trackPartialCacheFile.create(recursive: true); } // Write the stream to the file based on the range - final partialCacheFile = - await trackPartialCacheFile.open(mode: FileMode.writeOnlyAppend); - int fileLength = 0; - try { - await partialCacheFile.setPosition(contentRange.start); - await partialCacheFile.writeFrom(bytes); - fileLength = await partialCacheFile.length(); - } finally { - await partialCacheFile.close(); - } + final partialCacheFileSink = + trackPartialCacheFile.openWrite(mode: FileMode.writeOnlyAppend); + final contentRange = res.headers.value("content-range") != null + ? ContentRangeHeader.parse(res.headers.value("content-range") ?? "") + : ContentRangeHeader(0, 0, 0); - if (fileLength == contentRange.total) { - await trackPartialCacheFile.rename(trackCacheFile.path); - } + resStream.listen( + (data) { + partialCacheFileSink.add(data); + }, + onError: (e, stack) { + partialCacheFileSink.close(); + }, + onDone: () async { + await partialCacheFileSink.close(); - if (contentRange.total == fileLength && track.codec != SourceCodecs.weba) { - final imageBytes = await ServiceUtils.downloadImage( - (track.album?.images).asUrlString( - placeholder: ImagePlaceholder.albumArt, - index: 1, - ), - ); + final fileLength = await trackPartialCacheFile.length(); + if (fileLength != contentRange.total) return; - await MetadataGod.writeMetadata( - file: trackCacheFile.path, - metadata: track.toMetadata( - fileLength: fileLength, - imageBytes: imageBytes, - ), - ); - } + await trackPartialCacheFile.rename(trackCacheFile.path); - return (bytes: bytes, response: res); + if (track.qualityPreset!.getFileExtension() == "weba") return; + + final imageBytes = await ServiceUtils.downloadImage( + track.query.album.images.asUrlString( + placeholder: ImagePlaceholder.albumArt, + index: 1, + ), + ); + + await MetadataGod.writeMetadata( + file: trackCacheFile.path, + metadata: track.query.toMetadata( + imageBytes: imageBytes, + fileLength: fileLength, + ), + ).catchError((e, stackTrace) { + AppLogger.reportError(e, stackTrace); + }); + }, + cancelOnError: true, + ); + + res.data?.stream = + resStream; // To avoid Stream has been already listened to exception + return res; } - /// @get('/stream/') - Future getStreamTrackId(Request request, String trackId) async { + /// @head('/stream/') + Future headStreamTrackId(Request request, String trackId) async { try { - final track = - playlist.tracks.firstWhere((element) => element.id == trackId); + final sourcedTrack = await _getSourcedTrack(request, trackId); - final activeSourcedTrack = ref.read(activeSourcedTrackProvider); - final sourcedTrack = activeSourcedTrack?.id == track.id - ? activeSourcedTrack - : await ref.read(sourcedTrackProvider(SpotubeMedia(track)).future); + if (sourcedTrack == null) { + return Response.notFound("Track not found in the current queue"); + } - ref.read(activeSourcedTrackProvider.notifier).update(sourcedTrack); - - final (bytes: audioBytes, response: res) = - await streamTrack(sourcedTrack!, request.headers); + final res = await streamTrackInformation( + request, + sourcedTrack, + ); return Response( res.statusCode!, - body: audioBytes, headers: res.headers.map, ); } catch (e, stack) { @@ -188,6 +297,61 @@ class ServerPlaybackRoutes { return Response.internalServerError(); } } + + /// @get('/stream/') + Future getStreamTrackId(Request request, String trackId) async { + try { + final sourcedTrack = await _getSourcedTrack(request, trackId); + + if (sourcedTrack == null) { + return Response.notFound("Track not found in the current queue"); + } + + final res = await streamTrack( + request, + sourcedTrack, + request.headers, + ); + + if (res.data is ResponseBody) { + return Response( + res.statusCode!, + body: (res.data as ResponseBody).stream, + headers: res.headers.map, + ); + } + + return Response( + res.statusCode!, + body: res.data, + headers: res.headers.map, + ); + } catch (e, stack) { + AppLogger.reportError(e, stack); + return Response.internalServerError(); + } + } + + /// @get('/playback/toggle-playback') + Future togglePlayback(Request request) async { + audioPlayer.isPlaying + ? await audioPlayer.pause() + : await audioPlayer.resume(); + + return Response.ok("Playback toggled"); + } + + /// @get('/playback/previous') + Future previousTrack(Request request) async { + await audioPlayer.skipToPrevious(); + return Response.ok("Previous track"); + } + + /// @get('/playback/next') + Future nextTrack(Request request) async { + await audioPlayer.skipToNext(); + return Response.ok("Next track"); + } } final serverPlaybackRoutesProvider = diff --git a/lib/provider/server/server.dart b/lib/provider/server/server.dart index 131f1ea4..d10815bf 100644 --- a/lib/provider/server/server.dart +++ b/lib/provider/server/server.dart @@ -5,30 +5,51 @@ import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:shelf/shelf_io.dart'; import 'package:spotube/provider/server/pipeline.dart'; import 'package:spotube/provider/server/router.dart'; +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/logger/logger.dart'; final serverProvider = FutureProvider( (ref) async { + final enabledRemoteConnect = ref.watch( + userPreferencesProvider.select((value) => value.enableConnect), + ); + final connectPort = ref.watch( + userPreferencesProvider.select((value) => value.connectPort), + ); final pipeline = ref.watch(pipelineProvider); final router = ref.watch(serverRouterProvider); - final port = Random().nextInt(17500) + 5000; - SpotubeMedia.serverPort = port; + // When connect port is -1, we need to generate a random port + // but we shouldn't reset it if it's already been set (caused by a state change) + if (connectPort == -1) { + if (SpotubeMedia.serverPort == 0) { + final port = Random().nextInt(17500) + 5000; + SpotubeMedia.serverPort = port; + } + } else { + SpotubeMedia.serverPort = connectPort; + } final server = await serve( pipeline.addHandler(router.call), - InternetAddress.anyIPv4, - port, + enabledRemoteConnect + ? InternetAddress.anyIPv4 + : InternetAddress.loopbackIPv4, + SpotubeMedia.serverPort, ); - AppLogger.log - .t('Playback server at http://${server.address.host}:${server.port}'); + AppLogger.log.t( + 'Playback server at http://${server.address.host}:${server.port}', + ); ref.onDispose(() { server.close(); }); - return (server: server, port: port); + return ( + server: server, + port: SpotubeMedia.serverPort, + ); }, ); diff --git a/lib/provider/server/sourced_track.dart b/lib/provider/server/sourced_track.dart deleted file mode 100644 index 58531523..00000000 --- a/lib/provider/server/sourced_track.dart +++ /dev/null @@ -1,48 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/models/local_track.dart'; -import 'package:spotube/provider/audio_player/audio_player.dart'; -import 'package:spotube/services/audio_player/audio_player.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; - -class SourcedTrackNotifier - extends FamilyAsyncNotifier { - @override - build(media) async { - final track = media?.track; - if (track == null || track is LocalTrack) { - return null; - } - - ref.listen( - audioPlayerProvider.select((value) => value.tracks), - (old, next) { - if (next.isEmpty || next.none((element) => element.id == track.id)) { - ref.invalidateSelf(); - } - }, - ); - - final sourcedTrack = - await SourcedTrack.fetchFromTrack(track: track, ref: ref); - - return sourcedTrack; - } - - Future switchToAlternativeSources() async { - if (arg == null) { - return null; - } - return await update((prev) async { - return await SourcedTrack.fetchFromTrackAltSource( - track: arg!.track, - ref: ref, - ); - }); - } -} - -final sourcedTrackProvider = AsyncNotifierProviderFamily( - () => SourcedTrackNotifier(), -); diff --git a/lib/provider/server/sourced_track_provider.dart b/lib/provider/server/sourced_track_provider.dart new file mode 100644 index 00000000..7934ecc7 --- /dev/null +++ b/lib/provider/server/sourced_track_provider.dart @@ -0,0 +1,49 @@ +import 'dart:async'; + +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/metadata_plugin/audio_source/quality_presets.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/sourced_track/sourced_track.dart'; + +class SourcedTrackNotifier + extends FamilyAsyncNotifier { + @override + FutureOr build(query) { + ref.watch(audioSourcePluginProvider); + ref.watch(audioSourcePresetsProvider); + + return SourcedTrack.fetchFromTrack(query: query, ref: ref); + } + + Future refreshStreamingUrl() async { + return await update((prev) async { + return await prev.refreshStream(); + }); + } + + Future copyWithSibling() async { + return await update((prev) async { + return prev.copyWithSibling(); + }); + } + + Future swapWithSibling( + SpotubeAudioSourceMatchObject sibling, + ) async { + return await update((prev) async { + return await prev.swapWithSibling(sibling) ?? prev; + }); + } + + Future swapWithNextSibling() async { + return await update((prev) async { + return await prev.swapWithSibling(prev.siblings.first) as SourcedTrack; + }); + } +} + +final sourcedTrackProvider = AsyncNotifierProviderFamily( + () => SourcedTrackNotifier(), +); diff --git a/lib/provider/skip_segments/skip_segments.dart b/lib/provider/skip_segments/skip_segments.dart index 005797f4..dc06f326 100644 --- a/lib/provider/skip_segments/skip_segments.dart +++ b/lib/provider/skip_segments/skip_segments.dart @@ -1,9 +1,10 @@ import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/database/database.dart'; import 'package:spotube/services/logger/logger.dart'; import 'package:dio/dio.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/provider/server/active_sourced_track.dart'; +import 'package:spotube/provider/server/active_track_sources.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/services/dio/dio.dart'; @@ -81,31 +82,23 @@ Future> getAndCacheSkipSegments( final segmentProvider = FutureProvider( (ref) async { - final track = ref.watch(activeSourcedTrackProvider); - if (track == null) return null; + final snapshot = await ref.watch(activeTrackSourcesProvider.future); + if (snapshot == null) return null; + final (:track, :source, :notifier) = snapshot; + if (track is SpotubeLocalTrackObject) return null; + if (!source!.source.toLowerCase().contains("youtube")) return null; - final skipNonMusic = ref.watch( - userPreferencesProvider.select( - (s) { - final isPipedYTMusicMode = s.audioSource == AudioSource.piped && - s.searchMode == SearchMode.youtubeMusic; - - return s.skipNonMusic && !isPipedYTMusicMode; - }, - ), - ); + final skipNonMusic = + ref.watch(userPreferencesProvider.select((s) => s.skipNonMusic)); if (!skipNonMusic) { - return SourcedSegments( - segments: [], - source: track.sourceInfo.id, - ); + return SourcedSegments(segments: [], source: source.info.id); } - final segments = await getAndCacheSkipSegments(track.sourceInfo.id, ref); + final segments = await getAndCacheSkipSegments(source.info.id, ref); return SourcedSegments( - source: track.sourceInfo.id, + source: source.info.id, segments: segments, ); }, diff --git a/lib/provider/spotify/album/favorite.dart b/lib/provider/spotify/album/favorite.dart deleted file mode 100644 index cf444d49..00000000 --- a/lib/provider/spotify/album/favorite.dart +++ /dev/null @@ -1,86 +0,0 @@ -part of '../spotify.dart'; - -class FavoriteAlbumState extends PaginatedState { - FavoriteAlbumState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - FavoriteAlbumState copyWith({items, offset, limit, hasMore}) { - return FavoriteAlbumState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class FavoriteAlbumNotifier - extends PaginatedAsyncNotifier { - @override - Future> fetch(int offset, int limit) { - return spotify.me - .savedAlbums() - .getPage(limit, offset) - .then((value) => value.items?.toList() ?? []); - } - - @override - build() async { - ref.watch(spotifyProvider); - final items = await fetch(0, 20); - return FavoriteAlbumState( - items: items, - offset: 0, - limit: 20, - hasMore: items.length == 20, - ); - } - - Future addFavorites(List ids) async { - if (state.value == null) return; - - state = await AsyncValue.guard(() async { - await spotify.me.saveAlbums(ids); - final albums = await spotify.albums.list(ids); - - return state.value!.copyWith( - items: [ - ...state.value!.items, - ...albums, - ], - ); - }); - - for (final id in ids) { - ref.invalidate(albumsIsSavedProvider(id)); - } - } - - Future removeFavorites(List ids) async { - if (state.value == null) return; - - state = await AsyncValue.guard(() async { - await spotify.me.removeAlbums(ids); - - return state.value!.copyWith( - items: state.value!.items - .where((element) => !ids.contains(element.id)) - .toList(), - ); - }); - - for (final id in ids) { - ref.invalidate(albumsIsSavedProvider(id)); - } - } -} - -final favoriteAlbumsProvider = - AsyncNotifierProvider( - () => FavoriteAlbumNotifier(), -); diff --git a/lib/provider/spotify/album/is_saved.dart b/lib/provider/spotify/album/is_saved.dart deleted file mode 100644 index 987ccdf2..00000000 --- a/lib/provider/spotify/album/is_saved.dart +++ /dev/null @@ -1,10 +0,0 @@ -part of '../spotify.dart'; - -final albumsIsSavedProvider = FutureProvider.autoDispose.family( - (ref, albumId) async { - final spotify = ref.watch(spotifyProvider); - return spotify.me.containsSavedAlbums([albumId]).then( - (value) => value[albumId] ?? false, - ); - }, -); diff --git a/lib/provider/spotify/album/releases.dart b/lib/provider/spotify/album/releases.dart deleted file mode 100644 index 43d2e474..00000000 --- a/lib/provider/spotify/album/releases.dart +++ /dev/null @@ -1,87 +0,0 @@ -part of '../spotify.dart'; - -class AlbumReleasesState extends PaginatedState { - AlbumReleasesState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - AlbumReleasesState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return AlbumReleasesState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class AlbumReleasesNotifier - extends PaginatedAsyncNotifier { - AlbumReleasesNotifier() : super(); - - @override - fetch(int offset, int limit) async { - final market = ref.read(userPreferencesProvider).market; - - final albums = await spotify.browse - .newReleases(country: market) - .getPage(limit, offset); - - return albums.items?.map((album) => album.toAlbum()).toList() ?? []; - } - - @override - build() async { - ref.watch(spotifyProvider); - ref.watch( - userPreferencesProvider.select((s) => s.market), - ); - ref.watch(allFollowedArtistsProvider); - - final albums = await fetch(0, 20); - - return AlbumReleasesState( - items: albums, - offset: 0, - limit: 20, - hasMore: albums.length == 20, - ); - } -} - -final albumReleasesProvider = - AsyncNotifierProvider( - () => AlbumReleasesNotifier(), -); - -final userArtistAlbumReleasesProvider = Provider>((ref) { - final newReleases = ref.watch(albumReleasesProvider); - final userArtistsQuery = ref.watch(allFollowedArtistsProvider); - - if (newReleases.isLoading || userArtistsQuery.isLoading) { - return const []; - } - - final userArtists = - userArtistsQuery.asData?.value.map((s) => s.id!).toList() ?? const []; - - final allReleases = newReleases.asData?.value.items; - final userArtistReleases = allReleases?.where((album) { - return album.artists?.any((artist) => userArtists.contains(artist.id!)) == - true; - }).toList(); - - if (userArtistReleases?.isEmpty == true) { - return allReleases?.toList() ?? []; - } - return userArtistReleases ?? []; -}); diff --git a/lib/provider/spotify/album/tracks.dart b/lib/provider/spotify/album/tracks.dart deleted file mode 100644 index e39abad5..00000000 --- a/lib/provider/spotify/album/tracks.dart +++ /dev/null @@ -1,61 +0,0 @@ -part of '../spotify.dart'; - -class AlbumTracksState extends PaginatedState { - AlbumTracksState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - AlbumTracksState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return AlbumTracksState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class AlbumTracksNotifier extends AutoDisposeFamilyPaginatedAsyncNotifier { - AlbumTracksNotifier() : super(); - - @override - fetch(arg, offset, limit) async { - final tracks = await spotify.albums.tracks(arg.id!).getPage(limit, offset); - final items = tracks.items?.map((e) => e.asTrack(arg)).toList() ?? []; - - return ( - items: items, - hasMore: !tracks.isLast, - nextOffset: tracks.nextOffset, - ); - } - - @override - build(arg) async { - ref.cacheFor(); - - ref.watch(spotifyProvider); - final (:items, :nextOffset, :hasMore) = await fetch(arg, 0, 20); - return AlbumTracksState( - items: items, - offset: nextOffset, - limit: 20, - hasMore: hasMore, - ); - } -} - -final albumTracksProvider = AutoDisposeAsyncNotifierProviderFamily< - AlbumTracksNotifier, AlbumTracksState, AlbumSimple>( - () => AlbumTracksNotifier(), -); diff --git a/lib/provider/spotify/artist/albums.dart b/lib/provider/spotify/artist/albums.dart deleted file mode 100644 index f3fb682f..00000000 --- a/lib/provider/spotify/artist/albums.dart +++ /dev/null @@ -1,68 +0,0 @@ -part of '../spotify.dart'; - -class ArtistAlbumsState extends PaginatedState { - ArtistAlbumsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - ArtistAlbumsState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return ArtistAlbumsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class ArtistAlbumsNotifier extends AutoDisposeFamilyPaginatedAsyncNotifier< - Album, ArtistAlbumsState, String> { - ArtistAlbumsNotifier() : super(); - - @override - fetch(arg, offset, limit) async { - final market = ref.read(userPreferencesProvider).market; - final albums = await spotify.artists - .albums(arg, country: market) - .getPage(limit, offset); - - final items = albums.items?.toList() ?? []; - - return ( - items: items, - hasMore: !albums.isLast, - nextOffset: albums.nextOffset, - ); - } - - @override - build(arg) async { - ref.cacheFor(); - - ref.watch(spotifyProvider); - ref.watch( - userPreferencesProvider.select((s) => s.market), - ); - final (:items, :hasMore, :nextOffset) = await fetch(arg, 0, 20); - return ArtistAlbumsState( - items: items, - offset: nextOffset, - limit: 20, - hasMore: hasMore, - ); - } -} - -final artistAlbumsProvider = AutoDisposeAsyncNotifierProviderFamily< - ArtistAlbumsNotifier, ArtistAlbumsState, String>( - () => ArtistAlbumsNotifier(), -); diff --git a/lib/provider/spotify/artist/artist.dart b/lib/provider/spotify/artist/artist.dart deleted file mode 100644 index c69badd2..00000000 --- a/lib/provider/spotify/artist/artist.dart +++ /dev/null @@ -1,10 +0,0 @@ -part of '../spotify.dart'; - -final artistProvider = - FutureProvider.autoDispose.family((ref, String artistId) { - ref.cacheFor(); - - final spotify = ref.watch(spotifyProvider); - - return spotify.artists.get(artistId); -}); diff --git a/lib/provider/spotify/artist/following.dart b/lib/provider/spotify/artist/following.dart deleted file mode 100644 index 4e6bcfe8..00000000 --- a/lib/provider/spotify/artist/following.dart +++ /dev/null @@ -1,104 +0,0 @@ -part of '../spotify.dart'; - -class FollowedArtistsState extends CursorPaginatedState { - FollowedArtistsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - FollowedArtistsState copyWith({ - List? items, - String? offset, - int? limit, - bool? hasMore, - }) { - return FollowedArtistsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class FollowedArtistsNotifier - extends CursorPaginatedAsyncNotifier { - FollowedArtistsNotifier() : super(); - - @override - fetch(offset, limit) async { - final artists = await spotify.me.following(FollowingType.artist).getPage( - limit, - offset ?? '', - ); - - return (artists.items?.toList() ?? [], artists.after); - } - - @override - build() async { - ref.watch(spotifyProvider); - final (artists, nextCursor) = await fetch(null, 50); - return FollowedArtistsState( - items: artists, - offset: nextCursor, - limit: 50, - hasMore: artists.length == 50, - ); - } - - Future saveArtists(List artistIds) async { - if (state.value == null) return; - await spotify.me.follow(FollowingType.artist, artistIds); - - state = await AsyncValue.guard(() async { - final artists = await spotify.artists.list(artistIds); - - return state.value!.copyWith( - items: [ - ...state.value!.items, - ...artists, - ], - ); - }); - - for (final id in artistIds) { - ref.invalidate(artistIsFollowingProvider(id)); - } - } - - Future removeArtists(List artistIds) async { - if (state.value == null) return; - await spotify.me.unfollow(FollowingType.artist, artistIds); - - state = await AsyncValue.guard(() async { - final artists = state.value!.items.where((artist) { - return !artistIds.contains(artist.id); - }).toList(); - - return state.value!.copyWith( - items: artists, - ); - }); - - for (final id in artistIds) { - ref.invalidate(artistIsFollowingProvider(id)); - } - } -} - -final followedArtistsProvider = - AsyncNotifierProvider( - () => FollowedArtistsNotifier(), -); - -final allFollowedArtistsProvider = FutureProvider>( - (ref) async { - final spotify = ref.watch(spotifyProvider); - final artists = await spotify.me.following(FollowingType.artist).all(); - return artists.toList(); - }, -); diff --git a/lib/provider/spotify/artist/is_following.dart b/lib/provider/spotify/artist/is_following.dart deleted file mode 100644 index db1be184..00000000 --- a/lib/provider/spotify/artist/is_following.dart +++ /dev/null @@ -1,10 +0,0 @@ -part of '../spotify.dart'; - -final artistIsFollowingProvider = FutureProvider.family( - (ref, String artistId) async { - final spotify = ref.watch(spotifyProvider); - return spotify.me.checkFollowing(FollowingType.artist, [artistId]).then( - (value) => value[artistId] ?? false, - ); - }, -); diff --git a/lib/provider/spotify/artist/related.dart b/lib/provider/spotify/artist/related.dart deleted file mode 100644 index 317feba3..00000000 --- a/lib/provider/spotify/artist/related.dart +++ /dev/null @@ -1,11 +0,0 @@ -part of '../spotify.dart'; - -final relatedArtistsProvider = FutureProvider.autoDispose - .family, String>((ref, artistId) async { - ref.cacheFor(); - - final spotify = ref.watch(spotifyProvider); - final artists = await spotify.artists.relatedArtists(artistId); - - return artists.toList(); -}); diff --git a/lib/provider/spotify/artist/top_tracks.dart b/lib/provider/spotify/artist/top_tracks.dart deleted file mode 100644 index a2862c3d..00000000 --- a/lib/provider/spotify/artist/top_tracks.dart +++ /dev/null @@ -1,14 +0,0 @@ -part of '../spotify.dart'; - -final artistTopTracksProvider = - FutureProvider.autoDispose.family, String>( - (ref, artistId) async { - ref.cacheFor(); - - final spotify = ref.watch(spotifyProvider); - final market = ref.watch(userPreferencesProvider.select((s) => s.market)); - final tracks = await spotify.artists.topTracks(artistId, market); - - return tracks.toList(); - }, -); diff --git a/lib/provider/spotify/artist/wikipedia.dart b/lib/provider/spotify/artist/wikipedia.dart deleted file mode 100644 index b2e2e6dc..00000000 --- a/lib/provider/spotify/artist/wikipedia.dart +++ /dev/null @@ -1,12 +0,0 @@ -part of '../spotify.dart'; - -final artistWikipediaSummaryProvider = FutureProvider.autoDispose - .family((ref, artist) async { - final query = artist.name!.replaceAll(" ", "_"); - final res = await wikipedia.pageContent.pageSummaryTitleGet(query); - - if (res?.type != "standard") { - return await wikipedia.pageContent.pageSummaryTitleGet("${query}_(singer)"); - } - return res; -}); diff --git a/lib/provider/spotify/category/categories.dart b/lib/provider/spotify/category/categories.dart deleted file mode 100644 index 6237b64c..00000000 --- a/lib/provider/spotify/category/categories.dart +++ /dev/null @@ -1,19 +0,0 @@ -part of '../spotify.dart'; - -final categoriesProvider = FutureProvider( - (ref) async { - final spotify = ref.watch(spotifyProvider); - final market = ref.watch(userPreferencesProvider.select((s) => s.market)); - final locale = ref.watch(userPreferencesProvider.select((s) => s.locale)); - final categories = await spotify.categories - .list( - country: market, - locale: Intl.canonicalizedLocale( - locale.toString(), - ), - ) - .all(); - - return categories.toList()..shuffle(); - }, -); diff --git a/lib/provider/spotify/category/genres.dart b/lib/provider/spotify/category/genres.dart deleted file mode 100644 index b4b75b7b..00000000 --- a/lib/provider/spotify/category/genres.dart +++ /dev/null @@ -1,6 +0,0 @@ -part of '../spotify.dart'; - -final categoryGenresProvider = FutureProvider>((ref) async { - final customSpotify = ref.watch(customSpotifyEndpointProvider); - return await customSpotify.listGenreSeeds(); -}); diff --git a/lib/provider/spotify/category/playlists.dart b/lib/provider/spotify/category/playlists.dart deleted file mode 100644 index 9f1034be..00000000 --- a/lib/provider/spotify/category/playlists.dart +++ /dev/null @@ -1,73 +0,0 @@ -part of '../spotify.dart'; - -class CategoryPlaylistsState extends PaginatedState { - CategoryPlaylistsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - CategoryPlaylistsState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return CategoryPlaylistsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class CategoryPlaylistsNotifier extends AutoDisposeFamilyPaginatedAsyncNotifier< - PlaylistSimple, CategoryPlaylistsState, String> { - CategoryPlaylistsNotifier() : super(); - - @override - fetch(arg, offset, limit) async { - final preferences = ref.read(userPreferencesProvider); - final playlists = await Pages( - spotify, - "v1/browse/categories/$arg/playlists?country=${preferences.market.name}&locale=${preferences.locale}", - (json) => json == null ? null : PlaylistSimple.fromJson(json), - 'playlists', - (json) => PlaylistsFeatured.fromJson(json), - ).getPage(limit, offset); - - final items = playlists.items?.whereNotNull().toList() ?? []; - - return ( - items: items, - hasMore: !playlists.isLast, - nextOffset: playlists.nextOffset, - ); - } - - @override - build(arg) async { - ref.cacheFor(); - - ref.watch(spotifyProvider); - ref.watch(userPreferencesProvider.select((s) => s.locale)); - ref.watch(userPreferencesProvider.select((s) => s.market)); - - final (:items, :hasMore, :nextOffset) = await fetch(arg, 0, 8); - - return CategoryPlaylistsState( - items: items, - offset: nextOffset, - limit: 8, - hasMore: hasMore, - ); - } -} - -final categoryPlaylistsProvider = AutoDisposeAsyncNotifierProviderFamily< - CategoryPlaylistsNotifier, CategoryPlaylistsState, String>( - () => CategoryPlaylistsNotifier(), -); diff --git a/lib/provider/spotify/playlist/favorite.dart b/lib/provider/spotify/playlist/favorite.dart deleted file mode 100644 index 000001ad..00000000 --- a/lib/provider/spotify/playlist/favorite.dart +++ /dev/null @@ -1,136 +0,0 @@ -part of '../spotify.dart'; - -class FavoritePlaylistsState extends PaginatedState { - FavoritePlaylistsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - FavoritePlaylistsState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return FavoritePlaylistsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class FavoritePlaylistsNotifier - extends PaginatedAsyncNotifier { - FavoritePlaylistsNotifier() : super(); - - @override - fetch(int offset, int limit) async { - final playlists = await spotify.playlists.me.getPage( - limit, - offset, - ); - - return playlists.items?.toList() ?? []; - } - - @override - build() async { - ref.watch(spotifyProvider); - final playlists = await fetch(0, 20); - - return FavoritePlaylistsState( - items: playlists, - offset: 0, - limit: 20, - hasMore: playlists.length == 20, - ); - } - - void updatePlaylist(PlaylistSimple playlist) { - if (state.value == null) return; - - if (state.value!.items.none((e) => e.id == playlist.id)) return; - - state = AsyncData( - state.value!.copyWith( - items: state.value!.items - .map((element) => element.id == playlist.id ? playlist : element) - .toList(), - ), - ); - } - - Future addFavorite(PlaylistSimple playlist) async { - await update((state) async { - await spotify.playlists.followPlaylist(playlist.id!); - return state.copyWith( - items: [...state.items, playlist], - ); - }); - - ref.invalidate(isFavoritePlaylistProvider(playlist.id!)); - } - - Future removeFavorite(PlaylistSimple playlist) async { - await update((state) async { - await spotify.playlists.unfollowPlaylist(playlist.id!); - return state.copyWith( - items: state.items.where((e) => e.id != playlist.id).toList(), - ); - }); - - ref.invalidate(isFavoritePlaylistProvider(playlist.id!)); - } - - Future addTracks(String playlistId, List trackIds) async { - if (state.value == null) return; - - final spotify = ref.read(spotifyProvider); - - await spotify.playlists.addTracks( - trackIds.map((id) => 'spotify:track:$id').toList(), - playlistId, - ); - - ref.invalidate(playlistTracksProvider(playlistId)); - } - - Future removeTracks(String playlistId, List trackIds) async { - if (state.value == null) return; - - final spotify = ref.read(spotifyProvider); - - await spotify.playlists.removeTracks( - trackIds.map((id) => 'spotify:track:$id').toList(), - playlistId, - ); - - ref.invalidate(playlistTracksProvider(playlistId)); - } -} - -final favoritePlaylistsProvider = - AsyncNotifierProvider( - () => FavoritePlaylistsNotifier(), -); - -final isFavoritePlaylistProvider = FutureProvider.family( - (ref, id) async { - final spotify = ref.watch(spotifyProvider); - final me = ref.watch(meProvider); - - if (me.value == null) { - return false; - } - - final follows = - await spotify.playlists.followedByUsers(id, [me.value!.id!]); - - return follows[me.value!.id!] ?? false; - }, -); diff --git a/lib/provider/spotify/playlist/featured.dart b/lib/provider/spotify/playlist/featured.dart deleted file mode 100644 index 69057e5d..00000000 --- a/lib/provider/spotify/playlist/featured.dart +++ /dev/null @@ -1,58 +0,0 @@ -part of '../spotify.dart'; - -class FeaturedPlaylistsState extends PaginatedState { - FeaturedPlaylistsState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - FeaturedPlaylistsState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return FeaturedPlaylistsState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class FeaturedPlaylistsNotifier - extends PaginatedAsyncNotifier { - FeaturedPlaylistsNotifier() : super(); - - @override - fetch(int offset, int limit) async { - final playlists = await spotify.playlists.featured.getPage( - limit, - offset, - ); - - return playlists.items?.toList() ?? []; - } - - @override - build() async { - ref.watch(spotifyProvider); - final playlists = await fetch(0, 20); - - return FeaturedPlaylistsState( - items: playlists, - offset: 0, - limit: 20, - hasMore: playlists.length == 20, - ); - } -} - -final featuredPlaylistsProvider = - AsyncNotifierProvider( - () => FeaturedPlaylistsNotifier(), -); diff --git a/lib/provider/spotify/playlist/generate.dart b/lib/provider/spotify/playlist/generate.dart deleted file mode 100644 index 0832003e..00000000 --- a/lib/provider/spotify/playlist/generate.dart +++ /dev/null @@ -1,40 +0,0 @@ -part of '../spotify.dart'; - -final generatePlaylistProvider = FutureProvider.autoDispose - .family, GeneratePlaylistProviderInput>( - (ref, input) async { - final spotify = ref.watch(spotifyProvider); - final market = ref.watch( - userPreferencesProvider.select((s) => s.market), - ); - - final recommendation = await spotify.recommendations - .get( - limit: input.limit, - seedArtists: input.seedArtists?.toList(), - seedGenres: input.seedGenres?.toList(), - seedTracks: input.seedTracks?.toList(), - market: market, - max: (input.max?.toJson()?..removeWhere((key, value) => value == null)) - ?.cast(), - min: (input.min?.toJson()?..removeWhere((key, value) => value == null)) - ?.cast(), - target: (input.target?.toJson() - ?..removeWhere((key, value) => value == null)) - ?.cast(), - ) - .catchError((e, stackTrace) { - AppLogger.reportError(e, stackTrace); - return Recommendations(); - }); - - if (recommendation.tracks?.isEmpty ?? true) { - return []; - } - - final tracks = await spotify.tracks - .list(recommendation.tracks!.map((e) => e.id!).toList()); - - return tracks.toList(); - }, -); diff --git a/lib/provider/spotify/playlist/liked.dart b/lib/provider/spotify/playlist/liked.dart deleted file mode 100644 index 27c3e2b6..00000000 --- a/lib/provider/spotify/playlist/liked.dart +++ /dev/null @@ -1,33 +0,0 @@ -part of '../spotify.dart'; - -class LikedTracksNotifier extends AsyncNotifier> { - @override - FutureOr> build() async { - final spotify = ref.watch(spotifyProvider); - final savedTracked = await spotify.tracks.me.saved.all(); - - return savedTracked.map((e) => e.track!).toList(); - } - - Future toggleFavorite(Track track) async { - if (state.value == null) return; - final spotify = ref.read(spotifyProvider); - - await update((tracks) async { - final isLiked = tracks.map((e) => e.id).contains(track.id); - - if (isLiked) { - await spotify.tracks.me.removeOne(track.id!); - return tracks.where((e) => e.id != track.id).toList(); - } else { - await spotify.tracks.me.saveOne(track.id!); - return [track, ...tracks]; - } - }); - } -} - -final likedTracksProvider = - AsyncNotifierProvider>( - () => LikedTracksNotifier(), -); diff --git a/lib/provider/spotify/playlist/playlist.dart b/lib/provider/spotify/playlist/playlist.dart deleted file mode 100644 index 0eec3a87..00000000 --- a/lib/provider/spotify/playlist/playlist.dart +++ /dev/null @@ -1,106 +0,0 @@ -part of '../spotify.dart'; - -typedef PlaylistInput = ({ - String playlistName, - bool? public, - bool? collaborative, - String? description, - String? base64Image, -}); - -class PlaylistNotifier extends FamilyAsyncNotifier { - @override - FutureOr build(String arg) { - final spotify = ref.watch(spotifyProvider); - return spotify.playlists.get(arg); - } - - Future create(PlaylistInput input, [ValueChanged? onError]) async { - if (state is AsyncLoading) return; - state = const AsyncLoading(); - - final spotify = ref.read(spotifyProvider); - final me = ref.read(meProvider); - - if (me.value == null) return; - - state = await AsyncValue.guard(() async { - try { - final playlist = await spotify.playlists.createPlaylist( - me.value!.id!, - input.playlistName, - collaborative: input.collaborative, - description: input.description, - public: input.public, - ); - - if (input.base64Image != null) { - await spotify.playlists.updatePlaylistImage( - playlist.id!, - input.base64Image!, - ); - } - - return playlist; - } catch (e) { - onError?.call(e); - rethrow; - } - }); - - ref.invalidate(favoritePlaylistsProvider); - } - - Future modify(PlaylistInput input, [ValueChanged? onError]) async { - if (state.value == null) return; - - final spotify = ref.read(spotifyProvider); - - await update((state) async { - try { - await spotify.playlists.updatePlaylist( - state.id!, - input.playlistName, - collaborative: input.collaborative, - description: input.description, - public: input.public, - ); - - if (input.base64Image != null) { - await spotify.playlists.updatePlaylistImage( - state.id!, - input.base64Image!, - ); - - final playlist = await spotify.playlists.get(state.id!); - - ref.read(favoritePlaylistsProvider.notifier).updatePlaylist(playlist); - return playlist; - } - - final playlist = Playlist.fromJson( - { - ...state.toJson(), - 'name': input.playlistName, - 'collaborative': input.collaborative, - 'description': input.description, - 'public': input.public, - }, - ); - - ref.read(favoritePlaylistsProvider.notifier).updatePlaylist(playlist); - - return playlist; - } catch (e, stack) { - onError?.call(e); - AppLogger.reportError(e, stack); - rethrow; - } - }); - } -} - -final playlistProvider = - AsyncNotifierProvider.family( - () => PlaylistNotifier(), -); diff --git a/lib/provider/spotify/playlist/tracks.dart b/lib/provider/spotify/playlist/tracks.dart deleted file mode 100644 index 379ad110..00000000 --- a/lib/provider/spotify/playlist/tracks.dart +++ /dev/null @@ -1,70 +0,0 @@ -part of '../spotify.dart'; - -class PlaylistTracksState extends PaginatedState { - PlaylistTracksState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - PlaylistTracksState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return PlaylistTracksState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class PlaylistTracksNotifier extends AutoDisposeFamilyPaginatedAsyncNotifier< - Track, PlaylistTracksState, String> { - PlaylistTracksNotifier() : super(); - - @override - fetch(arg, offset, limit) async { - final tracks = await spotify.playlists - .getTracksByPlaylistId(arg) - .getPage(limit, offset); - - /// Filter out tracks with null id because some personal playlists - /// may contain local tracks that are not available in the Spotify catalog - final items = tracks.items - ?.where((track) => track.id != null && track.type == "track") - .toList() ?? - []; - - return ( - items: items, - hasMore: !tracks.isLast, - nextOffset: tracks.nextOffset, - ); - } - - @override - build(arg) async { - ref.cacheFor(); - - ref.watch(spotifyProvider); - final (items: tracks, :hasMore, :nextOffset) = await fetch(arg, 0, 20); - - return PlaylistTracksState( - items: tracks, - offset: nextOffset, - limit: 20, - hasMore: hasMore, - ); - } -} - -final playlistTracksProvider = AutoDisposeAsyncNotifierProviderFamily< - PlaylistTracksNotifier, PlaylistTracksState, String>( - () => PlaylistTracksNotifier(), -); diff --git a/lib/provider/spotify/search/search.dart b/lib/provider/spotify/search/search.dart deleted file mode 100644 index 5bbc02e4..00000000 --- a/lib/provider/spotify/search/search.dart +++ /dev/null @@ -1,88 +0,0 @@ -part of '../spotify.dart'; - -final searchTermStateProvider = StateProvider.autoDispose( - (ref) { - ref.cacheFor(const Duration(minutes: 2)); - return ""; - }, -); - -class SearchState extends PaginatedState { - SearchState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - SearchState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }) { - return SearchState( - items: items ?? this.items, - offset: offset ?? this.offset, - limit: limit ?? this.limit, - hasMore: hasMore ?? this.hasMore, - ); - } -} - -class SearchNotifier extends AutoDisposeFamilyPaginatedAsyncNotifier, SearchType> { - SearchNotifier() : super(); - - @override - fetch(arg, offset, limit) async { - if (state.value == null) { - return ( - items: [], - hasMore: false, - nextOffset: 0, - ); - } - final results = await spotify.search - .get( - ref.read(searchTermStateProvider), - types: [arg], - market: ref.read(userPreferencesProvider).market, - ) - .getPage(limit, offset); - - final items = results.expand((e) => e.items ?? []).toList().cast(); - - return ( - items: items, - hasMore: items.length == limit, - nextOffset: offset + limit, - ); - } - - @override - build(arg) async { - ref.cacheFor(const Duration(minutes: 2)); - - ref.watch(searchTermStateProvider); - ref.watch(spotifyProvider); - ref.watch( - userPreferencesProvider.select((value) => value.market), - ); - - final (:items, :hasMore, :nextOffset) = await fetch(arg, 0, 10); - - return SearchState( - items: items, - offset: nextOffset, - limit: 10, - hasMore: hasMore, - ); - } -} - -final searchProvider = AsyncNotifierProvider.autoDispose - .family( - () => SearchNotifier(), -); diff --git a/lib/provider/spotify/spotify.dart b/lib/provider/spotify/spotify.dart deleted file mode 100644 index 8cf60120..00000000 --- a/lib/provider/spotify/spotify.dart +++ /dev/null @@ -1,79 +0,0 @@ -library spotify; - -import 'dart:async'; - -import 'package:drift/drift.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/provider/spotify/utils/json_cast.dart'; -import 'package:spotube/services/logger/logger.dart'; -import 'package:collection/collection.dart'; -import 'package:dio/dio.dart'; -import 'package:flutter/material.dart'; -import 'package:hive_flutter/hive_flutter.dart'; -import 'package:intl/intl.dart'; -import 'package:lrc/lrc.dart'; -import 'package:package_info_plus/package_info_plus.dart'; -import 'package:spotify/spotify.dart'; -import 'package:hooks_riverpod/hooks_riverpod.dart'; -// ignore: depend_on_referenced_packages, implementation_imports -import 'package:riverpod/src/async_notifier.dart'; -import 'package:spotube/extensions/album_simple.dart'; -import 'package:spotube/extensions/track.dart'; -import 'package:spotube/models/lyrics.dart'; -import 'package:spotube/models/spotify/recommendation_seeds.dart'; -import 'package:spotube/models/spotify_friends.dart'; -import 'package:spotube/provider/custom_spotify_endpoint_provider.dart'; -import 'package:spotube/provider/spotify_provider.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:spotube/services/dio/dio.dart'; -import 'package:spotube/services/wikipedia/wikipedia.dart'; - -import 'package:wikipedia_api/wikipedia_api.dart'; - -part 'album/favorite.dart'; -part 'album/tracks.dart'; -part 'album/releases.dart'; -part 'album/is_saved.dart'; - -part 'artist/artist.dart'; -part 'artist/is_following.dart'; -part 'artist/following.dart'; -part 'artist/top_tracks.dart'; -part 'artist/albums.dart'; -part 'artist/wikipedia.dart'; -part 'artist/related.dart'; - -part 'category/genres.dart'; -part 'category/categories.dart'; -part 'category/playlists.dart'; - -part 'lyrics/synced.dart'; - -part 'playlist/favorite.dart'; -part 'playlist/playlist.dart'; -part 'playlist/liked.dart'; -part 'playlist/tracks.dart'; -part 'playlist/featured.dart'; -part 'playlist/generate.dart'; - -part 'search/search.dart'; - -part 'user/me.dart'; -part 'user/friends.dart'; - -part 'tracks/track.dart'; - -part 'views/view.dart'; - -part 'utils/mixin.dart'; -part 'utils/state.dart'; -part 'utils/provider.dart'; -part 'utils/persistence.dart'; -part 'utils/async.dart'; - -part 'utils/provider/paginated.dart'; -part 'utils/provider/cursor.dart'; -part 'utils/provider/paginated_family.dart'; -part 'utils/provider/cursor_family.dart'; diff --git a/lib/provider/spotify/tracks/track.dart b/lib/provider/spotify/tracks/track.dart deleted file mode 100644 index e3913b1f..00000000 --- a/lib/provider/spotify/tracks/track.dart +++ /dev/null @@ -1,10 +0,0 @@ -part of '../spotify.dart'; - -final trackProvider = - FutureProvider.autoDispose.family((ref, id) async { - ref.cacheFor(); - - final spotify = ref.watch(spotifyProvider); - - return spotify.tracks.get(id); -}); diff --git a/lib/provider/spotify/user/friends.dart b/lib/provider/spotify/user/friends.dart deleted file mode 100644 index b9cc0f46..00000000 --- a/lib/provider/spotify/user/friends.dart +++ /dev/null @@ -1,7 +0,0 @@ -part of '../spotify.dart'; - -final friendsProvider = FutureProvider((ref) async { - final customSpotify = ref.watch(customSpotifyEndpointProvider); - - return customSpotify.getFriendActivity(); -}); diff --git a/lib/provider/spotify/user/me.dart b/lib/provider/spotify/user/me.dart deleted file mode 100644 index c5949e1f..00000000 --- a/lib/provider/spotify/user/me.dart +++ /dev/null @@ -1,6 +0,0 @@ -part of '../spotify.dart'; - -final meProvider = FutureProvider((ref) async { - final spotify = ref.watch(spotifyProvider); - return spotify.me.get(); -}); diff --git a/lib/provider/spotify/utils/async.dart b/lib/provider/spotify/utils/async.dart deleted file mode 100644 index 1040d682..00000000 --- a/lib/provider/spotify/utils/async.dart +++ /dev/null @@ -1,5 +0,0 @@ -part of '../spotify.dart'; - -extension PaginationExtension on AsyncValue { - bool get isLoadingNextPage => this is AsyncData && this is AsyncLoadingNext; -} diff --git a/lib/provider/spotify/utils/json_cast.dart b/lib/provider/spotify/utils/json_cast.dart deleted file mode 100644 index 30700971..00000000 --- a/lib/provider/spotify/utils/json_cast.dart +++ /dev/null @@ -1,21 +0,0 @@ -Map castNestedJson(Map map) { - return Map.castFrom( - map.map((key, value) { - if (value is Map) { - return MapEntry( - key, - castNestedJson(value), - ); - } else if (value is Iterable) { - return MapEntry( - key, - value.map((e) { - if (e is Map) return castNestedJson(e); - return e; - }).toList(), - ); - } - return MapEntry(key, value); - }), - ); -} diff --git a/lib/provider/spotify/utils/mixin.dart b/lib/provider/spotify/utils/mixin.dart deleted file mode 100644 index 0da14c6f..00000000 --- a/lib/provider/spotify/utils/mixin.dart +++ /dev/null @@ -1,24 +0,0 @@ -part of '../spotify.dart'; - -// ignore: invalid_use_of_internal_member -mixin SpotifyMixin on AsyncNotifierBase { - SpotifyApi get spotify => ref.read(spotifyProvider); -} - -extension on AutoDisposeAsyncNotifierProviderRef { - // When invoked keeps your provider alive for [duration] - void cacheFor([Duration duration = const Duration(minutes: 5)]) { - final link = keepAlive(); - final timer = Timer(duration, () => link.close()); - onDispose(() => timer.cancel()); - } -} - -extension on AutoDisposeRef { - // When invoked keeps your provider alive for [duration] - void cacheFor([Duration duration = const Duration(minutes: 5)]) { - final link = keepAlive(); - final timer = Timer(duration, () => link.close()); - onDispose(() => timer.cancel()); - } -} diff --git a/lib/provider/spotify/utils/persistence.dart b/lib/provider/spotify/utils/persistence.dart deleted file mode 100644 index 57f41dec..00000000 --- a/lib/provider/spotify/utils/persistence.dart +++ /dev/null @@ -1,40 +0,0 @@ -part of '../spotify.dart'; - -// ignore: invalid_use_of_internal_member -mixin Persistence on BuildlessAsyncNotifier { - LazyBox get store => Hive.lazyBox("spotube_cache"); - - FutureOr fromJson(Map json); - Map toJson(T data); - - FutureOr onInit() {} - - Future load() async { - final json = await store.get(runtimeType.toString()); - if (json != null || - (json is Map && json.entries.isNotEmpty) || - (json is List && json.isNotEmpty)) { - state = AsyncData( - await fromJson( - castNestedJson(json), - ), - ); - } - - await onInit(); - } - - Future save() async { - await store.put( - runtimeType.toString(), - state.value == null ? null : toJson(state.value as T), - ); - } - - @override - set state(AsyncValue value) { - if (state == value) return; - super.state = value; - save(); - } -} diff --git a/lib/provider/spotify/utils/provider.dart b/lib/provider/spotify/utils/provider.dart deleted file mode 100644 index 50458c3a..00000000 --- a/lib/provider/spotify/utils/provider.dart +++ /dev/null @@ -1,6 +0,0 @@ -part of '../spotify.dart'; - -// ignore: subtype_of_sealed_class -class AsyncLoadingNext extends AsyncData { - const AsyncLoadingNext(super.value); -} diff --git a/lib/provider/spotify/utils/provider/cursor.dart b/lib/provider/spotify/utils/provider/cursor.dart deleted file mode 100644 index c241827e..00000000 --- a/lib/provider/spotify/utils/provider/cursor.dart +++ /dev/null @@ -1,56 +0,0 @@ -part of '../../spotify.dart'; - -mixin CursorPaginatedAsyncNotifierMixin> - // ignore: invalid_use_of_internal_member - on AsyncNotifierBase { - Future<(List items, String nextCursor)> fetch(String? offset, int limit); - - Future fetchMore() async { - if (state.value == null || !state.value!.hasMore) return; - - state = AsyncLoadingNext(state.asData!.value); - - state = await AsyncValue.guard( - () async { - final items = await fetch(state.value!.offset, state.value!.limit); - return state.value!.copyWith( - hasMore: items.$1.length == state.value!.limit, - items: [ - ...state.value!.items, - ...items.$1, - ], - offset: items.$2, - ) as T; - }, - ); - } - - Future> fetchAll() async { - if (state.value == null) return []; - if (!state.value!.hasMore) return state.value!.items; - - bool hasMore = true; - while (hasMore) { - await update((state) async { - final items = await fetch(state.offset, state.limit); - - hasMore = items.$1.length == state.limit; - return state.copyWith( - items: [...state.items, ...items.$1], - offset: items.$2, - hasMore: hasMore, - ) as T; - }); - } - - return state.value!.items; - } -} - -abstract class CursorPaginatedAsyncNotifier> extends AsyncNotifier - with CursorPaginatedAsyncNotifierMixin, SpotifyMixin {} - -abstract class AutoDisposeCursorPaginatedAsyncNotifier> extends AutoDisposeAsyncNotifier - with CursorPaginatedAsyncNotifierMixin, SpotifyMixin {} diff --git a/lib/provider/spotify/utils/provider/cursor_family.dart b/lib/provider/spotify/utils/provider/cursor_family.dart deleted file mode 100644 index ea8577de..00000000 --- a/lib/provider/spotify/utils/provider/cursor_family.dart +++ /dev/null @@ -1,113 +0,0 @@ -part of '../../spotify.dart'; - -abstract class FamilyCursorPaginatedAsyncNotifier< - K, - T extends CursorPaginatedState, - A> extends FamilyAsyncNotifier with SpotifyMixin { - Future<(List items, String nextCursor)> fetch( - A arg, - String? offset, - int limit, - ); - - Future fetchMore() async { - if (state.value == null || !state.value!.hasMore) return; - - state = AsyncLoadingNext(state.asData!.value); - - state = await AsyncValue.guard( - () async { - final items = await fetch(arg, state.value!.offset, state.value!.limit); - return state.value!.copyWith( - hasMore: items.$1.length == state.value!.limit, - items: [ - ...state.value!.items, - ...items.$1, - ], - offset: items.$2, - ) as T; - }, - ); - } - - Future> fetchAll() async { - if (state.value == null) return []; - if (!state.value!.hasMore) return state.value!.items; - - bool hasMore = true; - while (hasMore) { - await update((state) async { - final items = await fetch( - arg, - state.offset, - state.limit, - ); - - hasMore = items.$1.length == state.limit; - return state.copyWith( - items: [...state.items, ...items.$1], - offset: items.$2, - hasMore: hasMore, - ) as T; - }); - } - - return state.value!.items; - } -} - -abstract class AutoDisposeFamilyCursorPaginatedAsyncNotifier< - K, - T extends CursorPaginatedState, - A> extends AutoDisposeFamilyAsyncNotifier with SpotifyMixin { - Future<(List items, String nextCursor)> fetch( - A arg, - String? offset, - int limit, - ); - - Future fetchMore() async { - if (state.value == null || !state.value!.hasMore) return; - - state = AsyncLoadingNext(state.asData!.value); - - state = await AsyncValue.guard( - () async { - final items = await fetch(arg, state.value!.offset, state.value!.limit); - return state.value!.copyWith( - hasMore: items.$1.length == state.value!.limit, - items: [ - ...state.value!.items, - ...items.$1, - ], - offset: items.$2, - ) as T; - }, - ); - } - - Future> fetchAll() async { - if (state.value == null) return []; - if (!state.value!.hasMore) return state.value!.items; - - bool hasMore = true; - while (hasMore) { - await update((state) async { - final items = await fetch( - arg, - state.offset, - state.limit, - ); - - hasMore = items.$1.length == state.limit; - return state.copyWith( - items: [...state.items, ...items.$1], - offset: items.$2, - hasMore: hasMore, - ) as T; - }); - } - - return state.value!.items; - } -} diff --git a/lib/provider/spotify/utils/provider/paginated.dart b/lib/provider/spotify/utils/provider/paginated.dart deleted file mode 100644 index 30b66e67..00000000 --- a/lib/provider/spotify/utils/provider/paginated.dart +++ /dev/null @@ -1,63 +0,0 @@ -part of '../../spotify.dart'; - -mixin PaginatedAsyncNotifierMixin> - // ignore: invalid_use_of_internal_member - on AsyncNotifierBase { - Future> fetch(int offset, int limit); - - Future fetchMore() async { - if (state.value == null || !state.value!.hasMore) return; - - state = AsyncLoadingNext(state.asData!.value); - - state = await AsyncValue.guard( - () async { - final items = await fetch( - state.value!.offset + state.value!.limit, - state.value!.limit, - ); - return state.value!.copyWith( - hasMore: items.length == state.value!.limit, - items: [ - ...state.value!.items, - ...items, - ], - offset: state.value!.offset + state.value!.limit, - ) as T; - }, - ); - } - - Future> fetchAll() async { - if (state.value == null) return []; - if (!state.value!.hasMore) return state.value!.items; - - bool hasMore = true; - while (hasMore) { - await update((state) async { - final items = await fetch( - state.offset + state.limit, - state.limit, - ); - - hasMore = items.length == state.limit; - return state.copyWith( - items: [...state.items, ...items], - offset: state.offset + state.limit, - hasMore: hasMore, - ) as T; - }); - } - - return state.value!.items; - } -} - -abstract class PaginatedAsyncNotifier> - extends AsyncNotifier - with PaginatedAsyncNotifierMixin, SpotifyMixin {} - -abstract class AutoDisposePaginatedAsyncNotifier> - extends AutoDisposeAsyncNotifier - with PaginatedAsyncNotifierMixin, SpotifyMixin {} diff --git a/lib/provider/spotify/utils/provider/paginated_family.dart b/lib/provider/spotify/utils/provider/paginated_family.dart deleted file mode 100644 index c08c8673..00000000 --- a/lib/provider/spotify/utils/provider/paginated_family.dart +++ /dev/null @@ -1,120 +0,0 @@ -part of '../../spotify.dart'; - -typedef PseudoPaginatedProps = ({ - List items, - int nextOffset, - bool hasMore, -}); - -abstract class FamilyPaginatedAsyncNotifier< - K, - T extends BasePaginatedState, - A> extends FamilyAsyncNotifier with SpotifyMixin { - Future> fetch(A arg, int offset, int limit); - - Future fetchMore() async { - if (state.value == null || !state.value!.hasMore) return; - - state = AsyncLoadingNext(state.asData!.value); - - state = await AsyncValue.guard( - () async { - final (:items, :hasMore, :nextOffset) = await fetch( - arg, - state.value!.offset, - state.value!.limit, - ); - return state.value!.copyWith( - hasMore: hasMore, - items: [ - ...state.value!.items, - ...items, - ], - offset: nextOffset, - ) as T; - }, - ); - } - - Future> fetchAll() async { - if (state.value == null) return []; - if (!state.value!.hasMore) return state.value!.items; - - bool hasMore = true; - while (hasMore) { - await update((state) async { - final res = await fetch( - arg, - state.offset, - state.limit, - ); - - hasMore = res.hasMore; - return state.copyWith( - items: [...state.items, ...res.items], - offset: res.nextOffset, - hasMore: hasMore, - ) as T; - }); - } - - return state.value!.items; - } -} - -abstract class AutoDisposeFamilyPaginatedAsyncNotifier< - K, - T extends BasePaginatedState, - A> extends AutoDisposeFamilyAsyncNotifier with SpotifyMixin { - Future> fetch(A arg, int offset, int limit); - - Future fetchMore() async { - if (state.value == null || !state.value!.hasMore) return; - - state = AsyncLoadingNext(state.asData!.value); - - state = await AsyncValue.guard( - () async { - final (:items, :hasMore, :nextOffset) = await fetch( - arg, - state.value!.offset, - state.value!.limit, - ); - - return state.value!.copyWith( - hasMore: hasMore, - items: [ - ...state.value!.items, - ...items, - ], - offset: nextOffset, - ) as T; - }, - ); - } - - Future> fetchAll() async { - if (state.value == null) return []; - if (!state.value!.hasMore) return state.value!.items; - - bool hasMore = true; - while (hasMore) { - await update((state) async { - final res = await fetch( - arg, - state.offset, - state.limit, - ); - - hasMore = res.hasMore; - return state.copyWith( - items: [...state.items, ...res.items], - offset: res.nextOffset, - hasMore: hasMore, - ) as T; - }); - } - - return state.value!.items; - } -} diff --git a/lib/provider/spotify/utils/state.dart b/lib/provider/spotify/utils/state.dart deleted file mode 100644 index 4b79ac7d..00000000 --- a/lib/provider/spotify/utils/state.dart +++ /dev/null @@ -1,56 +0,0 @@ -part of '../spotify.dart'; - -abstract class BasePaginatedState { - final List items; - final Cursor offset; - final int limit; - final bool hasMore; - - BasePaginatedState({ - required this.items, - required this.offset, - required this.limit, - required this.hasMore, - }); - - BasePaginatedState copyWith({ - List? items, - Cursor? offset, - int? limit, - bool? hasMore, - }); -} - -abstract class PaginatedState extends BasePaginatedState { - PaginatedState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - PaginatedState copyWith({ - List? items, - int? offset, - int? limit, - bool? hasMore, - }); -} - -abstract class CursorPaginatedState extends BasePaginatedState { - CursorPaginatedState({ - required super.items, - required super.offset, - required super.limit, - required super.hasMore, - }); - - @override - CursorPaginatedState copyWith({ - List? items, - String? offset, - int? limit, - bool? hasMore, - }); -} diff --git a/lib/provider/spotify/views/home.dart b/lib/provider/spotify/views/home.dart deleted file mode 100644 index ad6a076a..00000000 --- a/lib/provider/spotify/views/home.dart +++ /dev/null @@ -1,22 +0,0 @@ -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/custom_spotify_endpoint_provider.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -final homeViewProvider = FutureProvider((ref) async { - final country = ref.watch( - userPreferencesProvider.select((s) => s.market), - ); - final spTCookie = ref.watch( - authenticationProvider.select((s) => s.asData?.value?.getCookie("sp_t")), - ); - - if (spTCookie == null) return null; - - final spotify = ref.watch(customSpotifyEndpointProvider); - - return spotify.getHomeFeed( - country: country, - spTCookie: spTCookie, - ); -}); diff --git a/lib/provider/spotify/views/home_section.dart b/lib/provider/spotify/views/home_section.dart deleted file mode 100644 index 5eb9183d..00000000 --- a/lib/provider/spotify/views/home_section.dart +++ /dev/null @@ -1,26 +0,0 @@ -import 'package:hooks_riverpod/hooks_riverpod.dart'; -import 'package:spotube/models/spotify/home_feed.dart'; -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/provider/custom_spotify_endpoint_provider.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -final homeSectionViewProvider = - FutureProvider.family( - (ref, sectionUri) async { - final country = ref.watch( - userPreferencesProvider.select((s) => s.market), - ); - final spTCookie = ref.watch( - authenticationProvider.select((s) => s.asData?.value?.getCookie("sp_t")), - ); - - if (spTCookie == null) return null; - - final spotify = ref.watch(customSpotifyEndpointProvider); - - return spotify.getHomeFeedSection( - sectionUri, - country: country, - spTCookie: spTCookie, - ); -}); diff --git a/lib/provider/spotify/views/view.dart b/lib/provider/spotify/views/view.dart deleted file mode 100644 index ff565feb..00000000 --- a/lib/provider/spotify/views/view.dart +++ /dev/null @@ -1,19 +0,0 @@ -part of '../spotify.dart'; - -final viewProvider = FutureProvider.family, String>( - (ref, viewName) async { - final customSpotify = ref.watch(customSpotifyEndpointProvider); - final market = ref.watch( - userPreferencesProvider.select((s) => s.market), - ); - final locale = ref.watch( - userPreferencesProvider.select((s) => s.locale), - ); - - return customSpotify.getView( - viewName, - market: market, - locale: Intl.canonicalizedLocale(locale.toString()), - ); - }, -); diff --git a/lib/provider/spotify_provider.dart b/lib/provider/spotify_provider.dart deleted file mode 100644 index 5824cce0..00000000 --- a/lib/provider/spotify_provider.dart +++ /dev/null @@ -1,22 +0,0 @@ -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/collections/env.dart'; - -import 'package:spotube/provider/authentication/authentication.dart'; -import 'package:spotube/utils/primitive_utils.dart'; - -final spotifyProvider = Provider((ref) { - final authState = ref.watch(authenticationProvider); - final anonCred = PrimitiveUtils.getRandomElement(Env.spotifySecrets); - - if (authState.asData?.value == null) { - return SpotifyApi( - SpotifyApiCredentials( - anonCred["clientId"], - anonCred["clientSecret"], - ), - ); - } - - return SpotifyApi.withAccessToken(authState.asData!.value!.accessToken.value); -}); diff --git a/lib/provider/track_options/track_options_provider.dart b/lib/provider/track_options/track_options_provider.dart new file mode 100644 index 00000000..5aebf39c --- /dev/null +++ b/lib/provider/track_options/track_options_provider.dart @@ -0,0 +1,306 @@ +import 'dart:io'; + +import 'package:auto_route/auto_route.dart'; +import 'package:flutter/services.dart'; +import 'package:hooks_riverpod/hooks_riverpod.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:spotube/collections/routes.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/components/dialogs/playlist_add_track_dialog.dart'; +import 'package:spotube/components/dialogs/prompt_dialog.dart'; +import 'package:spotube/components/dialogs/track_details_dialog.dart'; +import 'package:spotube/extensions/context.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/provider/audio_player/audio_player.dart'; +import 'package:spotube/provider/blacklist_provider.dart'; +import 'package:spotube/provider/download_manager_provider.dart'; +import 'package:spotube/provider/local_tracks/local_tracks_provider.dart'; +import 'package:spotube/provider/metadata_plugin/core/auth.dart'; +import 'package:spotube/provider/metadata_plugin/library/playlists.dart'; +import 'package:spotube/provider/metadata_plugin/library/tracks.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; + +enum TrackOptionValue { + album, + share, + addToPlaylist, + addToQueue, + removeFromPlaylist, + removeFromQueue, + blacklist, + delete, + playNext, + favorite, + details, + download, + startRadio, +} + +class TrackOptionsActions { + final Ref ref; + final SpotubeTrackObject track; + + TrackOptionsActions(this.ref, this.track); + + AudioPlayerNotifier get playback => ref.read(audioPlayerProvider.notifier); + MetadataPluginSavedTracksNotifier get favoriteTracks => + ref.read(metadataPluginSavedTracksProvider.notifier); + MetadataPluginSavedPlaylistsNotifier get favoritePlaylistsNotifier => + ref.read(metadataPluginSavedPlaylistsProvider.notifier); + DownloadManagerNotifier get downloadManager => + ref.read(downloadManagerProvider.notifier); + BlackListNotifier get blacklist => ref.read(blacklistProvider.notifier); + + void actionShare(BuildContext context) { + Clipboard.setData(ClipboardData(text: track.externalUri)).then((_) { + if (context.mounted) { + showToast( + context: rootNavigatorKey.currentContext!, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n.copied_to_clipboard(track.externalUri), + textAlign: TextAlign.center, + ), + ); + }, + ); + } + }); + } + + Future actionAddToPlaylist( + BuildContext context, + String? playlistId, + ) async { + /// showDialog doesn't work for some reason. So we have to + /// manually push a Dialog Route in the Navigator to get it working + await showDialog( + context: context, + builder: (context) { + return PlaylistAddTrackDialog( + tracks: [track], + openFromPlaylist: playlistId, + ); + }, + ); + } + + Future actionStartRadio(BuildContext context) async { + final playback = ref.read(audioPlayerProvider.notifier); + final playlist = ref.read(audioPlayerProvider); + final metadataPlugin = await ref.read(metadataPluginProvider.future); + + if (metadataPlugin == null) { + throw MetadataPluginException.noDefaultMetadataPlugin(); + } + + final tracks = await metadataPlugin.track.radio(track.id); + + bool replaceQueue = false; + + if (context.mounted && playlist.tracks.isNotEmpty) { + replaceQueue = await showPromptDialog( + context: context, + title: context.l10n.how_to_start_radio, + message: context.l10n.replace_queue_question, + okText: context.l10n.replace, + cancelText: context.l10n.add_to_queue, + ); + } + + if (replaceQueue || playlist.tracks.isEmpty) { + await playback.stop(); + await playback.load([track], autoPlay: true); + + // we don't have to add those tracks as useEndlessPlayback will do it for us + return; + } else { + await playback.addTrack(track); + } + + await playback.addTracks( + tracks.toList() + ..removeWhere((e) { + final isDuplicate = playlist.tracks.any((t) => t.id == e.id); + return e.id == track.id || isDuplicate; + }), + ); + } + + Future action( + BuildContext context, + TrackOptionValue value, + String? playlistId, + ) async { + switch (value) { + case TrackOptionValue.album: + await context.navigateTo( + AlbumRoute(id: track.album.id, album: track.album), + ); + break; + case TrackOptionValue.delete: + await File((track as SpotubeLocalTrackObject).path).delete(); + ref.invalidate(localTracksProvider); + break; + case TrackOptionValue.addToQueue: + await playback.addTrack(track); + if (context.mounted) { + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n.added_track_to_queue(track.name), + textAlign: TextAlign.center, + ), + ); + }, + ); + } + break; + case TrackOptionValue.playNext: + await playback.addTracksAtFirst([track]); + + if (context.mounted) { + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n.track_will_play_next(track.name), + textAlign: TextAlign.center, + ), + ); + }, + ); + } + break; + case TrackOptionValue.removeFromQueue: + playback.removeTrack(track.id); + + if (context.mounted) { + showToast( + context: context, + location: ToastLocation.topRight, + builder: (context, overlay) { + return SurfaceCard( + child: Text( + context.l10n.removed_track_from_queue( + track.name, + ), + textAlign: TextAlign.center, + ), + ); + }, + ); + } + break; + case TrackOptionValue.favorite: + final isLikedTrack = await ref.read( + metadataPluginIsSavedTrackProvider(track.id).future, + ); + + if (isLikedTrack) { + await favoriteTracks.removeFavorite([track]); + } else { + await favoriteTracks.addFavorite([track]); + } + break; + case TrackOptionValue.addToPlaylist: + actionAddToPlaylist(context, playlistId); + break; + case TrackOptionValue.removeFromPlaylist: + favoritePlaylistsNotifier.removeTracks(playlistId ?? "", [track.id]); + break; + case TrackOptionValue.blacklist: + final isBlacklisted = blacklist.contains(track); + if (isBlacklisted == true) { + await ref.read(blacklistProvider.notifier).remove(track.id); + } else { + await ref.read(blacklistProvider.notifier).add( + BlacklistTableCompanion.insert( + name: track.name, + elementId: track.id, + elementType: BlacklistedType.track, + ), + ); + } + break; + case TrackOptionValue.share: + actionShare(context); + break; + case TrackOptionValue.details: + if (track is! SpotubeFullTrackObject) break; + showDialog( + context: context, + builder: (context) => ConstrainedBox( + constraints: const BoxConstraints(maxWidth: 400), + child: TrackDetailsDialog(track: track as SpotubeFullTrackObject), + ), + ); + break; + case TrackOptionValue.download: + if (track is SpotubeLocalTrackObject) break; + downloadManager.addToQueue(track as SpotubeFullTrackObject); + break; + case TrackOptionValue.startRadio: + actionStartRadio(context); + break; + } + } +} + +typedef TrackOptionFlags = ({ + bool isInQueue, + bool isBlacklisted, + bool isInDownloadQueue, + bool isActiveTrack, + bool isAuthenticated, + bool isLiked, + DownloadTask? downloadTask, +}); + +final trackOptionActionsProvider = + Provider.family( + (ref, track) => TrackOptionsActions(ref, track), +); + +final trackOptionsStateProvider = + Provider.family((ref, track) { + ref.watch(downloadManagerProvider); + ref.watch(blacklistProvider); + + final playlist = ref.watch(audioPlayerProvider); + final authenticated = ref.watch(metadataPluginAuthenticatedProvider); + final downloadManager = ref.watch(downloadManagerProvider.notifier); + final blacklist = ref.watch(blacklistProvider.notifier); + final isBlacklisted = blacklist.contains(track); + final isSavedTrack = ref.watch(metadataPluginIsSavedTrackProvider(track.id)); + + final downloadTask = playlist.activeTrack?.id == null + ? null + : downloadManager.getTaskByTrackId(playlist.activeTrack!.id); + final isInDownloadQueue = playlist.activeTrack == null || + playlist.activeTrack! is SpotubeLocalTrackObject + ? false + : const [ + DownloadStatus.queued, + DownloadStatus.downloading, + ].contains(downloadTask?.status); + + return ( + isInQueue: playlist.containsTrack(track), + isBlacklisted: isBlacklisted, + isInDownloadQueue: isInDownloadQueue, + isActiveTrack: playlist.activeTrack?.id == track.id, + isAuthenticated: authenticated.asData?.value ?? false, + isLiked: isSavedTrack.asData?.value ?? false, + downloadTask: downloadTask, + ); +}); diff --git a/lib/provider/tray_manager/tray_manager.dart b/lib/provider/tray_manager/tray_manager.dart index 9cc4becc..a976b09b 100644 --- a/lib/provider/tray_manager/tray_manager.dart +++ b/lib/provider/tray_manager/tray_manager.dart @@ -22,10 +22,10 @@ class SystemTrayManager with TrayListener { if (enabled) { await trayManager.setIcon( kIsWindows - ? 'assets/spotube-logo.ico' + ? 'assets/branding/spotube-logo.ico' : kIsFlatpak ? 'com.github.KRTirtho.Spotube' - : 'assets/spotube-logo.png', + : 'assets/branding/spotube-logo.png', ); trayManager.addListener(this); } else { diff --git a/lib/provider/user_preferences/user_preferences_provider.dart b/lib/provider/user_preferences/user_preferences_provider.dart index 053f0994..0b43d043 100644 --- a/lib/provider/user_preferences/user_preferences_provider.dart +++ b/lib/provider/user_preferences/user_preferences_provider.dart @@ -1,17 +1,15 @@ import 'package:drift/drift.dart'; -import 'package:flutter/material.dart'; + import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:path/path.dart'; import 'package:path_provider/path_provider.dart' as paths; -import 'package:spotify/spotify.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide join; import 'package:spotube/models/database/database.dart'; +import 'package:spotube/models/metadata/market.dart'; import 'package:spotube/modules/settings/color_scheme_picker_dialog.dart'; -import 'package:spotube/provider/audio_player/audio_player_streams.dart'; import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/provider/palette_provider.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/logger/logger.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; import 'package:spotube/utils/platform.dart'; import 'package:window_manager/window_manager.dart'; import 'package:open_file/open_file.dart'; @@ -91,9 +89,9 @@ class UserPreferencesNotifier extends Notifier { Future reset() async { final db = ref.read(databaseProvider); - final query = db.update(db.preferencesTable)..where((t) => t.id.equals(0)); + final query = db.update(db.preferencesTable); - await query.replace(PreferencesTableCompanion.insert()); + await query.replace(PreferencesTableCompanion.insert(id: const Value(0))); } static Future getMusicCacheDir() async { @@ -120,14 +118,6 @@ class UserPreferencesNotifier extends Notifier { } } - void setStreamMusicCodec(SourceCodecs codec) { - setData(PreferencesTableCompanion(streamMusicCodec: Value(codec))); - } - - void setDownloadMusicCodec(SourceCodecs codec) { - setData(PreferencesTableCompanion(downloadMusicCodec: Value(codec))); - } - void setThemeMode(ThemeMode mode) { setData(PreferencesTableCompanion(themeMode: Value(mode))); } @@ -143,21 +133,17 @@ class UserPreferencesNotifier extends Notifier { void setAlbumColorSync(bool sync) { setData(PreferencesTableCompanion(albumColorSync: Value(sync))); - if (!sync) { - ref.read(paletteProvider.notifier).state = null; - } else { - ref.read(audioPlayerStreamListenersProvider).updatePalette(); - } + // if (!sync) { + // ref.read(paletteProvider.notifier).state = null; + // } else { + // ref.read(audioPlayerStreamListenersProvider).updatePalette(); + // } } void setCheckUpdate(bool check) { setData(PreferencesTableCompanion(checkUpdate: Value(check))); } - void setAudioQuality(SourceQualities quality) { - setData(PreferencesTableCompanion(audioQuality: Value(quality))); - } - void setDownloadLocation(String downloadDir) { if (downloadDir.isEmpty) return; setData(PreferencesTableCompanion(downloadLocation: Value(downloadDir))); @@ -188,14 +174,6 @@ class UserPreferencesNotifier extends Notifier { setData(PreferencesTableCompanion(locale: Value(locale))); } - void setPipedInstance(String instance) { - setData(PreferencesTableCompanion(pipedInstance: Value(instance))); - } - - void setInvidiousInstance(String instance) { - setData(PreferencesTableCompanion(invidiousInstance: Value(instance))); - } - void setSearchMode(SearchMode mode) { setData(PreferencesTableCompanion(searchMode: Value(mode))); } @@ -204,8 +182,8 @@ class UserPreferencesNotifier extends Notifier { setData(PreferencesTableCompanion(skipNonMusic: Value(skip))); } - void setAudioSource(AudioSource type) { - setData(PreferencesTableCompanion(audioSource: Value(type))); + void setYoutubeClientEngine(YoutubeClientEngine engine) { + setData(PreferencesTableCompanion(youtubeClientEngine: Value(engine))); } void setSystemTitleBar(bool isSystemTitleBar) { @@ -237,6 +215,14 @@ class UserPreferencesNotifier extends Notifier { setData(PreferencesTableCompanion(enableConnect: Value(enable))); } + void setConnectPort(int port) { + assert( + port >= -1 && port <= 65535, + "Port must be between -1 and 65535, got $port", + ); + setData(PreferencesTableCompanion(connectPort: Value(port))); + } + void setCacheMusic(bool cache) { setData(PreferencesTableCompanion(cacheMusic: Value(cache))); } diff --git a/lib/provider/youtube_engine/youtube_engine.dart b/lib/provider/youtube_engine/youtube_engine.dart new file mode 100644 index 00000000..0aa37db5 --- /dev/null +++ b/lib/provider/youtube_engine/youtube_engine.dart @@ -0,0 +1,22 @@ +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:spotube/models/database/database.dart'; +import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +import 'package:spotube/services/youtube_engine/newpipe_engine.dart'; +import 'package:spotube/services/youtube_engine/youtube_explode_engine.dart'; +import 'package:spotube/services/youtube_engine/yt_dlp_engine.dart'; + +final youtubeEngineProvider = Provider((ref) { + final engineMode = ref.watch( + userPreferencesProvider.select((value) => value.youtubeClientEngine), + ); + + if (engineMode == YoutubeClientEngine.newPipe && + NewPipeEngine.isAvailableForPlatform) { + return NewPipeEngine(); + } else if (engineMode == YoutubeClientEngine.ytDlp && + YtDlpEngine.isAvailableForPlatform) { + return YtDlpEngine(); + } else { + return YouTubeExplodeEngine(); + } +}); diff --git a/lib/services/audio_player/audio_player.dart b/lib/services/audio_player/audio_player.dart index 4febecdf..2693f13a 100644 --- a/lib/services/audio_player/audio_player.dart +++ b/lib/services/audio_player/audio_player.dart @@ -1,82 +1,44 @@ import 'dart:io'; import 'package:media_kit/media_kit.dart' hide Track; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/services/logger/logger.dart'; import 'package:flutter/foundation.dart'; -import 'package:spotify/spotify.dart' hide Playlist; -import 'package:spotube/models/local_track.dart'; import 'package:spotube/services/audio_player/custom_player.dart'; import 'dart:async'; import 'package:media_kit/media_kit.dart' as mk; import 'package:spotube/services/audio_player/playback_state.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; import 'package:spotube/utils/platform.dart'; part 'audio_players_streams_mixin.dart'; part 'audio_player_impl.dart'; class SpotubeMedia extends mk.Media { - final Track track; - static int serverPort = 0; - SpotubeMedia( - this.track, { - Map? extras, - super.httpHeaders, - }) : super( - track is LocalTrack + static String get _host => + kIsWindows ? "localhost" : InternetAddress.anyIPv4.address; + + final SpotubeTrackObject track; + SpotubeMedia(this.track) + : assert( + track is SpotubeLocalTrackObject || track is SpotubeFullTrackObject, + "Track must be a either a local track or a full track object with ISRC", + ), + // If the track is a local track, use its path, otherwise use the server URL + super( + track is SpotubeLocalTrackObject ? track.path - : "http://${kIsWindows ? "localhost" : InternetAddress.anyIPv4.address}:$serverPort/stream/${track.id}", - extras: { - ...?extras, - "track": switch (track) { - LocalTrack() => track.toJson(), - SourcedTrack() => track.toJson(), - _ => track.toJson(), - }, - }, + : "http://$_host:$serverPort/stream/${track.id}", + extras: track.toJson(), ); - @override - String get uri { - return switch (track) { - /// [super.uri] must be used instead of [track.path] to prevent wrong - /// path format exceptions in Windows causing [extras] to be null - LocalTrack() => super.uri, - _ => - "http://${kIsWindows ? "localhost" : InternetAddress.anyIPv4.address}:" - "$serverPort/stream/${track.id}", - }; + factory SpotubeMedia.media(Media media) { + assert(media.extras != null, "[Media] must have extra metadata set"); + return SpotubeMedia(SpotubeTrackObject.fromJson(media.extras!)); } - - factory SpotubeMedia.fromMedia(mk.Media media) { - final track = media.uri.startsWith("http") - ? Track.fromJson(media.extras?["track"]) - : LocalTrack.fromJson(media.extras?["track"]); - return SpotubeMedia( - track, - extras: media.extras, - httpHeaders: media.httpHeaders, - ); - } - - // @override - // operator ==(Object other) { - // if (other is! SpotubeMedia) return false; - - // final isLocal = track is LocalTrack && other.track is LocalTrack; - // return isLocal - // ? (other.track as LocalTrack).path == (track as LocalTrack).path - // : other.track.id == track.id; - // } - - // @override - // int get hashCode => track is LocalTrack - // ? (track as LocalTrack).path.hashCode - // : track.id.hashCode; } abstract class AudioPlayerInterface { @@ -87,6 +49,7 @@ abstract class AudioPlayerInterface { configuration: const mk.PlayerConfiguration( title: "Spotube", logLevel: kDebugMode ? mk.MPVLogLevel.info : mk.MPVLogLevel.error, + async: true, ), ) { _mkPlayer.stream.error.listen((event) { diff --git a/lib/services/audio_player/audio_player_impl.dart b/lib/services/audio_player/audio_player_impl.dart index 82c8c906..afd209a3 100644 --- a/lib/services/audio_player/audio_player_impl.dart +++ b/lib/services/audio_player/audio_player_impl.dart @@ -131,4 +131,8 @@ class SpotubeAudioPlayer extends AudioPlayerInterface Future setAudioNormalization(bool normalize) async { await _mkPlayer.setAudioNormalization(normalize); } + + Future setDemuxerBufferSize(int sizeInBytes) async { + await _mkPlayer.setDemuxerBufferSize(sizeInBytes); + } } diff --git a/lib/services/audio_player/audio_players_streams_mixin.dart b/lib/services/audio_player/audio_players_streams_mixin.dart index 32405910..aeb8f1e3 100644 --- a/lib/services/audio_player/audio_players_streams_mixin.dart +++ b/lib/services/audio_player/audio_players_streams_mixin.dart @@ -146,7 +146,5 @@ mixin SpotubeAudioPlayersStreams on AudioPlayerInterface { Stream get errorStream => _mkPlayer.stream.error; - Stream get playlistStream => _mkPlayer.stream.playlist.map((s) { - return s; - }); + Stream get playlistStream => _mkPlayer.stream.playlist; } diff --git a/lib/services/audio_player/custom_player.dart b/lib/services/audio_player/custom_player.dart index f0dc8f13..7cbd51a5 100644 --- a/lib/services/audio_player/custom_player.dart +++ b/lib/services/audio_player/custom_player.dart @@ -12,19 +12,15 @@ import 'package:spotube/utils/platform.dart'; /// This class adds a state stream to the [Player] class. class CustomPlayer extends Player { final StreamController _playerStateStream; - final StreamController _shuffleStream; late final List _subscriptions; - bool _shuffled; int _androidAudioSessionId = 0; String _packageName = ""; AndroidAudioManager? _androidAudioManager; CustomPlayer({super.configuration}) - : _playerStateStream = StreamController.broadcast(), - _shuffleStream = StreamController.broadcast(), - _shuffled = false { + : _playerStateStream = StreamController.broadcast() { nativePlayer.setProperty("network-timeout", "120"); _subscriptions = [ @@ -86,10 +82,10 @@ class CustomPlayer extends Player { } } - bool get shuffled => _shuffled; + bool get shuffled => state.shuffle; Stream get playerStateStream => _playerStateStream.stream; - Stream get shuffleStream => _shuffleStream.stream; + Stream get shuffleStream => stream.shuffle; Stream get indexChangeStream { int oldIndex = state.playlist.index; return stream.playlist.map((event) => event.index).where((newIndex) { @@ -103,22 +99,14 @@ class CustomPlayer extends Player { @override Future setShuffle(bool shuffle) async { - _shuffled = shuffle; await super.setShuffle(shuffle); - _shuffleStream.add(shuffle); - await Future.delayed(const Duration(milliseconds: 100)); - if (shuffle) { - await move(state.playlist.index, 0); - } } @override Future stop() async { await super.stop(); - _shuffled = false; _playerStateStream.add(AudioPlaybackState.stopped); - _shuffleStream.add(false); } @override @@ -133,8 +121,23 @@ class CustomPlayer extends Player { NativePlayer get nativePlayer => platform as NativePlayer; Future insert(int index, Media media) async { - await add(media); - await move(state.playlist.medias.length, index); + final addedMediaCompleter = Completer(); + final playlistStream = stream.playlist.listen( + (event) { + final mediaAddedIndex = + event.medias.indexWhere((m) => m.uri == media.uri); + if (mediaAddedIndex != -1 && !addedMediaCompleter.isCompleted) { + addedMediaCompleter.complete(mediaAddedIndex); + } + }, + ); + try { + await add(media); + final mediaAddedIndex = await addedMediaCompleter.future; + await move(mediaAddedIndex, index); + } finally { + playlistStream.cancel(); + } } Future setAudioNormalization(bool normalize) async { @@ -144,4 +147,12 @@ class CustomPlayer extends Player { await nativePlayer.setProperty('af', ''); } } + + Future setDemuxerBufferSize(int sizeInBytes) async { + await nativePlayer.setProperty('demuxer-max-bytes', sizeInBytes.toString()); + await nativePlayer.setProperty( + 'demuxer-max-back-bytes', + sizeInBytes.toString(), + ); + } } diff --git a/lib/services/audio_services/audio_services.dart b/lib/services/audio_services/audio_services.dart index 0b1843c4..c511da61 100644 --- a/lib/services/audio_services/audio_services.dart +++ b/lib/services/audio_services/audio_services.dart @@ -1,14 +1,12 @@ import 'package:audio_service/audio_service.dart'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/extensions/artist_simple.dart'; -import 'package:spotube/extensions/image.dart'; +import 'package:spotube/collections/env.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/audio_services/mobile_audio_service.dart'; import 'package:spotube/services/audio_services/windows_audio_service.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; import 'package:spotube/utils/platform.dart'; class AudioServices with WidgetsBindingObserver { @@ -27,12 +25,17 @@ class AudioServices with WidgetsBindingObserver { ? await AudioService.init( builder: () => MobileAudioService(playback), config: AudioServiceConfig( - androidNotificationChannelId: - kIsLinux ? 'spotube' : 'com.krtirtho.Spotube', + androidNotificationChannelId: switch (( + kIsLinux, + Env.releaseChannel + )) { + (true, _) => "spotube", + (_, ReleaseChannel.stable) => "oss.krtirtho.spotube", + (_, ReleaseChannel.nightly) => "oss.krtirtho.spotube.nightly", + }, androidNotificationChannelName: 'Spotube', androidNotificationOngoing: false, androidStopForegroundOnPause: false, - androidNotificationIcon: "drawable/ic_launcher_monochrome", androidNotificationChannelDescription: "Spotube Media Controls", ), ) @@ -42,20 +45,16 @@ class AudioServices with WidgetsBindingObserver { return AudioServices(mobile, smtc); } - Future addTrack(Track track) async { + Future addTrack(SpotubeTrackObject track) async { await smtc?.addTrack(track); mobile?.addItem(MediaItem( - id: track.id!, - album: track.album?.name ?? "", - title: track.name!, - artist: (track.artists)?.asString() ?? "", - duration: track is SourcedTrack - ? track.sourceInfo.duration - : Duration(milliseconds: track.durationMs ?? 0), - artUri: Uri.parse( - (track.album?.images).asUrlString( - placeholder: ImagePlaceholder.albumArt, - ), + id: track.id, + album: track.album.name, + title: track.name, + artist: track.artists.asString(), + duration: Duration(milliseconds: track.durationMs), + artUri: (track.album.images).asUri( + placeholder: ImagePlaceholder.albumArt, ), playable: true, )); diff --git a/lib/services/audio_services/windows_audio_service.dart b/lib/services/audio_services/windows_audio_service.dart index 8edc5069..6cf101ab 100644 --- a/lib/services/audio_services/windows_audio_service.dart +++ b/lib/services/audio_services/windows_audio_service.dart @@ -2,9 +2,7 @@ import 'dart:async'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:smtc_windows/smtc_windows.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/extensions/artist_simple.dart'; -import 'package:spotube/extensions/image.dart'; +import 'package:spotube/models/metadata/metadata.dart'; import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/audio_player/playback_state.dart'; @@ -77,15 +75,15 @@ class WindowsAudioService { ]); } - Future addTrack(Track track) async { + Future addTrack(SpotubeTrackObject track) async { if (!smtc.enabled) { await smtc.enableSmtc(); } await smtc.updateMetadata( MusicMetadata( - title: track.name!, - albumArtist: track.artists?.firstOrNull?.name ?? "Unknown", - artist: track.artists?.asString() ?? "Unknown", + title: track.name, + albumArtist: track.artists.firstOrNull?.name ?? "Unknown", + artist: track.artists.asString(), album: track.album?.name ?? "Unknown", thumbnail: (track.album?.images).asUrlString( placeholder: ImagePlaceholder.albumArt, diff --git a/lib/services/connectivity_adapter.dart b/lib/services/connectivity_adapter.dart index 86765671..f6b760c8 100644 --- a/lib/services/connectivity_adapter.dart +++ b/lib/services/connectivity_adapter.dart @@ -1,6 +1,7 @@ import 'dart:async'; import 'dart:io'; +import 'package:connectivity_plus/connectivity_plus.dart'; import 'package:dio/dio.dart'; import 'package:flutter/widgets.dart'; import 'package:spotube/services/logger/logger.dart'; @@ -19,6 +20,7 @@ class ConnectionCheckerService with WidgetsBindingObserver { onConnectivityChanged.listen((connected) { try { if (!connected && timer == null) { + // check every 30 seconds if we are connected when we are not connected timer = Timer.periodic(const Duration(seconds: 30), (timer) async { if (WidgetsBinding.instance.lifecycleState == AppLifecycleState.paused) { @@ -34,6 +36,10 @@ class ConnectionCheckerService with WidgetsBindingObserver { AppLogger.reportError(e, stack); } }); + + Connectivity().onConnectivityChanged.listen((event) async { + await isConnected; + }); } @override @@ -77,8 +83,9 @@ class ConnectionCheckerService with WidgetsBindingObserver { } return interfaces.any( - (interface) => - vpnNames.any((name) => interface.name.toLowerCase().contains(name)), + (interface) => vpnNames.any( + (name) => interface.name.toLowerCase().contains(name), + ), ); } @@ -105,14 +112,14 @@ class ConnectionCheckerService with WidgetsBindingObserver { await isVpnActive(); // when VPN is active that means we are connected } - bool isConnectedSync = false; + bool isConnectedSync = true; Future get isConnected async { final connected = await _isConnected(); - isConnectedSync = connected; if (connected != isConnectedSync /*previous value*/) { _connectionStreamController.add(connected); } + isConnectedSync = connected; return connected; } diff --git a/lib/services/custom_spotify_endpoints/spotify_endpoints.dart b/lib/services/custom_spotify_endpoints/spotify_endpoints.dart deleted file mode 100644 index 3b358366..00000000 --- a/lib/services/custom_spotify_endpoints/spotify_endpoints.dart +++ /dev/null @@ -1,222 +0,0 @@ -import 'dart:convert'; - -import 'package:dio/dio.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/models/spotify/home_feed.dart'; -import 'package:spotube/models/spotify_friends.dart'; -import 'package:timezone/timezone.dart' as tz; - -class CustomSpotifyEndpoints { - static const _baseUrl = 'https://api.spotify.com/v1'; - final String accessToken; - final Dio _client; - - CustomSpotifyEndpoints(this.accessToken) - : _client = Dio( - BaseOptions( - baseUrl: _baseUrl, - responseType: ResponseType.json, - headers: { - "content-type": "application/json", - if (accessToken.isNotEmpty) - "authorization": "Bearer $accessToken", - "accept": "application/json", - }, - ), - ); - - // views API - - /// Get a single view of given genre - /// - /// Currently known genres are: - /// - new-releases-page - /// - made-for-x-hub (it requires authentication) - /// - my-mix-genres (it requires authentication) - /// - artist-seed-mixes (it requires authentication) - /// - my-mix-decades (it requires authentication) - /// - my-mix-moods (it requires authentication) - /// - podcasts-and-more (it requires authentication) - /// - uniquely-yours-in-hub (it requires authentication) - /// - made-for-x-dailymix (it requires authentication) - /// - made-for-x-discovery (it requires authentication) - Future> getView( - String view, { - int limit = 20, - int contentLimit = 10, - List types = const [ - "album", - "playlist", - "artist", - "show", - "station", - "episode", - "merch", - "artist_concerts", - "uri_link" - ], - String imageStyle = "gradient_overlay", - String includeExternal = "audio", - String? locale, - Market? market, - Market? country, - }) async { - if (accessToken.isEmpty) { - throw Exception('[CustomSpotifyEndpoints.getView]: accessToken is empty'); - } - - final queryParams = { - 'limit': limit.toString(), - 'content_limit': contentLimit.toString(), - 'types': types.join(','), - 'image_style': imageStyle, - 'include_external': includeExternal, - 'timestamp': DateTime.now().toUtc().toIso8601String(), - if (locale != null) 'locale': locale, - if (market != null) 'market': market.name, - if (country != null) 'country': country.name, - }.entries.map((e) => '${e.key}=${e.value}').join('&'); - - final res = await _client.getUri( - Uri.parse('$_baseUrl/views/$view?$queryParams'), - ); - - if (res.statusCode == 200) { - return res.data; - } else { - throw Exception( - '[CustomSpotifyEndpoints.getView]: Failed to get view' - '\nStatus code: ${res.statusCode}' - '\nBody: ${res.data}', - ); - } - } - - Future> listGenreSeeds() async { - final res = await _client.getUri( - Uri.parse("$_baseUrl/recommendations/available-genre-seeds"), - ); - - if (res.statusCode == 200) { - final body = res.data; - return List.from(body["genres"] ?? []); - } else { - throw Exception( - '[CustomSpotifyEndpoints.listGenreSeeds]: Failed to get genre seeds' - '\nStatus code: ${res.statusCode}' - '\nBody: ${res.data}', - ); - } - } - - Future getFriendActivity() async { - final res = await _client.getUri( - Uri.parse("https://guc-spclient.spotify.com/presence-view/v1/buddylist"), - ); - return SpotifyFriends.fromJson(res.data); - } - - Future getHomeFeed({ - required String spTCookie, - required Market country, - }) async { - final headers = { - 'app-platform': 'WebPlayer', - 'authorization': 'Bearer $accessToken', - 'content-type': 'application/json;charset=UTF-8', - 'dnt': '1', - 'origin': 'https://open.spotify.com', - 'referer': 'https://open.spotify.com/' - }; - final response = await _client.getUri( - Uri( - scheme: "https", - host: "api-partner.spotify.com", - path: "/pathfinder/v1/query", - queryParameters: { - "operationName": "home", - "variables": jsonEncode({ - "timeZone": tz.local.name, - "sp_t": spTCookie, - "country": country.name, - "facet": null, - "sectionItemsLimit": 10 - }), - "extensions": jsonEncode( - { - "persistedQuery": { - "version": 1, - - /// GraphQL persisted Query hash - /// This can change overtime. We've to lookout for it - /// Docs: https://www.apollographql.com/docs/graphos/operations/persisted-queries/ - "sha256Hash": - "eb3fba2d388cf4fc4d696b1757a58584e9538a3b515ea742e9cc9465807340be", - } - }, - ), - }, - ), - options: Options(headers: headers), - ); - - final data = SpotifyHomeFeed.fromJson( - transformHomeFeedJsonMap(response.data), - ); - - return data; - } - - Future getHomeFeedSection( - String sectionUri, { - required String spTCookie, - required Market country, - }) async { - final headers = { - 'app-platform': 'WebPlayer', - 'authorization': 'Bearer $accessToken', - 'content-type': 'application/json;charset=UTF-8', - 'dnt': '1', - 'origin': 'https://open.spotify.com', - 'referer': 'https://open.spotify.com/' - }; - final response = await _client.getUri( - Uri( - scheme: "https", - host: "api-partner.spotify.com", - path: "/pathfinder/v1/query", - queryParameters: { - "operationName": "homeSection", - "variables": jsonEncode({ - "timeZone": tz.local.name, - "sp_t": spTCookie, - "country": country.name, - "uri": sectionUri - }), - "extensions": jsonEncode( - { - "persistedQuery": { - "version": 1, - - /// GraphQL persisted Query hash - /// This can change overtime. We've to lookout for it - /// Docs: https://www.apollographql.com/docs/graphos/operations/persisted-queries/ - "sha256Hash": - "eb3fba2d388cf4fc4d696b1757a58584e9538a3b515ea742e9cc9465807340be", - } - }, - ), - }, - ), - options: Options(headers: headers), - ); - - final data = SpotifyHomeFeedSection.fromJson( - transformSectionItemJsonMap( - response.data["data"]["homeSections"]["sections"][0], - ), - ); - - return data; - } -} diff --git a/lib/services/download_manager/chunked_download.dart b/lib/services/download_manager/chunked_download.dart deleted file mode 100644 index 80a3e78f..00000000 --- a/lib/services/download_manager/chunked_download.dart +++ /dev/null @@ -1,145 +0,0 @@ -import 'dart:async'; -import 'dart:io'; - -import 'package:dio/dio.dart'; - -/// Downloading by spiting as file in chunks -extension ChunkDownload on Dio { - Future chunkedDownload( - url, { - Map? queryParameters, - required String savePath, - ProgressCallback? onReceiveProgress, - CancelToken? cancelToken, - bool deleteOnError = true, - int chunkSize = 102400, // 100KB - int maxConcurrentChunk = 3, - String tempExtension = ".temp", - }) async { - int total = 0; - var progress = []; - - ProgressCallback createCallback(int chunkIndex) { - return (int received, _) { - progress[chunkIndex] = received; - if (onReceiveProgress != null && total != 0) { - onReceiveProgress(progress.reduce((a, b) => a + b), total); - } - }; - } - - // this is the last response - // status & headers will the last chunk's status & headers - final completer = Completer(); - - Future downloadChunk( - String url, { - required int start, - required int end, - required int chunkIndex, - }) async { - progress.add(0); - --end; - final res = await download( - url, - savePath + tempExtension + chunkIndex.toString(), - onReceiveProgress: createCallback(chunkIndex), - cancelToken: cancelToken, - queryParameters: queryParameters, - deleteOnError: deleteOnError, - options: Options( - responseType: ResponseType.bytes, - headers: {"range": "bytes=$start-$end"}, - ), - ); - - return res; - } - - Future mergeTempFiles(int chunk) async { - File headFile = File("$savePath${tempExtension}0"); - var raf = await headFile.open(mode: FileMode.writeOnlyAppend); - for (int i = 1; i < chunk; ++i) { - File chunkFile = File(savePath + tempExtension + i.toString()); - raf = await raf.writeFrom(await chunkFile.readAsBytes()); - await chunkFile.delete(); - } - await raf.close(); - - headFile = await headFile.rename(savePath); - } - - final firstResponse = await downloadChunk( - url, - start: 0, - end: chunkSize, - chunkIndex: 0, - ); - - final responses = [firstResponse]; - - if (firstResponse.statusCode == HttpStatus.partialContent) { - total = int.parse( - firstResponse.headers - .value(HttpHeaders.contentRangeHeader) - ?.split("/") - .lastOrNull ?? - '0', - ); - - final reserved = total - - int.parse( - firstResponse.headers.value(HttpHeaders.contentLengthHeader) ?? - // since its a partial content, the content length will be the chunk size - chunkSize.toString(), - ); - - int chunk = (reserved / chunkSize).ceil() + 1; - - if (chunk > 1) { - int currentChunkSize = chunkSize; - if (chunk > maxConcurrentChunk + 1) { - chunk = maxConcurrentChunk + 1; - currentChunkSize = (reserved / maxConcurrentChunk).ceil(); - } - - responses.addAll( - await Future.wait( - List.generate(maxConcurrentChunk, (i) { - int start = chunkSize + i * currentChunkSize; - return downloadChunk( - url, - start: start, - end: start + currentChunkSize, - chunkIndex: i + 1, - ); - }), - ), - ); - } - - await mergeTempFiles(chunk).then((_) { - final response = responses.last; - final isPartialStatus = - response.statusCode == HttpStatus.partialContent; - - completer.complete( - Response( - data: response.data, - headers: response.headers, - requestOptions: response.requestOptions, - statusCode: isPartialStatus ? HttpStatus.ok : response.statusCode, - statusMessage: isPartialStatus ? 'Ok' : response.statusMessage, - extra: response.extra, - isRedirect: response.isRedirect, - redirects: response.redirects, - ), - ); - }).catchError((e) { - completer.completeError(e); - }); - } - - return completer.future; - } -} diff --git a/lib/services/download_manager/download_manager.dart b/lib/services/download_manager/download_manager.dart deleted file mode 100644 index d2072bd7..00000000 --- a/lib/services/download_manager/download_manager.dart +++ /dev/null @@ -1,416 +0,0 @@ -import 'dart:async'; -import 'dart:collection'; -import 'dart:io'; -import 'package:collection/collection.dart'; - -import 'package:dio/dio.dart'; -import 'package:flutter/foundation.dart'; -import 'package:path/path.dart' as path; -import 'package:path_provider/path_provider.dart'; - -import 'package:spotube/services/download_manager/chunked_download.dart'; -import 'package:spotube/services/download_manager/download_request.dart'; -import 'package:spotube/services/download_manager/download_status.dart'; -import 'package:spotube/services/download_manager/download_task.dart'; -import 'package:spotube/services/logger/logger.dart'; -import 'package:spotube/utils/primitive_utils.dart'; - -export './download_request.dart'; -export './download_status.dart'; -export './download_task.dart'; - -typedef DownloadStatusEvent = ({ - DownloadStatus status, - DownloadRequest request -}); - -class DownloadManager { - final Map _cache = {}; - final Queue _queue = Queue(); - var dio = Dio(); - static const partialExtension = ".partial"; - static const tempExtension = ".temp"; - - // var tasks = StreamController(); - - final _statusStreamController = - StreamController.broadcast(); - Stream get statusStream => - _statusStreamController.stream; - - int maxConcurrentTasks = 2; - int runningTasks = 0; - - static final DownloadManager _dm = DownloadManager._internal(); - - DownloadManager._internal(); - - factory DownloadManager({int? maxConcurrentTasks}) { - if (maxConcurrentTasks != null) { - _dm.maxConcurrentTasks = maxConcurrentTasks; - } - return _dm; - } - - void Function(int, int) createCallback(url, int partialFileLength) => - (int received, int total) { - getDownload(url)?.progress.value = - (received + partialFileLength) / (total + partialFileLength); - - if (total == -1) {} - }; - - Future download( - String url, - String savePath, - CancelToken cancelToken, { - forceDownload = false, - }) async { - late String partialFilePath; - late File partialFile; - try { - final task = getDownload(url); - - if (task == null || task.status.value == DownloadStatus.canceled) { - return; - } - setStatus(task, DownloadStatus.downloading); - - final file = File(savePath.toString()); - - await Directory(path.dirname(savePath)).create(recursive: true); - - final tmpDirPath = await Directory( - path.join( - (await getTemporaryDirectory()).path, - "spotube-downloads", - ), - ).create(recursive: true); - - partialFilePath = path.join( - tmpDirPath.path, - path.basename(savePath) + partialExtension, - ); - partialFile = File(partialFilePath); - - final fileExist = await file.exists(); - final partialFileExist = await partialFile.exists(); - - if (fileExist) { - setStatus(task, DownloadStatus.completed); - } else if (partialFileExist) { - final partialFileLength = await partialFile.length(); - - final response = await dio.download( - url, - partialFilePath + tempExtension, - onReceiveProgress: createCallback(url, partialFileLength), - options: Options( - headers: { - HttpHeaders.rangeHeader: 'bytes=$partialFileLength-', - HttpHeaders.connectionHeader: "close", - }, - ), - cancelToken: cancelToken, - deleteOnError: true, - ); - - if (response.statusCode == HttpStatus.partialContent) { - final ioSink = partialFile.openWrite(mode: FileMode.writeOnlyAppend); - final partialChunkFile = File(partialFilePath + tempExtension); - await ioSink.addStream(partialChunkFile.openRead()); - await partialChunkFile.delete(); - await ioSink.close(); - - await partialFile.copy(savePath); - await partialFile.delete(); - - setStatus(task, DownloadStatus.completed); - } - } else { - final response = await dio.chunkedDownload( - url, - savePath: partialFilePath, - onReceiveProgress: createCallback(url, 0), - cancelToken: cancelToken, - deleteOnError: true, - ); - - if (response.statusCode == HttpStatus.ok) { - await partialFile.copy(savePath); - await partialFile.delete(); - setStatus(task, DownloadStatus.completed); - } - } - } catch (e, stackTrace) { - AppLogger.reportError(e, stackTrace); - - var task = getDownload(url)!; - if (task.status.value != DownloadStatus.canceled && - task.status.value != DownloadStatus.paused) { - setStatus(task, DownloadStatus.failed); - runningTasks--; - - if (_queue.isNotEmpty) { - _startExecution(); - } - rethrow; - } else if (task.status.value == DownloadStatus.paused) { - final ioSink = partialFile.openWrite(mode: FileMode.writeOnlyAppend); - final f = File(partialFilePath + tempExtension); - if (await f.exists()) { - await ioSink.addStream(f.openRead()); - } - await ioSink.close(); - } - } - - runningTasks--; - - if (_queue.isNotEmpty) { - _startExecution(); - } - } - - void disposeNotifiers(DownloadTask task) { - // task.status.dispose(); - // task.progress.dispose(); - } - - void setStatus(DownloadTask? task, DownloadStatus status) { - if (task != null) { - task.status.value = status; - - // tasks.add(task); - if (status.isCompleted) { - disposeNotifiers(task); - } - - _statusStreamController.add((status: status, request: task.request)); - } - } - - Future addDownload(String url, String savedPath) async { - if (url.isEmpty) throw Exception("Invalid Url. Url is empty: $url"); - return _addDownloadRequest(DownloadRequest(url, savedPath)); - } - - Future _addDownloadRequest( - DownloadRequest downloadRequest, - ) async { - if (_cache[downloadRequest.url] != null) { - if (!_cache[downloadRequest.url]!.status.value.isCompleted && - _cache[downloadRequest.url]!.request == downloadRequest) { - // Do nothing - return _cache[downloadRequest.url]!; - } else { - _queue.remove(_cache[downloadRequest.url]?.request); - } - } - - _queue.add(DownloadRequest(downloadRequest.url, downloadRequest.path)); - - final task = DownloadTask(_queue.last); - - _cache[downloadRequest.url] = task; - - _startExecution(); - - return task; - } - - Future pauseDownload(String url) async { - var task = getDownload(url)!; - setStatus(task, DownloadStatus.paused); - task.request.cancelToken.cancel(); - - _queue.remove(task.request); - } - - Future cancelDownload(String url) async { - var task = getDownload(url)!; - setStatus(task, DownloadStatus.canceled); - _queue.remove(task.request); - task.request.cancelToken.cancel(); - } - - Future resumeDownload(String url) async { - var task = getDownload(url)!; - setStatus(task, DownloadStatus.downloading); - task.request.cancelToken = CancelToken(); - _queue.add(task.request); - - _startExecution(); - } - - Future removeDownload(String url) async { - cancelDownload(url); - _cache.remove(url); - } - - // Do not immediately call getDownload After addDownload, rather use the returned DownloadTask from addDownload - DownloadTask? getDownload(String url) { - return _cache[url]; - } - - Future whenDownloadComplete(String url, - {Duration timeout = const Duration(hours: 2)}) async { - DownloadTask? task = getDownload(url); - - if (task != null) { - return task.whenDownloadComplete(timeout: timeout); - } else { - return Future.error("Not found"); - } - } - - List getAllDownloads() { - return _cache.values.toList(); - } - - // Batch Download Mechanism - Future addBatchDownloads(List urls, String savePath) async { - for (final url in urls) { - addDownload(url, savePath); - } - } - - List getBatchDownloads(List urls) { - return urls.map((e) => _cache[e]).toList(); - } - - Future pauseBatchDownloads(List urls) async { - for (var element in urls) { - pauseDownload(element); - } - } - - Future cancelBatchDownloads(List urls) async { - for (var element in urls) { - cancelDownload(element); - } - } - - Future resumeBatchDownloads(List urls) async { - for (var element in urls) { - resumeDownload(element); - } - } - - ValueNotifier getBatchDownloadProgress(List urls) { - ValueNotifier progress = ValueNotifier(0); - var total = urls.length; - - if (total == 0) { - return progress; - } - - if (total == 1) { - return getDownload(urls.first)?.progress ?? progress; - } - - var progressMap = {}; - - for (var url in urls) { - DownloadTask? task = getDownload(url); - - if (task != null) { - progressMap[url] = 0.0; - - if (task.status.value.isCompleted) { - progressMap[url] = 1.0; - progress.value = progressMap.values.sum / total; - } - - void progressListener() { - progressMap[url] = task.progress.value; - progress.value = progressMap.values.sum / total; - } - - task.progress.addListener(progressListener); - - void listener() { - if (task.status.value.isCompleted) { - progressMap[url] = 1.0; - progress.value = progressMap.values.sum / total; - task.status.removeListener(listener); - task.progress.removeListener(progressListener); - } - } - - task.status.addListener(listener); - } else { - total--; - } - } - - return progress; - } - - Future?> whenBatchDownloadsComplete(List urls, - {Duration timeout = const Duration(hours: 2)}) async { - var completer = Completer?>(); - - var completed = 0; - var total = urls.length; - - for (final url in urls) { - DownloadTask? task = getDownload(url); - - if (task != null) { - if (task.status.value.isCompleted) { - completed++; - - if (completed == total) { - completer.complete(getBatchDownloads(urls)); - } - } - - void listener() { - if (task.status.value.isCompleted) { - completed++; - - if (completed == total) { - completer.complete(getBatchDownloads(urls)); - task.status.removeListener(listener); - } - } - } - - task.status.addListener(listener); - } else { - total--; - - if (total == 0) { - completer.complete(null); - } - } - } - - return completer.future.timeout(timeout); - } - - void _startExecution() async { - if (runningTasks == maxConcurrentTasks || _queue.isEmpty) { - return; - } - - while (_queue.isNotEmpty && runningTasks < maxConcurrentTasks) { - runningTasks++; - var currentRequest = _queue.removeFirst(); - - await download( - currentRequest.url, - currentRequest.path, - currentRequest.cancelToken, - ); - - await Future.delayed(const Duration(milliseconds: 500), null); - } - } - - /// This function is used for get file name with extension from url - String getFileNameFromUrl(String url) { - return PrimitiveUtils.toSafeFileName(url.split('/').last); - } -} diff --git a/lib/services/download_manager/download_request.dart b/lib/services/download_manager/download_request.dart deleted file mode 100644 index 80c4af37..00000000 --- a/lib/services/download_manager/download_request.dart +++ /dev/null @@ -1,24 +0,0 @@ -import 'package:dio/dio.dart'; - -class DownloadRequest { - final String url; - final String path; - var cancelToken = CancelToken(); - var forceDownload = false; - - DownloadRequest( - this.url, - this.path, - ); - - @override - bool operator ==(Object other) => - identical(this, other) || - other is DownloadRequest && - runtimeType == other.runtimeType && - url == other.url && - path == other.path; - - @override - int get hashCode => url.hashCode ^ path.hashCode; -} diff --git a/lib/services/download_manager/download_status.dart b/lib/services/download_manager/download_status.dart deleted file mode 100644 index b97080fa..00000000 --- a/lib/services/download_manager/download_status.dart +++ /dev/null @@ -1,26 +0,0 @@ -enum DownloadStatus { - queued, - downloading, - completed, - failed, - paused, - canceled; - - bool get isCompleted { - switch (this) { - case DownloadStatus.queued: - return false; - case DownloadStatus.downloading: - return false; - case DownloadStatus.paused: - return false; - case DownloadStatus.completed: - return true; - case DownloadStatus.failed: - return true; - - case DownloadStatus.canceled: - return true; - } - } -} diff --git a/lib/services/download_manager/download_task.dart b/lib/services/download_manager/download_task.dart deleted file mode 100644 index d79cf95b..00000000 --- a/lib/services/download_manager/download_task.dart +++ /dev/null @@ -1,35 +0,0 @@ -import 'dart:async'; - -import 'package:flutter/foundation.dart'; -import 'package:spotube/services/download_manager/download_request.dart'; -import 'package:spotube/services/download_manager/download_status.dart'; - -class DownloadTask { - final DownloadRequest request; - ValueNotifier status = ValueNotifier(DownloadStatus.queued); - ValueNotifier progress = ValueNotifier(0); - - DownloadTask( - this.request, - ); - - Future whenDownloadComplete( - {Duration timeout = const Duration(hours: 2)}) async { - var completer = Completer(); - - if (status.value.isCompleted) { - completer.complete(status.value); - } - - void listener() { - if (status.value.isCompleted) { - completer.complete(status.value); - status.removeListener(listener); - } - } - - status.addListener(listener); - - return completer.future.timeout(timeout); - } -} diff --git a/lib/services/kv_store/kv_store.dart b/lib/services/kv_store/kv_store.dart index efe83abf..e334322e 100644 --- a/lib/services/kv_store/kv_store.dart +++ b/lib/services/kv_store/kv_store.dart @@ -2,6 +2,7 @@ import 'dart:convert'; import 'package:encrypt/encrypt.dart'; import 'package:shared_preferences/shared_preferences.dart'; +import 'package:spotube/models/database/database.dart'; import 'package:spotube/services/wm_tools/wm_tools.dart'; import 'package:uuid/uuid.dart'; @@ -87,4 +88,31 @@ abstract class KVStoreService { sharedPreferences.getBool('hasMigratedToDrift') ?? false; static Future setHasMigratedToDrift(bool value) async => await sharedPreferences.setBool('hasMigratedToDrift', value); + + static Map? get _youtubeEnginePaths { + final jsonRaw = sharedPreferences.getString('ytDlpPath'); + + if (jsonRaw == null) { + return null; + } + + return jsonDecode(jsonRaw); + } + + static String? getYoutubeEnginePath(YoutubeClientEngine engine) { + return _youtubeEnginePaths?[engine.name]; + } + + static Future setYoutubeEnginePath( + YoutubeClientEngine engine, + String path, + ) async { + await sharedPreferences.setString( + 'ytDlpPath', + jsonEncode({ + ...?_youtubeEnginePaths, + engine.name: path, + }), + ); + } } diff --git a/lib/services/logger/logger.dart b/lib/services/logger/logger.dart index 1df7b5aa..1f15bf92 100644 --- a/lib/services/logger/logger.dart +++ b/lib/services/logger/logger.dart @@ -3,12 +3,26 @@ import 'dart:io'; import 'dart:isolate'; import 'package:flutter/foundation.dart'; -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart' hide join; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:logger/logger.dart'; import 'package:path/path.dart'; import 'package:path_provider/path_provider.dart'; import 'package:spotube/utils/platform.dart'; +import 'package:logging/logging.dart' as logging; + +final _loggingToLoggerLevel = { + logging.Level.ALL: Level.all, + logging.Level.FINEST: Level.trace, + logging.Level.FINER: Level.debug, + logging.Level.FINE: Level.info, + logging.Level.CONFIG: Level.info, + logging.Level.INFO: Level.info, + logging.Level.WARNING: Level.warning, + logging.Level.SEVERE: Level.error, + logging.Level.SHOUT: Level.fatal, + logging.Level.OFF: Level.off, +}; class AppLogger { static late final Logger log; @@ -20,6 +34,24 @@ class AppLogger { ); } + static void _initInternalPackageLoggers() { + if (!kDebugMode) return; + logging.hierarchicalLoggingEnabled = true; + logging.Logger('YoutubeExplode.StreamsClient') + ..level = logging.Level.SEVERE + ..onRecord.listen( + (record) { + log.log( + _loggingToLoggerLevel[record.level] ?? Level.info, + record.message, + error: record.error, + stackTrace: record.stackTrace, + time: record.time, + ); + }, + ); + } + static R? runZoned(R Function() body) { return runZonedGuarded( () { @@ -46,6 +78,8 @@ class AppLogger { ); } + _initInternalPackageLoggers(); + getLogsPath().then((value) => logFile = value); return body(); diff --git a/lib/services/metadata/apis/localstorage.dart b/lib/services/metadata/apis/localstorage.dart new file mode 100644 index 00000000..4c511e77 --- /dev/null +++ b/lib/services/metadata/apis/localstorage.dart @@ -0,0 +1,78 @@ +import 'package:hetu_spotube_plugin/hetu_spotube_plugin.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +class SharedPreferencesLocalStorage implements Localstorage { + final SharedPreferences _prefs; + final String pluginSlug; + + SharedPreferencesLocalStorage(this._prefs, this.pluginSlug); + + String prefix(String key) { + return 'spotube_plugin.$pluginSlug.$key'; + } + + @override + Future clear() { + return _prefs.clear(); + } + + @override + Future containsKey(String key) async { + return _prefs.containsKey(prefix(key)); + } + + @override + Future getBool(String key) async { + return _prefs.getBool(prefix(key)); + } + + @override + Future getDouble(String key) async { + return _prefs.getDouble(prefix(key)); + } + + @override + Future getInt(String key) async { + return _prefs.getInt(prefix(key)); + } + + @override + Future getString(String key) async { + return _prefs.getString(prefix(key)); + } + + @override + Future?> getStringList(String key) async { + return _prefs.getStringList(prefix(key)); + } + + @override + Future remove(String key) async { + await _prefs.remove(prefix(key)); + } + + @override + Future setBool(String key, bool value) async { + await _prefs.setBool(prefix(key), value); + } + + @override + Future setDouble(String key, double value) async { + await _prefs.setDouble(prefix(key), value); + } + + @override + Future setInt(String key, int value) async { + await _prefs.setInt(prefix(key), value); + } + + @override + Future setString(String key, String value) async { + await _prefs.setString(prefix(key), value); + } + + @override + Future setStringList(String key, List value) async { + await _prefs.setStringList(prefix(key), value); + } +} diff --git a/lib/services/metadata/endpoints/album.dart b/lib/services/metadata/endpoints/album.dart new file mode 100644 index 00000000..8a858343 --- /dev/null +++ b/lib/services/metadata/endpoints/album.dart @@ -0,0 +1,75 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginAlbumEndpoint { + final Hetu hetu; + MetadataPluginAlbumEndpoint(this.hetu); + + HTInstance get hetuMetadataAlbum => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("album") + as HTInstance; + + Future getAlbum(String id) async { + final raw = + await hetuMetadataAlbum.invoke("getAlbum", positionalArgs: [id]) as Map; + + return SpotubeFullAlbumObject.fromJson( + raw.cast(), + ); + } + + Future> tracks( + String id, { + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataAlbum.invoke( + "tracks", + positionalArgs: [id], + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => + SpotubeFullTrackObject.fromJson(json.cast()), + ); + } + + Future> releases({ + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataAlbum.invoke( + "releases", + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => + SpotubeSimpleAlbumObject.fromJson(json.cast()), + ); + } + + Future save(List ids) async { + await hetuMetadataAlbum.invoke( + "save", + positionalArgs: [ids], + ); + } + + Future unsave(List ids) async { + await hetuMetadataAlbum.invoke( + "unsave", + positionalArgs: [ids], + ); + } +} diff --git a/lib/services/metadata/endpoints/artist.dart b/lib/services/metadata/endpoints/artist.dart new file mode 100644 index 00000000..d008ce61 --- /dev/null +++ b/lib/services/metadata/endpoints/artist.dart @@ -0,0 +1,101 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginArtistEndpoint { + final Hetu hetu; + MetadataPluginArtistEndpoint(this.hetu); + + HTInstance get hetuMetadataArtist => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("artist") + as HTInstance; + + Future getArtist(String id) async { + final raw = await hetuMetadataArtist + .invoke("getArtist", positionalArgs: [id]) as Map; + + return SpotubeFullArtistObject.fromJson( + raw.cast(), + ); + } + + Future> topTracks( + String id, { + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataArtist.invoke( + "topTracks", + positionalArgs: [id], + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => SpotubeFullTrackObject.fromJson( + json.cast(), + ), + ); + } + + Future> albums( + String id, { + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataArtist.invoke( + "albums", + positionalArgs: [id], + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => SpotubeSimpleAlbumObject.fromJson( + json.cast(), + ), + ); + } + + Future save(List ids) async { + await hetuMetadataArtist.invoke( + "save", + positionalArgs: [ids], + ); + } + + Future unsave(List ids) async { + await hetuMetadataArtist.invoke( + "unsave", + positionalArgs: [ids], + ); + } + + Future> related( + String id, { + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataArtist.invoke( + "related", + positionalArgs: [id], + namedArgs: { + "offset": offset, + "limit": limit ?? 20, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => SpotubeFullArtistObject.fromJson( + json.cast(), + ), + ); + } +} diff --git a/lib/services/metadata/endpoints/audio_source.dart b/lib/services/metadata/endpoints/audio_source.dart new file mode 100644 index 00000000..d22449c6 --- /dev/null +++ b/lib/services/metadata/endpoints/audio_source.dart @@ -0,0 +1,38 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginAudioSourceEndpoint { + final Hetu hetu; + MetadataPluginAudioSourceEndpoint(this.hetu); + + HTInstance get hetuMetadataAudioSource => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("audioSource") + as HTInstance; + + List get supportedPresets { + final raw = hetuMetadataAudioSource.memberGet("supportedPresets") as List; + + return raw + .map((e) => SpotubeAudioSourceContainerPreset.fromJson(e)) + .toList(); + } + + Future> matches( + SpotubeFullTrackObject track, + ) async { + final raw = await hetuMetadataAudioSource + .invoke("matches", positionalArgs: [track.toJson()]) as List; + + return raw.map((e) => SpotubeAudioSourceMatchObject.fromJson(e)).toList(); + } + + Future> streams( + SpotubeAudioSourceMatchObject match, + ) async { + final raw = await hetuMetadataAudioSource + .invoke("streams", positionalArgs: [match.toJson()]) as List; + + return raw.map((e) => SpotubeAudioSourceStreamObject.fromJson(e)).toList(); + } +} diff --git a/lib/services/metadata/endpoints/auth.dart b/lib/services/metadata/endpoints/auth.dart new file mode 100644 index 00000000..7c2077be --- /dev/null +++ b/lib/services/metadata/endpoints/auth.dart @@ -0,0 +1,33 @@ +import 'package:desktop_webview_window/desktop_webview_window.dart'; +import 'package:flutter_inappwebview/flutter_inappwebview.dart'; +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_std/hetu_std.dart'; +import 'package:spotube/utils/platform.dart'; + +class MetadataAuthEndpoint { + final Hetu hetu; + + MetadataAuthEndpoint(this.hetu); + + Stream get authStateStream => + hetu.eval("metadataPlugin.auth.authStateStream"); + + Future authenticate() async { + await hetu.eval("metadataPlugin.auth.authenticate()"); + } + + bool isAuthenticated() { + return hetu.eval("metadataPlugin.auth.isAuthenticated()") as bool; + } + + Future logout() async { + await hetu.eval("metadataPlugin.auth.logout()"); + if (kIsMobile) { + WebStorageManager.instance().deleteAllData(); + CookieManager.instance().deleteAllCookies(); + } + if (kIsDesktop) { + await WebviewWindow.clearAll(); + } + } +} diff --git a/lib/services/metadata/endpoints/browse.dart b/lib/services/metadata/endpoints/browse.dart new file mode 100644 index 00000000..c8105ad1 --- /dev/null +++ b/lib/services/metadata/endpoints/browse.dart @@ -0,0 +1,87 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginBrowseEndpoint { + final Hetu hetu; + MetadataPluginBrowseEndpoint(this.hetu); + + HTInstance get hetuMetadataBrowse => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("browse") + as HTInstance; + + Future>> + sections({ + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataBrowse.invoke( + "sections", + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject< + SpotubeBrowseSectionObject>.fromJson( + raw.cast(), + (Map json) => SpotubeBrowseSectionObject.fromJson( + json.cast(), + (json) { + final isPlaylist = json["owner"] != null; + final isAlbum = json["artists"] != null; + if (isPlaylist) { + return SpotubeSimplePlaylistObject.fromJson( + json.cast(), + ); + } else if (isAlbum) { + return SpotubeSimpleAlbumObject.fromJson( + json.cast(), + ); + } else { + return SpotubeFullArtistObject.fromJson( + json.cast(), + ); + } + }, + ), + ); + } + + Future> sectionItems( + String id, { + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataBrowse.invoke( + "sectionItems", + positionalArgs: [id], + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (json) { + final isPlaylist = json["owner"] != null; + final isAlbum = json["artists"] != null; + if (isPlaylist) { + return SpotubeSimplePlaylistObject.fromJson( + json.cast(), + ); + } else if (isAlbum) { + return SpotubeSimpleAlbumObject.fromJson( + json.cast(), + ); + } else { + return SpotubeFullArtistObject.fromJson( + json.cast(), + ); + } + }, + ); + } +} diff --git a/lib/services/metadata/endpoints/core.dart b/lib/services/metadata/endpoints/core.dart new file mode 100644 index 00000000..a8f86128 --- /dev/null +++ b/lib/services/metadata/endpoints/core.dart @@ -0,0 +1,53 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginCore { + final Hetu hetu; + + MetadataPluginCore(this.hetu); + + HTInstance get hetuMetadataPluginUpdater => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("core") + as HTInstance; + + Future checkUpdate( + PluginConfiguration pluginConfig, + ) async { + final result = await hetuMetadataPluginUpdater.invoke( + "checkUpdate", + positionalArgs: [pluginConfig.toJson()], + ); + + return result == null + ? null + : PluginUpdateAvailable.fromJson( + (result as Map).cast(), + ); + } + + Future get support async { + final result = await hetuMetadataPluginUpdater.memberGet("support"); + + return result as String; + } + + /// [details] is a map containing the scrobble information, such as: + /// - [id] -> The unique identifier of the track. + /// - [title] -> The title of the track. + /// - [artists] -> List of artists + /// - [id] -> The unique identifier of the artist. + /// - [name] -> The name of the artist. + /// - [album] -> The album of the track + /// - [id] -> The unique identifier of the album. + /// - [name] -> The name of the album. + /// - [timestamp] -> The timestamp of the scrobble (optional). + /// - [duration_ms] -> The duration of the track in milliseconds (optional). + /// - [isrc] -> The ISRC code of the track (optional). + Future scrobble(Map details) { + return hetuMetadataPluginUpdater.invoke( + "scrobble", + positionalArgs: [details], + ); + } +} diff --git a/lib/services/metadata/endpoints/playlist.dart b/lib/services/metadata/endpoints/playlist.dart new file mode 100644 index 00000000..c7f20487 --- /dev/null +++ b/lib/services/metadata/endpoints/playlist.dart @@ -0,0 +1,135 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginPlaylistEndpoint { + final Hetu hetu; + MetadataPluginPlaylistEndpoint(this.hetu); + + HTInstance get hetuMetadataPlaylist => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("playlist") + as HTInstance; + + Future getPlaylist(String id) async { + final raw = await hetuMetadataPlaylist + .invoke("getPlaylist", positionalArgs: [id]) as Map; + + return SpotubeFullPlaylistObject.fromJson( + raw.cast(), + ); + } + + Future> tracks( + String id, { + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataPlaylist.invoke( + "tracks", + positionalArgs: [id], + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => + SpotubeFullTrackObject.fromJson(json.cast()), + ); + } + + Future create( + String userId, { + required String name, + String? description, + bool? public, + bool? collaborative, + }) async { + final raw = await hetuMetadataPlaylist.invoke( + "create", + positionalArgs: [userId], + namedArgs: { + "name": name, + "description": description, + "public": public, + "collaborative": collaborative, + }..removeWhere((key, value) => value == null), + ) as Map?; + + if (raw == null) return null; + + return SpotubeFullPlaylistObject.fromJson( + raw.cast(), + ); + } + + Future update( + String playlistId, { + String? name, + String? description, + bool? public, + bool? collaborative, + }) async { + await hetuMetadataPlaylist.invoke( + "update", + positionalArgs: [playlistId], + namedArgs: { + "name": name, + "description": description, + "public": public, + "collaborative": collaborative, + }..removeWhere((key, value) => value == null), + ); + } + + Future addTracks( + String playlistId, { + required List trackIds, + int? position, + }) async { + await hetuMetadataPlaylist.invoke( + "addTracks", + positionalArgs: [playlistId], + namedArgs: { + "trackIds": trackIds, + "position": position, + }..removeWhere((key, value) => value == null), + ); + } + + Future removeTracks( + String playlistId, { + required List trackIds, + }) async { + await hetuMetadataPlaylist.invoke( + "removeTracks", + positionalArgs: [playlistId], + namedArgs: { + "trackIds": trackIds, + }..removeWhere((key, value) => value == null), + ); + } + + Future save(String playlistId) async { + await hetuMetadataPlaylist.invoke( + "save", + positionalArgs: [playlistId], + ); + } + + Future unsave(String playlistId) async { + await hetuMetadataPlaylist.invoke( + "unsave", + positionalArgs: [playlistId], + ); + } + + Future deletePlaylist(String playlistId) async { + return await hetuMetadataPlaylist.invoke( + "deletePlaylist", + positionalArgs: [playlistId], + ); + } +} diff --git a/lib/services/metadata/endpoints/search.dart b/lib/services/metadata/endpoints/search.dart new file mode 100644 index 00000000..c2e14765 --- /dev/null +++ b/lib/services/metadata/endpoints/search.dart @@ -0,0 +1,160 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginSearchEndpoint { + final Hetu hetu; + MetadataPluginSearchEndpoint(this.hetu); + + HTInstance get hetuMetadataSearch => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("search") + as HTInstance; + + List get chips { + return (hetuMetadataSearch.memberGet("chips") as List).cast(); + } + + Future all(String query) async { + if (query.isEmpty) { + return SpotubeSearchResponseObject( + albums: [], + artists: [], + playlists: [], + tracks: [], + ); + } + + final raw = await hetuMetadataSearch.invoke( + "all", + positionalArgs: [query], + ) as Map; + + return SpotubeSearchResponseObject.fromJson(raw.cast()); + } + + Future> albums( + String query, { + int? limit, + int? offset, + }) async { + if (query.isEmpty) { + return SpotubePaginationResponseObject( + items: [], + total: 0, + limit: limit ?? 20, + hasMore: false, + nextOffset: null, + ); + } + + final raw = await hetuMetadataSearch.invoke( + "albums", + positionalArgs: [query], + namedArgs: { + "limit": limit, + "offset": offset, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (json) => SpotubeSimpleAlbumObject.fromJson(json.cast()), + ); + } + + Future> artists( + String query, { + int? limit, + int? offset, + }) async { + if (query.isEmpty) { + return SpotubePaginationResponseObject( + items: [], + total: 0, + limit: limit ?? 20, + hasMore: false, + nextOffset: null, + ); + } + + final raw = await hetuMetadataSearch.invoke( + "artists", + positionalArgs: [query], + namedArgs: { + "limit": limit, + "offset": offset, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (json) => SpotubeFullArtistObject.fromJson( + json.cast(), + ), + ); + } + + Future> + playlists( + String query, { + int? limit, + int? offset, + }) async { + if (query.isEmpty) { + return SpotubePaginationResponseObject( + items: [], + total: 0, + limit: limit ?? 20, + hasMore: false, + nextOffset: null, + ); + } + + final raw = await hetuMetadataSearch.invoke( + "playlists", + positionalArgs: [query], + namedArgs: { + "limit": limit, + "offset": offset, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject< + SpotubeSimplePlaylistObject>.fromJson( + raw.cast(), + (json) => SpotubeSimplePlaylistObject.fromJson( + json.cast(), + ), + ); + } + + Future> tracks( + String query, { + int? limit, + int? offset, + }) async { + if (query.isEmpty) { + return SpotubePaginationResponseObject( + items: [], + total: 0, + limit: limit ?? 20, + hasMore: false, + nextOffset: null, + ); + } + + final raw = await hetuMetadataSearch.invoke( + "tracks", + positionalArgs: [query], + namedArgs: { + "limit": limit, + "offset": offset, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (json) => SpotubeFullTrackObject.fromJson(json.cast()), + ); + } +} diff --git a/lib/services/metadata/endpoints/track.dart b/lib/services/metadata/endpoints/track.dart new file mode 100644 index 00000000..31535970 --- /dev/null +++ b/lib/services/metadata/endpoints/track.dart @@ -0,0 +1,44 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginTrackEndpoint { + final Hetu hetu; + MetadataPluginTrackEndpoint(this.hetu); + + HTInstance get hetuMetadataTrack => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("track") + as HTInstance; + + Future getTrack(String id) async { + final raw = + await hetuMetadataTrack.invoke("getTrack", positionalArgs: [id]) as Map; + + return SpotubeFullTrackObject.fromJson( + raw.cast(), + ); + } + + Future save(List ids) async { + await hetuMetadataTrack.invoke("save", positionalArgs: [ids]); + } + + Future unsave(List ids) async { + await hetuMetadataTrack.invoke("unsave", positionalArgs: [ids]); + } + + Future> radio(String id) async { + final result = await hetuMetadataTrack.invoke( + "radio", + positionalArgs: [id], + ); + + return (result as List) + .map( + (e) => SpotubeFullTrackObject.fromJson( + (e as Map).cast(), + ), + ) + .toList(); + } +} diff --git a/lib/services/metadata/endpoints/user.dart b/lib/services/metadata/endpoints/user.dart new file mode 100644 index 00000000..3c8f0e42 --- /dev/null +++ b/lib/services/metadata/endpoints/user.dart @@ -0,0 +1,132 @@ +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_script/values.dart'; +import 'package:spotube/models/metadata/metadata.dart'; + +class MetadataPluginUserEndpoint { + final Hetu hetu; + MetadataPluginUserEndpoint(this.hetu); + + HTInstance get hetuMetadataUser => + (hetu.fetch("metadataPlugin") as HTInstance).memberGet("user") + as HTInstance; + + Future me() async { + final raw = await hetuMetadataUser.invoke("me") as Map; + + return SpotubeUserObject.fromJson( + raw.cast(), + ); + } + + Future> savedTracks({ + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataUser.invoke( + "savedTracks", + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => + SpotubeFullTrackObject.fromJson(json.cast()), + ); + } + + Future> + savedPlaylists({ + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataUser.invoke( + "savedPlaylists", + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject< + SpotubeSimplePlaylistObject>.fromJson( + raw.cast(), + (Map json) => + SpotubeSimplePlaylistObject.fromJson(json.cast()), + ); + } + + Future> + savedAlbums({ + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataUser.invoke( + "savedAlbums", + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => + SpotubeSimpleAlbumObject.fromJson(json.cast()), + ); + } + + Future> + savedArtists({ + int? offset, + int? limit, + }) async { + final raw = await hetuMetadataUser.invoke( + "savedArtists", + namedArgs: { + "offset": offset, + "limit": limit, + }..removeWhere((key, value) => value == null), + ) as Map; + + return SpotubePaginationResponseObject.fromJson( + raw.cast(), + (Map json) => + SpotubeFullArtistObject.fromJson(json.cast()), + ); + } + + Future isSavedPlaylist(String playlistId) async { + return await hetuMetadataUser.invoke( + "isSavedPlaylist", + positionalArgs: [playlistId], + ) as bool; + } + + Future> isSavedTracks(List ids) async { + final values = await hetuMetadataUser.invoke( + "isSavedTracks", + positionalArgs: [ids], + ); + return (values as List).cast(); + } + + Future> isSavedAlbums(List ids) async { + final values = await hetuMetadataUser.invoke( + "isSavedAlbums", + positionalArgs: [ids], + ) as List; + return values.cast(); + } + + Future> isSavedArtists(List ids) async { + final values = await hetuMetadataUser.invoke( + "isSavedArtists", + positionalArgs: [ids], + ) as List; + + return values.cast(); + } +} diff --git a/lib/services/metadata/errors/exceptions.dart b/lib/services/metadata/errors/exceptions.dart new file mode 100644 index 00000000..5bb5ac57 --- /dev/null +++ b/lib/services/metadata/errors/exceptions.dart @@ -0,0 +1,85 @@ +enum MetadataPluginErrorCode { + pluginApiVersionMismatch, + invalidPluginConfiguration, + failedToGetReleaseInfo, + noReleasesFound, + assetUrlNotFound, + pluginConfigJsonNotFound, + unsupportedPluginDownloadWebsite, + pluginDownloadFailed, + duplicatePlugin, + pluginByteCodeFileNotFound, + noDefaultMetadataPlugin, + noDefaultAudiSourcePlugin, +} + +class MetadataPluginException implements Exception { + final String message; + final MetadataPluginErrorCode errorCode; + + MetadataPluginException._(this.message, {required this.errorCode}); + MetadataPluginException.pluginApiVersionMismatch() + : this._( + 'Plugin API version mismatch', + errorCode: MetadataPluginErrorCode.pluginApiVersionMismatch, + ); + MetadataPluginException.invalidPluginConfiguration() + : this._( + 'Invalid plugin configuration', + errorCode: MetadataPluginErrorCode.invalidPluginConfiguration, + ); + MetadataPluginException.failedToGetRelease() + : this._( + 'Failed to get release information', + errorCode: MetadataPluginErrorCode.failedToGetReleaseInfo, + ); + MetadataPluginException.noReleasesFound() + : this._( + 'No releases found for the plugin', + errorCode: MetadataPluginErrorCode.noReleasesFound, + ); + + MetadataPluginException.assetUrlNotFound() + : this._( + 'No asset URL found for the plugin release', + errorCode: MetadataPluginErrorCode.assetUrlNotFound, + ); + MetadataPluginException.pluginConfigJsonNotFound() + : this._( + 'Plugin configuration JSON, plugin.json file not found', + errorCode: MetadataPluginErrorCode.pluginConfigJsonNotFound, + ); + MetadataPluginException.unsupportedPluginDownloadWebsite() + : this._( + 'Unsupported plugin download website. Please use GitHub or Codeberg.', + errorCode: MetadataPluginErrorCode.unsupportedPluginDownloadWebsite, + ); + MetadataPluginException.pluginDownloadFailed() + : this._( + 'Failed to download the plugin. Please check your internet connection or try again later.', + errorCode: MetadataPluginErrorCode.pluginDownloadFailed, + ); + MetadataPluginException.duplicatePlugin() + : this._( + 'Same plugin already exists with the same name and version.', + errorCode: MetadataPluginErrorCode.duplicatePlugin, + ); + MetadataPluginException.pluginByteCodeFileNotFound() + : this._( + 'Plugin byte code file, plugin.out not found. Please ensure the plugin is correctly packaged.', + errorCode: MetadataPluginErrorCode.pluginByteCodeFileNotFound, + ); + MetadataPluginException.noDefaultMetadataPlugin() + : this._( + 'No default metadata plugin is set. Please set a default plugin in the settings.', + errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin, + ); + MetadataPluginException.noDefaultAudioSourcePlugin() + : this._( + 'No default audio source plugin is set. Please set a default plugin in the settings.', + errorCode: MetadataPluginErrorCode.noDefaultAudiSourcePlugin, + ); + + @override + String toString() => 'MetadataPluginException: $message'; +} diff --git a/lib/services/metadata/metadata.dart b/lib/services/metadata/metadata.dart new file mode 100644 index 00000000..5860e0d6 --- /dev/null +++ b/lib/services/metadata/metadata.dart @@ -0,0 +1,181 @@ +import 'dart:typed_data'; + +import 'package:auto_route/auto_route.dart'; +import 'package:hetu_otp_util/hetu_otp_util.dart'; +import 'package:hetu_script/hetu_script.dart'; +import 'package:hetu_spotube_plugin/hetu_spotube_plugin.dart' as spotube_plugin; +import 'package:hetu_spotube_plugin/hetu_spotube_plugin.dart' + hide YouTubeEngine; +import 'package:hetu_std/hetu_std.dart'; +import 'package:pub_semver/pub_semver.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:spotube/collections/routes.dart'; +import 'package:spotube/collections/routes.gr.dart'; +import 'package:spotube/components/titlebar/titlebar.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/services/metadata/apis/localstorage.dart'; +import 'package:spotube/services/metadata/endpoints/album.dart'; +import 'package:spotube/services/metadata/endpoints/artist.dart'; +import 'package:spotube/services/metadata/endpoints/audio_source.dart'; +import 'package:spotube/services/metadata/endpoints/auth.dart'; +import 'package:spotube/services/metadata/endpoints/browse.dart'; +import 'package:spotube/services/metadata/endpoints/playlist.dart'; +import 'package:spotube/services/metadata/endpoints/search.dart'; +import 'package:spotube/services/metadata/endpoints/track.dart'; +import 'package:spotube/services/metadata/endpoints/core.dart'; +import 'package:spotube/services/metadata/endpoints/user.dart'; +import 'package:spotube/services/youtube_engine/youtube_engine.dart'; + +const defaultMetadataLimit = "20"; + +class MetadataPlugin { + static final pluginApiVersion = Version.parse("2.0.0"); + + static Future create( + YouTubeEngine youtubeEngine, + PluginConfiguration config, + Uint8List byteCode, + ) async { + final sharedPreferences = await SharedPreferences.getInstance(); + BuildContext? pageContext; + + final hetu = Hetu(); + hetu.init(); + + HetuStdLoader.loadBindings(hetu); + HetuSpotubePluginLoader.loadBindings( + hetu, + localStorageImpl: SharedPreferencesLocalStorage( + sharedPreferences, + config.slug, + ), + onNavigatorPush: (route) { + return rootNavigatorKey.currentContext?.router + .pushWidget(Builder(builder: (context) { + pageContext = context; + return Scaffold( + headers: const [ + TitleBar( + automaticallyImplyLeading: true, + ) + ], + child: route, + ); + })); + }, + onNavigatorPop: () { + pageContext?.maybePop(); + }, + onShowForm: (title, fields) async { + if (rootNavigatorKey.currentContext == null) { + return []; + } + + return await rootNavigatorKey.currentContext!.router + .push>?>( + SettingsMetadataProviderFormRoute( + title: title, + fields: + fields.map((e) => MetadataFormFieldObject.fromJson(e)).toList(), + ), + ); + }, + createYoutubeEngine: () { + return spotube_plugin.YouTubeEngine( + search: (query) async { + final result = await youtubeEngine.searchVideos(query); + return result + .map((video) => { + 'id': video.id.value, + 'title': video.title, + 'author': video.author, + 'duration': video.duration?.inSeconds, + 'description': video.description, + 'uploadDate': video.uploadDate?.toIso8601String(), + 'viewCount': video.engagement.viewCount, + 'likeCount': video.engagement.likeCount, + 'isLive': video.isLive, + }) + .toList(); + }, + getVideo: (videoId) async { + final video = await youtubeEngine.getVideo(videoId); + return { + 'id': video.id.value, + 'title': video.title, + 'author': video.author, + 'duration': video.duration?.inSeconds, + 'description': video.description, + 'uploadDate': video.uploadDate?.toIso8601String(), + 'viewCount': video.engagement.viewCount, + 'likeCount': video.engagement.likeCount, + 'isLive': video.isLive, + }; + }, + streamManifest: (videoId) { + return youtubeEngine.getStreamManifest(videoId).then( + (manifest) { + final streams = manifest.audioOnly + .map( + (stream) => { + 'url': stream.url.toString(), + 'quality': stream.qualityLabel, + 'bitrate': stream.bitrate.bitsPerSecond, + 'container': stream.container.name, + 'videoId': stream.videoId, + }, + ) + .toList(); + return streams; + }, + ); + }, + ); + }, + ); + + await HetuStdLoader.loadBytecodeFlutter(hetu); + await HetuOtpUtilLoader.loadBytecodeFlutter(hetu); + await HetuSpotubePluginLoader.loadBytecodeFlutter(hetu); + + hetu.loadBytecode(bytes: byteCode, moduleName: "plugin"); + hetu.eval(""" + import "module:plugin" as plugin + + var Plugin = plugin.${config.entryPoint} + + var metadataPlugin = Plugin() + """); + + return MetadataPlugin._(hetu); + } + + final Hetu hetu; + + late final MetadataAuthEndpoint auth; + + late final MetadataPluginAudioSourceEndpoint audioSource; + late final MetadataPluginAlbumEndpoint album; + late final MetadataPluginArtistEndpoint artist; + late final MetadataPluginBrowseEndpoint browse; + late final MetadataPluginSearchEndpoint search; + late final MetadataPluginPlaylistEndpoint playlist; + late final MetadataPluginTrackEndpoint track; + late final MetadataPluginUserEndpoint user; + late final MetadataPluginCore core; + + MetadataPlugin._(this.hetu) { + auth = MetadataAuthEndpoint(hetu); + + audioSource = MetadataPluginAudioSourceEndpoint(hetu); + artist = MetadataPluginArtistEndpoint(hetu); + album = MetadataPluginAlbumEndpoint(hetu); + browse = MetadataPluginBrowseEndpoint(hetu); + search = MetadataPluginSearchEndpoint(hetu); + playlist = MetadataPluginPlaylistEndpoint(hetu); + track = MetadataPluginTrackEndpoint(hetu); + user = MetadataPluginUserEndpoint(hetu); + core = MetadataPluginCore(hetu); + } +} diff --git a/lib/services/song_link/model.dart b/lib/services/song_link/model.dart deleted file mode 100644 index ae9d3833..00000000 --- a/lib/services/song_link/model.dart +++ /dev/null @@ -1,19 +0,0 @@ -part of './song_link.dart'; - -@freezed -class SongLink with _$SongLink { - const factory SongLink({ - required String displayName, - required String linkId, - required String platform, - required bool show, - required String? uniqueId, - required String? country, - required String? url, - required String? nativeAppUriMobile, - required String? nativeAppUriDesktop, - }) = _SongLink; - - factory SongLink.fromJson(Map json) => - _$SongLinkFromJson(json); -} diff --git a/lib/services/song_link/song_link.dart b/lib/services/song_link/song_link.dart deleted file mode 100644 index e3cffa52..00000000 --- a/lib/services/song_link/song_link.dart +++ /dev/null @@ -1,54 +0,0 @@ -library song_link; - -import 'dart:convert'; - -import 'package:spotube/services/logger/logger.dart'; -import 'package:dio/dio.dart'; -import 'package:freezed_annotation/freezed_annotation.dart'; -import 'package:html/parser.dart'; - -part 'model.dart'; - -part 'song_link.freezed.dart'; -part 'song_link.g.dart'; - -abstract class SongLinkService { - static final dio = Dio(); - static Future> links(String spotifyId) async { - try { - final res = await dio.get( - "https://song.link/s/$spotifyId", - options: Options( - headers: { - "Accept": - "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8", - "User-Agent": - "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36" - }, - responseType: ResponseType.plain, - ), - ); - - final document = parse(res.data); - - final script = document.getElementById("__NEXT_DATA__")?.text; - - if (script == null) { - return []; - } - - final pageProps = jsonDecode(script) as Map; - final songLinks = pageProps["props"]?["pageProps"]?["pageData"] - ?["sections"] - ?.firstWhere( - (section) => section?["sectionId"] == "section|auto|links|listen", - )?["links"] as List?; - - return songLinks?.map((link) => SongLink.fromJson(link)).toList() ?? - []; - } catch (e, stackTrace) { - AppLogger.reportError(e, stackTrace); - return []; - } - } -} diff --git a/lib/services/song_link/song_link.freezed.dart b/lib/services/song_link/song_link.freezed.dart deleted file mode 100644 index c704cde3..00000000 --- a/lib/services/song_link/song_link.freezed.dart +++ /dev/null @@ -1,333 +0,0 @@ -// coverage:ignore-file -// GENERATED CODE - DO NOT MODIFY BY HAND -// ignore_for_file: type=lint -// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark - -part of 'song_link.dart'; - -// ************************************************************************** -// FreezedGenerator -// ************************************************************************** - -T _$identity(T value) => value; - -final _privateConstructorUsedError = UnsupportedError( - 'It seems like you constructed your class using `MyClass._()`. This constructor is only meant to be used by freezed and you are not supposed to need it nor use it.\nPlease check the documentation here for more information: https://github.com/rrousselGit/freezed#adding-getters-and-methods-to-our-models'); - -SongLink _$SongLinkFromJson(Map json) { - return _SongLink.fromJson(json); -} - -/// @nodoc -mixin _$SongLink { - String get displayName => throw _privateConstructorUsedError; - String get linkId => throw _privateConstructorUsedError; - String get platform => throw _privateConstructorUsedError; - bool get show => throw _privateConstructorUsedError; - String? get uniqueId => throw _privateConstructorUsedError; - String? get country => throw _privateConstructorUsedError; - String? get url => throw _privateConstructorUsedError; - String? get nativeAppUriMobile => throw _privateConstructorUsedError; - String? get nativeAppUriDesktop => throw _privateConstructorUsedError; - - /// Serializes this SongLink to a JSON map. - Map toJson() => throw _privateConstructorUsedError; - - /// Create a copy of SongLink - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - $SongLinkCopyWith get copyWith => - throw _privateConstructorUsedError; -} - -/// @nodoc -abstract class $SongLinkCopyWith<$Res> { - factory $SongLinkCopyWith(SongLink value, $Res Function(SongLink) then) = - _$SongLinkCopyWithImpl<$Res, SongLink>; - @useResult - $Res call( - {String displayName, - String linkId, - String platform, - bool show, - String? uniqueId, - String? country, - String? url, - String? nativeAppUriMobile, - String? nativeAppUriDesktop}); -} - -/// @nodoc -class _$SongLinkCopyWithImpl<$Res, $Val extends SongLink> - implements $SongLinkCopyWith<$Res> { - _$SongLinkCopyWithImpl(this._value, this._then); - - // ignore: unused_field - final $Val _value; - // ignore: unused_field - final $Res Function($Val) _then; - - /// Create a copy of SongLink - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? displayName = null, - Object? linkId = null, - Object? platform = null, - Object? show = null, - Object? uniqueId = freezed, - Object? country = freezed, - Object? url = freezed, - Object? nativeAppUriMobile = freezed, - Object? nativeAppUriDesktop = freezed, - }) { - return _then(_value.copyWith( - displayName: null == displayName - ? _value.displayName - : displayName // ignore: cast_nullable_to_non_nullable - as String, - linkId: null == linkId - ? _value.linkId - : linkId // ignore: cast_nullable_to_non_nullable - as String, - platform: null == platform - ? _value.platform - : platform // ignore: cast_nullable_to_non_nullable - as String, - show: null == show - ? _value.show - : show // ignore: cast_nullable_to_non_nullable - as bool, - uniqueId: freezed == uniqueId - ? _value.uniqueId - : uniqueId // ignore: cast_nullable_to_non_nullable - as String?, - country: freezed == country - ? _value.country - : country // ignore: cast_nullable_to_non_nullable - as String?, - url: freezed == url - ? _value.url - : url // ignore: cast_nullable_to_non_nullable - as String?, - nativeAppUriMobile: freezed == nativeAppUriMobile - ? _value.nativeAppUriMobile - : nativeAppUriMobile // ignore: cast_nullable_to_non_nullable - as String?, - nativeAppUriDesktop: freezed == nativeAppUriDesktop - ? _value.nativeAppUriDesktop - : nativeAppUriDesktop // ignore: cast_nullable_to_non_nullable - as String?, - ) as $Val); - } -} - -/// @nodoc -abstract class _$$SongLinkImplCopyWith<$Res> - implements $SongLinkCopyWith<$Res> { - factory _$$SongLinkImplCopyWith( - _$SongLinkImpl value, $Res Function(_$SongLinkImpl) then) = - __$$SongLinkImplCopyWithImpl<$Res>; - @override - @useResult - $Res call( - {String displayName, - String linkId, - String platform, - bool show, - String? uniqueId, - String? country, - String? url, - String? nativeAppUriMobile, - String? nativeAppUriDesktop}); -} - -/// @nodoc -class __$$SongLinkImplCopyWithImpl<$Res> - extends _$SongLinkCopyWithImpl<$Res, _$SongLinkImpl> - implements _$$SongLinkImplCopyWith<$Res> { - __$$SongLinkImplCopyWithImpl( - _$SongLinkImpl _value, $Res Function(_$SongLinkImpl) _then) - : super(_value, _then); - - /// Create a copy of SongLink - /// with the given fields replaced by the non-null parameter values. - @pragma('vm:prefer-inline') - @override - $Res call({ - Object? displayName = null, - Object? linkId = null, - Object? platform = null, - Object? show = null, - Object? uniqueId = freezed, - Object? country = freezed, - Object? url = freezed, - Object? nativeAppUriMobile = freezed, - Object? nativeAppUriDesktop = freezed, - }) { - return _then(_$SongLinkImpl( - displayName: null == displayName - ? _value.displayName - : displayName // ignore: cast_nullable_to_non_nullable - as String, - linkId: null == linkId - ? _value.linkId - : linkId // ignore: cast_nullable_to_non_nullable - as String, - platform: null == platform - ? _value.platform - : platform // ignore: cast_nullable_to_non_nullable - as String, - show: null == show - ? _value.show - : show // ignore: cast_nullable_to_non_nullable - as bool, - uniqueId: freezed == uniqueId - ? _value.uniqueId - : uniqueId // ignore: cast_nullable_to_non_nullable - as String?, - country: freezed == country - ? _value.country - : country // ignore: cast_nullable_to_non_nullable - as String?, - url: freezed == url - ? _value.url - : url // ignore: cast_nullable_to_non_nullable - as String?, - nativeAppUriMobile: freezed == nativeAppUriMobile - ? _value.nativeAppUriMobile - : nativeAppUriMobile // ignore: cast_nullable_to_non_nullable - as String?, - nativeAppUriDesktop: freezed == nativeAppUriDesktop - ? _value.nativeAppUriDesktop - : nativeAppUriDesktop // ignore: cast_nullable_to_non_nullable - as String?, - )); - } -} - -/// @nodoc -@JsonSerializable() -class _$SongLinkImpl implements _SongLink { - const _$SongLinkImpl( - {required this.displayName, - required this.linkId, - required this.platform, - required this.show, - required this.uniqueId, - required this.country, - required this.url, - required this.nativeAppUriMobile, - required this.nativeAppUriDesktop}); - - factory _$SongLinkImpl.fromJson(Map json) => - _$$SongLinkImplFromJson(json); - - @override - final String displayName; - @override - final String linkId; - @override - final String platform; - @override - final bool show; - @override - final String? uniqueId; - @override - final String? country; - @override - final String? url; - @override - final String? nativeAppUriMobile; - @override - final String? nativeAppUriDesktop; - - @override - String toString() { - return 'SongLink(displayName: $displayName, linkId: $linkId, platform: $platform, show: $show, uniqueId: $uniqueId, country: $country, url: $url, nativeAppUriMobile: $nativeAppUriMobile, nativeAppUriDesktop: $nativeAppUriDesktop)'; - } - - @override - bool operator ==(Object other) { - return identical(this, other) || - (other.runtimeType == runtimeType && - other is _$SongLinkImpl && - (identical(other.displayName, displayName) || - other.displayName == displayName) && - (identical(other.linkId, linkId) || other.linkId == linkId) && - (identical(other.platform, platform) || - other.platform == platform) && - (identical(other.show, show) || other.show == show) && - (identical(other.uniqueId, uniqueId) || - other.uniqueId == uniqueId) && - (identical(other.country, country) || other.country == country) && - (identical(other.url, url) || other.url == url) && - (identical(other.nativeAppUriMobile, nativeAppUriMobile) || - other.nativeAppUriMobile == nativeAppUriMobile) && - (identical(other.nativeAppUriDesktop, nativeAppUriDesktop) || - other.nativeAppUriDesktop == nativeAppUriDesktop)); - } - - @JsonKey(includeFromJson: false, includeToJson: false) - @override - int get hashCode => Object.hash(runtimeType, displayName, linkId, platform, - show, uniqueId, country, url, nativeAppUriMobile, nativeAppUriDesktop); - - /// Create a copy of SongLink - /// with the given fields replaced by the non-null parameter values. - @JsonKey(includeFromJson: false, includeToJson: false) - @override - @pragma('vm:prefer-inline') - _$$SongLinkImplCopyWith<_$SongLinkImpl> get copyWith => - __$$SongLinkImplCopyWithImpl<_$SongLinkImpl>(this, _$identity); - - @override - Map toJson() { - return _$$SongLinkImplToJson( - this, - ); - } -} - -abstract class _SongLink implements SongLink { - const factory _SongLink( - {required final String displayName, - required final String linkId, - required final String platform, - required final bool show, - required final String? uniqueId, - required final String? country, - required final String? url, - required final String? nativeAppUriMobile, - required final String? nativeAppUriDesktop}) = _$SongLinkImpl; - - factory _SongLink.fromJson(Map json) = - _$SongLinkImpl.fromJson; - - @override - String get displayName; - @override - String get linkId; - @override - String get platform; - @override - bool get show; - @override - String? get uniqueId; - @override - String? get country; - @override - String? get url; - @override - String? get nativeAppUriMobile; - @override - String? get nativeAppUriDesktop; - - /// Create a copy of SongLink - /// with the given fields replaced by the non-null parameter values. - @override - @JsonKey(includeFromJson: false, includeToJson: false) - _$$SongLinkImplCopyWith<_$SongLinkImpl> get copyWith => - throw _privateConstructorUsedError; -} diff --git a/lib/services/song_link/song_link.g.dart b/lib/services/song_link/song_link.g.dart deleted file mode 100644 index 7658a74c..00000000 --- a/lib/services/song_link/song_link.g.dart +++ /dev/null @@ -1,32 +0,0 @@ -// GENERATED CODE - DO NOT MODIFY BY HAND - -part of 'song_link.dart'; - -// ************************************************************************** -// JsonSerializableGenerator -// ************************************************************************** - -_$SongLinkImpl _$$SongLinkImplFromJson(Map json) => _$SongLinkImpl( - displayName: json['displayName'] as String, - linkId: json['linkId'] as String, - platform: json['platform'] as String, - show: json['show'] as bool, - uniqueId: json['uniqueId'] as String?, - country: json['country'] as String?, - url: json['url'] as String?, - nativeAppUriMobile: json['nativeAppUriMobile'] as String?, - nativeAppUriDesktop: json['nativeAppUriDesktop'] as String?, - ); - -Map _$$SongLinkImplToJson(_$SongLinkImpl instance) => - { - 'displayName': instance.displayName, - 'linkId': instance.linkId, - 'platform': instance.platform, - 'show': instance.show, - 'uniqueId': instance.uniqueId, - 'country': instance.country, - 'url': instance.url, - 'nativeAppUriMobile': instance.nativeAppUriMobile, - 'nativeAppUriDesktop': instance.nativeAppUriDesktop, - }; diff --git a/lib/services/sourced_track/enums.dart b/lib/services/sourced_track/enums.dart deleted file mode 100644 index e47ee6bd..00000000 --- a/lib/services/sourced_track/enums.dart +++ /dev/null @@ -1,18 +0,0 @@ -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/source_map.dart'; - -enum SourceCodecs { - m4a._("M4a (Best for downloaded music)"), - weba._("WebA (Best for streamed music)\nDoesn't support audio metadata"); - - final String label; - const SourceCodecs._(this.label); -} - -enum SourceQualities { - high, - medium, - low, -} - -typedef SiblingType = ({T info, SourceMap? source}); diff --git a/lib/services/sourced_track/exceptions.dart b/lib/services/sourced_track/exceptions.dart index 85bc5b27..4817c9fb 100644 --- a/lib/services/sourced_track/exceptions.dart +++ b/lib/services/sourced_track/exceptions.dart @@ -1,12 +1,12 @@ -import 'package:spotify/spotify.dart'; +import 'package:spotube/models/metadata/metadata.dart'; class TrackNotFoundError extends Error { - final Track track; + final SpotubeTrackObject track; TrackNotFoundError(this.track); @override String toString() { - return '[TrackNotFoundError] ${track.name} - ${track.artists?.map((e) => e.name).join(", ")}'; + return '[TrackNotFoundError] ${track.name} - ${track.artists.join(", ")}'; } } diff --git a/lib/services/sourced_track/models/source_info.dart b/lib/services/sourced_track/models/source_info.dart deleted file mode 100644 index 4ba90355..00000000 --- a/lib/services/sourced_track/models/source_info.dart +++ /dev/null @@ -1,33 +0,0 @@ -import 'package:json_annotation/json_annotation.dart'; - -part 'source_info.g.dart'; - -@JsonSerializable() -class SourceInfo { - final String id; - final String title; - final String artist; - final String artistUrl; - final String? album; - - final String thumbnail; - final String pageUrl; - - final Duration duration; - - SourceInfo({ - required this.id, - required this.title, - required this.artist, - required this.thumbnail, - required this.pageUrl, - required this.duration, - required this.artistUrl, - this.album, - }); - - factory SourceInfo.fromJson(Map json) => - _$SourceInfoFromJson(json); - - Map toJson() => _$SourceInfoToJson(this); -} diff --git a/lib/services/sourced_track/models/source_info.g.dart b/lib/services/sourced_track/models/source_info.g.dart deleted file mode 100644 index 54671f63..00000000 --- a/lib/services/sourced_track/models/source_info.g.dart +++ /dev/null @@ -1,30 +0,0 @@ -// GENERATED CODE - DO NOT MODIFY BY HAND - -part of 'source_info.dart'; - -// ************************************************************************** -// JsonSerializableGenerator -// ************************************************************************** - -SourceInfo _$SourceInfoFromJson(Map json) => SourceInfo( - id: json['id'] as String, - title: json['title'] as String, - artist: json['artist'] as String, - thumbnail: json['thumbnail'] as String, - pageUrl: json['pageUrl'] as String, - duration: Duration(microseconds: (json['duration'] as num).toInt()), - artistUrl: json['artistUrl'] as String, - album: json['album'] as String?, - ); - -Map _$SourceInfoToJson(SourceInfo instance) => - { - 'id': instance.id, - 'title': instance.title, - 'artist': instance.artist, - 'artistUrl': instance.artistUrl, - 'album': instance.album, - 'thumbnail': instance.thumbnail, - 'pageUrl': instance.pageUrl, - 'duration': instance.duration.inMicroseconds, - }; diff --git a/lib/services/sourced_track/models/source_map.dart b/lib/services/sourced_track/models/source_map.dart deleted file mode 100644 index f99f95e4..00000000 --- a/lib/services/sourced_track/models/source_map.dart +++ /dev/null @@ -1,58 +0,0 @@ -import 'package:json_annotation/json_annotation.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; - -part 'source_map.g.dart'; - -@JsonSerializable() -class SourceQualityMap { - final String high; - final String medium; - final String low; - - const SourceQualityMap({ - required this.high, - required this.medium, - required this.low, - }); - - factory SourceQualityMap.fromJson(Map json) => - _$SourceQualityMapFromJson(json); - - Map toJson() => _$SourceQualityMapToJson(this); - - operator [](SourceQualities key) { - switch (key) { - case SourceQualities.high: - return high; - case SourceQualities.medium: - return medium; - case SourceQualities.low: - return low; - } - } -} - -@JsonSerializable() -class SourceMap { - final SourceQualityMap? weba; - final SourceQualityMap? m4a; - - const SourceMap({ - this.weba, - this.m4a, - }); - - factory SourceMap.fromJson(Map json) => - _$SourceMapFromJson(json); - - Map toJson() => _$SourceMapToJson(this); - - operator [](SourceCodecs key) { - switch (key) { - case SourceCodecs.weba: - return weba; - case SourceCodecs.m4a: - return m4a; - } - } -} diff --git a/lib/services/sourced_track/models/source_map.g.dart b/lib/services/sourced_track/models/source_map.g.dart deleted file mode 100644 index a581cc67..00000000 --- a/lib/services/sourced_track/models/source_map.g.dart +++ /dev/null @@ -1,36 +0,0 @@ -// GENERATED CODE - DO NOT MODIFY BY HAND - -part of 'source_map.dart'; - -// ************************************************************************** -// JsonSerializableGenerator -// ************************************************************************** - -SourceQualityMap _$SourceQualityMapFromJson(Map json) => SourceQualityMap( - high: json['high'] as String, - medium: json['medium'] as String, - low: json['low'] as String, - ); - -Map _$SourceQualityMapToJson(SourceQualityMap instance) => - { - 'high': instance.high, - 'medium': instance.medium, - 'low': instance.low, - }; - -SourceMap _$SourceMapFromJson(Map json) => SourceMap( - weba: json['weba'] == null - ? null - : SourceQualityMap.fromJson( - Map.from(json['weba'] as Map)), - m4a: json['m4a'] == null - ? null - : SourceQualityMap.fromJson( - Map.from(json['m4a'] as Map)), - ); - -Map _$SourceMapToJson(SourceMap instance) => { - 'weba': instance.weba?.toJson(), - 'm4a': instance.m4a?.toJson(), - }; diff --git a/lib/services/sourced_track/models/video_info.dart b/lib/services/sourced_track/models/video_info.dart deleted file mode 100644 index e3452c61..00000000 --- a/lib/services/sourced_track/models/video_info.dart +++ /dev/null @@ -1,136 +0,0 @@ -import 'package:invidious/invidious.dart'; -import 'package:piped_client/piped_client.dart'; -import 'package:spotube/models/database/database.dart'; - -import 'package:youtube_explode_dart/youtube_explode_dart.dart'; - -class YoutubeVideoInfo { - final SearchMode searchMode; - final String title; - final Duration duration; - final String thumbnailUrl; - final String id; - final int likes; - final int dislikes; - final int views; - final String channelName; - final String channelId; - final DateTime publishedAt; - - YoutubeVideoInfo({ - required this.searchMode, - required this.title, - required this.duration, - required this.thumbnailUrl, - required this.id, - required this.likes, - required this.dislikes, - required this.views, - required this.channelName, - required this.publishedAt, - required this.channelId, - }); - - YoutubeVideoInfo.fromJson(Map json) - : title = json['title'], - searchMode = SearchMode.fromString(json['searchMode']), - duration = Duration(seconds: json['duration']), - thumbnailUrl = json['thumbnailUrl'], - id = json['id'], - likes = json['likes'], - dislikes = json['dislikes'], - views = json['views'], - channelName = json['channelName'], - channelId = json['channelId'], - publishedAt = DateTime.tryParse(json['publishedAt']) ?? DateTime.now(); - - Map toJson() => { - 'title': title, - 'duration': duration.inSeconds, - 'thumbnailUrl': thumbnailUrl, - 'id': id, - 'likes': likes, - 'dislikes': dislikes, - 'views': views, - 'channelName': channelName, - 'channelId': channelId, - 'publishedAt': publishedAt.toIso8601String(), - 'searchMode': searchMode.name, - }; - - factory YoutubeVideoInfo.fromVideo(Video video) { - return YoutubeVideoInfo( - searchMode: SearchMode.youtube, - title: video.title, - duration: video.duration ?? Duration.zero, - thumbnailUrl: video.thumbnails.mediumResUrl, - id: video.id.value, - likes: video.engagement.likeCount ?? 0, - dislikes: video.engagement.dislikeCount ?? 0, - views: video.engagement.viewCount, - channelName: video.author, - channelId: '/c/${video.channelId.value}', - publishedAt: video.uploadDate ?? DateTime(2003, 9, 9), - ); - } - - factory YoutubeVideoInfo.fromSearchItemStream( - PipedSearchItemStream searchItem, - SearchMode searchMode, - ) { - return YoutubeVideoInfo( - searchMode: searchMode, - title: searchItem.title, - duration: searchItem.duration, - thumbnailUrl: searchItem.thumbnail, - id: searchItem.id, - likes: 0, - dislikes: 0, - views: searchItem.views, - channelName: searchItem.uploaderName, - channelId: searchItem.uploaderUrl ?? "", - publishedAt: searchItem.uploadedDate != null - ? DateTime.tryParse(searchItem.uploadedDate!) ?? DateTime(2003, 9, 9) - : DateTime(2003, 9, 9), - ); - } - - factory YoutubeVideoInfo.fromStreamResponse( - PipedStreamResponse stream, SearchMode searchMode) { - return YoutubeVideoInfo( - searchMode: searchMode, - title: stream.title, - duration: stream.duration, - thumbnailUrl: stream.thumbnailUrl, - id: stream.id, - likes: stream.likes, - dislikes: stream.dislikes, - views: stream.views, - channelName: stream.uploader, - publishedAt: stream.uploadedDate != null - ? DateTime.tryParse(stream.uploadedDate!) ?? DateTime(2003, 9, 9) - : DateTime(2003, 9, 9), - channelId: stream.uploaderUrl, - ); - } - - factory YoutubeVideoInfo.fromSearchResponse( - InvidiousSearchResponseVideo searchResponse, - SearchMode searchMode, - ) { - return YoutubeVideoInfo( - searchMode: searchMode, - title: searchResponse.title, - duration: Duration(seconds: searchResponse.lengthSeconds), - thumbnailUrl: searchResponse.videoThumbnails.first.url, - id: searchResponse.videoId, - likes: 0, - dislikes: 0, - views: searchResponse.viewCount, - channelName: searchResponse.author, - channelId: searchResponse.authorId, - publishedAt: - DateTime.fromMillisecondsSinceEpoch(searchResponse.published * 1000), - ); - } -} diff --git a/lib/services/sourced_track/sourced_track.dart b/lib/services/sourced_track/sourced_track.dart index 38f01498..385e5be6 100644 --- a/lib/services/sourced_track/sourced_track.dart +++ b/lib/services/sourced_track/sourced_track.dart @@ -1,263 +1,374 @@ -import 'dart:io'; +import 'dart:convert'; -import 'package:http/http.dart'; import 'package:collection/collection.dart'; import 'package:dio/dio.dart'; +import 'package:drift/drift.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; +import 'package:spotube/models/metadata/metadata.dart'; +import 'package:spotube/models/playback/track_sources.dart'; +import 'package:spotube/provider/database/database.dart'; +import 'package:spotube/provider/metadata_plugin/audio_source/quality_presets.dart'; +import 'package:spotube/provider/metadata_plugin/metadata_plugin_provider.dart'; +import 'package:spotube/services/dio/dio.dart'; +import 'package:spotube/services/logger/logger.dart'; +import 'package:spotube/services/metadata/errors/exceptions.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; import 'package:spotube/services/sourced_track/exceptions.dart'; -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/source_map.dart'; -import 'package:spotube/services/sourced_track/sources/invidious.dart'; -import 'package:spotube/services/sourced_track/sources/jiosaavn.dart'; -import 'package:spotube/services/sourced_track/sources/piped.dart'; -import 'package:spotube/services/sourced_track/sources/youtube.dart'; import 'package:spotube/utils/service_utils.dart'; -import 'package:youtube_explode_dart/youtube_explode_dart.dart'; -abstract class SourcedTrack extends Track { - final SourceMap source; - final List siblings; - final SourceInfo sourceInfo; +final officialMusicRegex = RegExp( + r"official\s(video|audio|music\svideo|lyric\svideo|visualizer)", + caseSensitive: false, +); + +class SourcedTrack extends BasicSourcedTrack { final Ref ref; SourcedTrack({ required this.ref, - required this.source, - required this.siblings, - required this.sourceInfo, - required Track track, - }) { - id = track.id; - name = track.name; - artists = track.artists; - album = track.album; - durationMs = track.durationMs; - discNumber = track.discNumber; - explicit = track.explicit; - externalIds = track.externalIds; - href = track.href; - isPlayable = track.isPlayable; - linkedFrom = track.linkedFrom; - popularity = track.popularity; - previewUrl = track.previewUrl; - trackNumber = track.trackNumber; - type = track.type; - uri = track.uri; - } - - static SourcedTrack fromJson( - Map json, { - required Ref ref, - }) { - final preferences = ref.read(userPreferencesProvider); - - final sourceInfo = SourceInfo.fromJson(json); - final source = SourceMap.fromJson(json); - final track = Track.fromJson(json); - final siblings = (json["siblings"] as List) - .map((sibling) => SourceInfo.fromJson(sibling)) - .toList() - .cast(); - - return switch (preferences.audioSource) { - AudioSource.youtube => YoutubeSourcedTrack( - ref: ref, - source: source, - siblings: siblings, - sourceInfo: sourceInfo, - track: track, - ), - AudioSource.piped => PipedSourcedTrack( - ref: ref, - source: source, - siblings: siblings, - sourceInfo: sourceInfo, - track: track, - ), - AudioSource.jiosaavn => JioSaavnSourcedTrack( - ref: ref, - source: source, - siblings: siblings, - sourceInfo: sourceInfo, - track: track, - ), - AudioSource.invidious => InvidiousSourcedTrack( - ref: ref, - source: source, - siblings: siblings, - sourceInfo: sourceInfo, - track: track, - ), - }; - } - - static String getSearchTerm(Track track) { - final artists = (track.artists ?? []) - .map((ar) => ar.name) - .toList() - .whereNotNull() - .toList(); - - final title = ServiceUtils.getTitle( - track.name!, - artists: artists, - onlyCleanArtist: true, - ).trim(); - - return "$title - ${artists.join(", ")}"; - } - - static fetchFromTrackAltSource({ - required Track track, - required Ref ref, - }) async { - final preferences = ref.read(userPreferencesProvider); - try { - return switch (preferences.audioSource) { - AudioSource.piped || - AudioSource.invidious || - AudioSource.jiosaavn => - await YoutubeSourcedTrack.fetchFromTrack(track: track, ref: ref), - AudioSource.youtube => - await JioSaavnSourcedTrack.fetchFromTrack(track: track, ref: ref), - }; - } on TrackNotFoundError catch (_) { - return switch (preferences.audioSource) { - AudioSource.piped || - AudioSource.youtube || - AudioSource.invidious => - await JioSaavnSourcedTrack.fetchFromTrack( - track: track, - ref: ref, - weakMatch: true, - ), - AudioSource.jiosaavn => - await YoutubeSourcedTrack.fetchFromTrack(track: track, ref: ref), - }; - } on HttpClientClosedException catch (_) { - return await PipedSourcedTrack.fetchFromTrack(track: track, ref: ref); - } on VideoUnplayableException catch (_) { - return await InvidiousSourcedTrack.fetchFromTrack(track: track, ref: ref); - } catch (e) { - if (e is DioException || e is ClientException || e is SocketException) { - return await JioSaavnSourcedTrack.fetchFromTrack( - track: track, - ref: ref, - weakMatch: preferences.audioSource == AudioSource.jiosaavn, - ); - } - rethrow; - } - } + required super.info, + required super.query, + required super.source, + required super.siblings, + required super.sources, + }); static Future fetchFromTrack({ - required Track track, + required SpotubeFullTrackObject query, required Ref ref, }) async { - final preferences = ref.read(userPreferencesProvider); - try { - return switch (preferences.audioSource) { - AudioSource.piped => - await PipedSourcedTrack.fetchFromTrack(track: track, ref: ref), - AudioSource.youtube => - await YoutubeSourcedTrack.fetchFromTrack(track: track, ref: ref), - AudioSource.jiosaavn => - await JioSaavnSourcedTrack.fetchFromTrack(track: track, ref: ref), - AudioSource.invidious => - await InvidiousSourcedTrack.fetchFromTrack(track: track, ref: ref), - }; - } on TrackNotFoundError catch (_) { - return switch (preferences.audioSource) { - AudioSource.piped || - AudioSource.youtube || - AudioSource.invidious => - await JioSaavnSourcedTrack.fetchFromTrack( - track: track, - ref: ref, - weakMatch: true, - ), - AudioSource.jiosaavn => - await YoutubeSourcedTrack.fetchFromTrack(track: track, ref: ref), - }; - } on HttpClientClosedException catch (_) { - return await PipedSourcedTrack.fetchFromTrack(track: track, ref: ref); - } on VideoUnplayableException catch (_) { - return await PipedSourcedTrack.fetchFromTrack(track: track, ref: ref); - } catch (e) { - if (e is DioException || e is ClientException || e is SocketException) { - return switch (preferences.audioSource) { - AudioSource.piped || - AudioSource.invidious => - await YoutubeSourcedTrack.fetchFromTrack( - track: track, - ref: ref, - ), - _ => await JioSaavnSourcedTrack.fetchFromTrack( - track: track, - ref: ref, - weakMatch: preferences.audioSource == AudioSource.jiosaavn, - ) - }; - } - rethrow; + final audioSource = await ref.read(audioSourcePluginProvider.future); + final audioSourceConfig = await ref.read(metadataPluginsProvider + .selectAsync((data) => data.defaultAudioSourcePluginConfig)); + if (audioSource == null || audioSourceConfig == null) { + throw MetadataPluginException.noDefaultAudioSourcePlugin(); } + + final database = ref.read(databaseProvider); + final cachedSource = await (database.select(database.sourceMatchTable) + ..where((s) => + s.trackId.equals(query.id) & + s.sourceType.equals(audioSourceConfig.slug)) + ..limit(1) + ..orderBy([ + (s) => + OrderingTerm(expression: s.createdAt, mode: OrderingMode.desc), + ])) + .get() + .then((s) => s.firstOrNull); + + if (cachedSource == null) { + final siblings = await fetchSiblings(ref: ref, query: query); + if (siblings.isEmpty) { + throw TrackNotFoundError(query); + } + + await database.into(database.sourceMatchTable).insert( + SourceMatchTableCompanion.insert( + trackId: query.id, + sourceInfo: Value(jsonEncode(siblings.first)), + sourceType: audioSourceConfig.slug, + ), + ); + + final manifest = await audioSource.audioSource.streams(siblings.first); + + return SourcedTrack( + ref: ref, + siblings: siblings.skip(1).toList(), + info: siblings.first, + source: audioSourceConfig.slug, + sources: manifest, + query: query, + ); + } + final item = SpotubeAudioSourceMatchObject.fromJson( + jsonDecode(cachedSource.sourceInfo), + ); + final manifest = await audioSource.audioSource.streams(item); + + final sourcedTrack = SourcedTrack( + ref: ref, + siblings: [], + sources: manifest, + info: item, + query: query, + source: audioSourceConfig.slug, + ); + + AppLogger.log.i("${query.name}: ${sourcedTrack.url}"); + + return sourcedTrack; } - static Future> fetchSiblings({ - required Track track, + static List rankResults( + List results, + SpotubeFullTrackObject track, + ) { + return results + .map((sibling) { + int score = 0; + + for (final artist in track.artists) { + final isSameChannelArtist = + sibling.artists.any((a) => a.toLowerCase() == artist.name); + + if (isSameChannelArtist) { + score += 1; + } + + final titleContainsArtist = + sibling.title.toLowerCase().contains(artist.name.toLowerCase()); + + if (titleContainsArtist) { + score += 1; + } + } + + final titleContainsTrackName = + sibling.title.toLowerCase().contains(track.name.toLowerCase()); + + final hasOfficialFlag = + officialMusicRegex.hasMatch(sibling.title.toLowerCase()); + + if (titleContainsTrackName) { + score += 3; + } + + if (hasOfficialFlag) { + score += 1; + } + + if (hasOfficialFlag && titleContainsTrackName) { + score += 2; + } + + return (sibling: sibling, score: score); + }) + .sorted((a, b) => b.score.compareTo(a.score)) + .map((e) => e.sibling) + .toList(); + } + + static Future> fetchSiblings({ + required SpotubeFullTrackObject query, required Ref ref, - }) { - final preferences = ref.read(userPreferencesProvider); + }) async { + final audioSource = await ref.read(audioSourcePluginProvider.future); - return switch (preferences.audioSource) { - AudioSource.piped => - PipedSourcedTrack.fetchSiblings(track: track, ref: ref), - AudioSource.youtube => - YoutubeSourcedTrack.fetchSiblings(track: track, ref: ref), - AudioSource.jiosaavn => - JioSaavnSourcedTrack.fetchSiblings(track: track, ref: ref), - AudioSource.invidious => - InvidiousSourcedTrack.fetchSiblings(track: track, ref: ref), - }; + if (audioSource == null) { + throw MetadataPluginException.noDefaultAudioSourcePlugin(); + } + + final videoResults = []; + + final searchResults = await audioSource.audioSource.matches(query); + + if (ServiceUtils.onlyContainsEnglish(query.name)) { + videoResults.addAll(searchResults); + } else { + videoResults.addAll(rankResults(searchResults, query)); + } + + return videoResults.toSet().toList(); } - Future copyWithSibling(); + Future copyWithSibling() async { + if (siblings.isNotEmpty) { + return this; + } + final fetchedSiblings = await fetchSiblings(ref: ref, query: query); - Future swapWithSibling(SourceInfo sibling); + return SourcedTrack( + ref: ref, + siblings: fetchedSiblings.where((s) => s.id != info.id).toList(), + source: source, + sources: sources, + info: info, + query: query, + ); + } + + Future swapWithSibling( + SpotubeAudioSourceMatchObject sibling, + ) async { + if (sibling.id == info.id) { + return null; + } + + final audioSource = await ref.read(audioSourcePluginProvider.future); + final audioSourceConfig = await ref.read(metadataPluginsProvider + .selectAsync((data) => data.defaultAudioSourcePluginConfig)); + if (audioSource == null || audioSourceConfig == null) { + throw MetadataPluginException.noDefaultAudioSourcePlugin(); + } + + // a sibling source that was fetched from the search results + final isStepSibling = siblings.none((s) => s.id == sibling.id); + + final newSourceInfo = isStepSibling + ? sibling + : siblings.firstWhere((s) => s.id == sibling.id); + + final newSiblings = siblings.where((s) => s.id != sibling.id).toList() + ..insert(0, info); + + final manifest = await audioSource.audioSource.streams(newSourceInfo); + + final database = ref.read(databaseProvider); + + // Delete the old Entry + await (database.sourceMatchTable.delete() + ..where( + (table) => + table.trackId.equals(query.id) & + table.sourceType.equals(audioSourceConfig.slug), + )) + .go(); + + await database.into(database.sourceMatchTable).insert( + SourceMatchTableCompanion.insert( + trackId: query.id, + sourceInfo: Value(jsonEncode(sibling)), + sourceType: audioSourceConfig.slug, + createdAt: Value(DateTime.now()), + ), + mode: InsertMode.replace, + ); + + return SourcedTrack( + ref: ref, + source: source, + siblings: newSiblings, + sources: manifest, + info: newSourceInfo, + query: query, + ); + } Future swapWithSiblingOfIndex(int index) { return swapWithSibling(siblings[index]); } - String get url { - final preferences = ref.read(userPreferencesProvider); + Future refreshStream() async { + final audioSource = await ref.read(audioSourcePluginProvider.future); + final audioSourceConfig = await ref.read(metadataPluginsProvider + .selectAsync((data) => data.defaultAudioSourcePluginConfig)); + if (audioSource == null || audioSourceConfig == null) { + throw MetadataPluginException.noDefaultAudioSourcePlugin(); + } - final codec = preferences.audioSource == AudioSource.jiosaavn - ? SourceCodecs.m4a - : preferences.streamMusicCodec; + List validStreams = []; - return getUrlOfCodec(codec); + final stringBuffer = StringBuffer(); + for (final source in sources) { + final res = await globalDio.head( + source.url, + options: + Options(validateStatus: (status) => status != null && status < 500), + ); + + stringBuffer.writeln( + "[${query.id}] ${res.statusCode} ${source.container} ${source.codec} ${source.bitrate}", + ); + + if (res.statusCode! < 400) { + validStreams.add(source); + } + } + + AppLogger.log.d(stringBuffer.toString()); + + if (validStreams.isEmpty) { + validStreams = await audioSource.audioSource.streams(info); + } + + final sourcedTrack = SourcedTrack( + ref: ref, + siblings: siblings, + source: source, + sources: validStreams, + info: info, + query: query, + ); + + AppLogger.log.i("Refreshing ${query.name}: ${sourcedTrack.url}"); + + return sourcedTrack; } - String getUrlOfCodec(SourceCodecs codec) { - final preferences = ref.read(userPreferencesProvider); + String? get url { + final preferences = ref.read(audioSourcePresetsProvider); - return source[codec]?[preferences.audioQuality] ?? - // this will ensure playback doesn't break - source[codec == SourceCodecs.m4a ? SourceCodecs.weba : SourceCodecs.m4a] - [preferences.audioQuality]; + return getUrlOfQuality( + preferences.presets[preferences.selectedStreamingContainerIndex], + preferences.selectedStreamingQualityIndex, + ); } - SourceCodecs get codec { - final preferences = ref.read(userPreferencesProvider); + /// Returns the URL of the track based on the codec and quality preferences. + /// If an exact match is not found, it will return the closest match based on + /// the user's audio quality preference. + /// + /// If no sources match the codec, it will return the first or last source + /// based on the user's audio quality preference. + SpotubeAudioSourceStreamObject? getStreamOfQuality( + SpotubeAudioSourceContainerPreset preset, + int qualityIndex, + ) { + if (sources.isEmpty) return null; - return preferences.audioSource == AudioSource.jiosaavn - ? SourceCodecs.m4a - : preferences.streamMusicCodec; + final quality = preset.qualities[qualityIndex]; + + final exactMatch = sources.firstWhereOrNull( + (source) { + if (source.container != preset.name) return false; + + if (quality case SpotubeAudioLosslessContainerQuality()) { + return source.sampleRate == quality.sampleRate && + source.bitDepth == quality.bitDepth; + } else { + return source.bitrate == + (quality as SpotubeAudioLossyContainerQuality).bitrate; + } + }, + ); + + if (exactMatch != null) { + return exactMatch; + } + + // Find the preset with closest quality to the supplied quality + return sources.where((source) { + return source.container == preset.name; + }).reduce((prev, curr) { + if (quality is SpotubeAudioLosslessContainerQuality) { + final prevDiff = ((prev.sampleRate ?? 0) - quality.sampleRate).abs() + + ((prev.bitDepth ?? 0) - quality.bitDepth).abs(); + final currDiff = ((curr.sampleRate ?? 0) - quality.sampleRate).abs() + + ((curr.bitDepth ?? 0) - quality.bitDepth).abs(); + return currDiff < prevDiff ? curr : prev; + } else { + final prevDiff = ((prev.bitrate ?? 0) - + (quality as SpotubeAudioLossyContainerQuality).bitrate) + .abs(); + final currDiff = ((curr.bitrate ?? 0) - quality.bitrate).abs(); + return currDiff < prevDiff ? curr : prev; + } + }); + } + + String? getUrlOfQuality( + SpotubeAudioSourceContainerPreset preset, + int qualityIndex, + ) { + return getStreamOfQuality(preset, qualityIndex)?.url; + } + + SpotubeAudioSourceContainerPreset? get qualityPreset { + final presetState = ref.read(audioSourcePresetsProvider); + return presetState.presets + .elementAtOrNull(presetState.selectedStreamingContainerIndex); } } diff --git a/lib/services/sourced_track/sources/invidious.dart b/lib/services/sourced_track/sources/invidious.dart deleted file mode 100644 index 2ec5068e..00000000 --- a/lib/services/sourced_track/sources/invidious.dart +++ /dev/null @@ -1,266 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:drift/drift.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:spotube/services/sourced_track/exceptions.dart'; -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/source_map.dart'; -import 'package:spotube/services/sourced_track/models/video_info.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:invidious/invidious.dart'; -import 'package:spotube/services/sourced_track/sources/youtube.dart'; -import 'package:spotube/utils/service_utils.dart'; - -final invidiousProvider = Provider( - (ref) { - final invidiousInstance = ref.watch( - userPreferencesProvider.select((s) => s.invidiousInstance), - ); - return InvidiousClient(server: invidiousInstance); - }, -); - -class InvidiousSourceInfo extends SourceInfo { - InvidiousSourceInfo({ - required super.id, - required super.title, - required super.artist, - required super.thumbnail, - required super.pageUrl, - required super.duration, - required super.artistUrl, - required super.album, - }); -} - -class InvidiousSourcedTrack extends SourcedTrack { - InvidiousSourcedTrack({ - required super.ref, - required super.source, - required super.siblings, - required super.sourceInfo, - required super.track, - }); - - static Future fetchFromTrack({ - required Track track, - required Ref ref, - }) async { - final database = ref.read(databaseProvider); - final cachedSource = await (database.select(database.sourceMatchTable) - ..where((s) => s.trackId.equals(track.id!)) - ..limit(1) - ..orderBy([ - (s) => - OrderingTerm(expression: s.createdAt, mode: OrderingMode.desc), - ])) - .getSingleOrNull(); - final invidiousClient = ref.read(invidiousProvider); - - if (cachedSource == null) { - final siblings = await fetchSiblings(ref: ref, track: track); - if (siblings.isEmpty) { - throw TrackNotFoundError(track); - } - - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: track.id!, - sourceId: siblings.first.info.id, - sourceType: const Value(SourceType.youtube), - ), - ); - - return InvidiousSourcedTrack( - ref: ref, - siblings: siblings.map((s) => s.info).skip(1).toList(), - source: siblings.first.source as SourceMap, - sourceInfo: siblings.first.info, - track: track, - ); - } else { - final manifest = - await invidiousClient.videos.get(cachedSource.sourceId, local: true); - - return InvidiousSourcedTrack( - ref: ref, - siblings: [], - source: toSourceMap(manifest), - sourceInfo: InvidiousSourceInfo( - id: manifest.videoId, - artist: manifest.author, - artistUrl: manifest.authorUrl, - pageUrl: "https://www.youtube.com/watch?v=${manifest.videoId}", - thumbnail: manifest.videoThumbnails.first.url, - title: manifest.title, - duration: Duration(seconds: manifest.lengthSeconds), - album: null, - ), - track: track, - ); - } - } - - static SourceMap toSourceMap(InvidiousVideoResponse manifest) { - final m4a = manifest.adaptiveFormats - .where((audio) => audio.type.contains("audio/mp4")) - .sorted((a, b) => int.parse(a.bitrate).compareTo(int.parse(b.bitrate))); - - final weba = manifest.adaptiveFormats - .where((audio) => audio.type.contains("audio/webm")) - .sorted((a, b) => int.parse(a.bitrate).compareTo(int.parse(b.bitrate))); - - return SourceMap( - m4a: SourceQualityMap( - high: m4a.first.url.toString(), - medium: (m4a.elementAtOrNull(m4a.length ~/ 2) ?? m4a[1]).url.toString(), - low: m4a.last.url.toString(), - ), - weba: SourceQualityMap( - high: weba.first.url.toString(), - medium: - (weba.elementAtOrNull(weba.length ~/ 2) ?? weba[1]).url.toString(), - low: weba.last.url.toString(), - ), - ); - } - - static Future toSiblingType( - int index, - YoutubeVideoInfo item, - InvidiousClient invidiousClient, - ) async { - SourceMap? sourceMap; - if (index == 0) { - final manifest = await invidiousClient.videos.get(item.id, local: true); - sourceMap = toSourceMap(manifest); - } - - final SiblingType sibling = ( - info: InvidiousSourceInfo( - id: item.id, - artist: item.channelName, - artistUrl: "https://www.youtube.com/${item.channelId}", - pageUrl: "https://www.youtube.com/watch?v=${item.id}", - thumbnail: item.thumbnailUrl, - title: item.title, - duration: item.duration, - album: null, - ), - source: sourceMap, - ); - - return sibling; - } - - static Future> fetchSiblings({ - required Track track, - required Ref ref, - }) async { - final invidiousClient = ref.read(invidiousProvider); - final preference = ref.read(userPreferencesProvider); - - final query = SourcedTrack.getSearchTerm(track); - - final searchResults = await invidiousClient.search.list( - query, - type: InvidiousSearchType.video, - ); - - if (ServiceUtils.onlyContainsEnglish(query)) { - return await Future.wait( - searchResults - .whereType() - .map( - (result) => YoutubeVideoInfo.fromSearchResponse( - result, - preference.searchMode, - ), - ) - .mapIndexed((i, r) => toSiblingType(i, r, invidiousClient)), - ); - } - - final rankedSiblings = YoutubeSourcedTrack.rankResults( - searchResults - .whereType() - .map( - (result) => YoutubeVideoInfo.fromSearchResponse( - result, - preference.searchMode, - ), - ) - .toList(), - track, - ); - - return await Future.wait( - rankedSiblings.mapIndexed((i, r) => toSiblingType(i, r, invidiousClient)), - ); - } - - @override - Future copyWithSibling() async { - if (siblings.isNotEmpty) { - return this; - } - final fetchedSiblings = await fetchSiblings(ref: ref, track: this); - - return InvidiousSourcedTrack( - ref: ref, - siblings: fetchedSiblings - .where((s) => s.info.id != sourceInfo.id) - .map((s) => s.info) - .toList(), - source: source, - sourceInfo: sourceInfo, - track: this, - ); - } - - @override - Future swapWithSibling(SourceInfo sibling) async { - if (sibling.id == sourceInfo.id) { - return null; - } - - // a sibling source that was fetched from the search results - final isStepSibling = siblings.none((s) => s.id == sibling.id); - - final newSourceInfo = isStepSibling - ? sibling - : siblings.firstWhere((s) => s.id == sibling.id); - final newSiblings = siblings.where((s) => s.id != sibling.id).toList() - ..insert(0, sourceInfo); - - final pipedClient = ref.read(invidiousProvider); - - final manifest = - await pipedClient.videos.get(newSourceInfo.id, local: true); - - final database = ref.read(databaseProvider); - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: id!, - sourceId: newSourceInfo.id, - sourceType: const Value(SourceType.youtube), - // Because we're sorting by createdAt in the query - // we have to update it to indicate priority - createdAt: Value(DateTime.now()), - ), - mode: InsertMode.replace, - ); - - return InvidiousSourcedTrack( - ref: ref, - siblings: newSiblings, - source: toSourceMap(manifest), - sourceInfo: newSourceInfo, - track: this, - ); - } -} diff --git a/lib/services/sourced_track/sources/jiosaavn.dart b/lib/services/sourced_track/sources/jiosaavn.dart deleted file mode 100644 index 1434e4f7..00000000 --- a/lib/services/sourced_track/sources/jiosaavn.dart +++ /dev/null @@ -1,238 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:drift/drift.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:spotube/services/sourced_track/exceptions.dart'; -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/source_map.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:jiosaavn/jiosaavn.dart'; -import 'package:spotube/extensions/string.dart'; - -final jiosaavnClient = JioSaavnClient(); - -class JioSaavnSourceInfo extends SourceInfo { - JioSaavnSourceInfo({ - required super.id, - required super.title, - required super.artist, - required super.thumbnail, - required super.pageUrl, - required super.duration, - required super.artistUrl, - required super.album, - }); -} - -class JioSaavnSourcedTrack extends SourcedTrack { - JioSaavnSourcedTrack({ - required super.ref, - required super.source, - required super.siblings, - required super.sourceInfo, - required super.track, - }); - - static Future fetchFromTrack({ - required Track track, - required Ref ref, - bool weakMatch = false, - }) async { - final database = ref.read(databaseProvider); - final cachedSource = await (database.select(database.sourceMatchTable) - ..where((s) => s.trackId.equals(track.id!)) - ..limit(1) - ..orderBy([ - (s) => - OrderingTerm(expression: s.createdAt, mode: OrderingMode.desc), - ])) - .getSingleOrNull(); - - if (cachedSource == null || - cachedSource.sourceType != SourceType.jiosaavn) { - final siblings = - await fetchSiblings(ref: ref, track: track, weakMatch: weakMatch); - - if (siblings.isEmpty) { - throw TrackNotFoundError(track); - } - - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: track.id!, - sourceId: siblings.first.info.id, - sourceType: const Value(SourceType.jiosaavn), - ), - ); - - return JioSaavnSourcedTrack( - ref: ref, - siblings: siblings.map((s) => s.info).skip(1).toList(), - source: siblings.first.source!, - sourceInfo: siblings.first.info, - track: track, - ); - } - - final [item] = - await jiosaavnClient.songs.detailsById([cachedSource.sourceId]); - - final (:info, :source) = toSiblingType(item); - - return JioSaavnSourcedTrack( - ref: ref, - siblings: [], - source: source!, - sourceInfo: info, - track: track, - ); - } - - static SiblingType toSiblingType(SongResponse result) { - final SiblingType sibling = ( - info: JioSaavnSourceInfo( - artist: [ - result.primaryArtists, - if (result.featuredArtists.isNotEmpty) ", ", - result.featuredArtists - ].join("").unescapeHtml(), - artistUrl: - "https://www.jiosaavn.com/artist/${result.primaryArtistsId.split(",").firstOrNull ?? ""}", - duration: Duration(seconds: int.parse(result.duration)), - id: result.id, - pageUrl: result.url, - thumbnail: result.image?.last.link ?? "", - title: result.name!.unescapeHtml(), - album: result.album.name, - ), - source: SourceMap( - m4a: SourceQualityMap( - high: result.downloadUrl! - .firstWhere((element) => element.quality == "320kbps") - .link, - medium: result.downloadUrl! - .firstWhere((element) => element.quality == "160kbps") - .link, - low: result.downloadUrl! - .firstWhere((element) => element.quality == "96kbps") - .link, - ), - ), - ); - - return sibling; - } - - static Future> fetchSiblings({ - required Track track, - required Ref ref, - bool weakMatch = false, - }) async { - final query = SourcedTrack.getSearchTerm(track); - - final SongSearchResponse(:results) = - await jiosaavnClient.search.songs(query, limit: 20); - - final trackArtistNames = track.artists?.map((ar) => ar.name).toList(); - - final matchedResults = results - .where( - (s) { - s.name?.unescapeHtml().contains(track.name!) ?? false; - - final sameName = s.name?.unescapeHtml() == track.name; - final artistNames = [ - s.primaryArtists, - if (s.featuredArtists.isNotEmpty) ", ", - s.featuredArtists - ].join("").unescapeHtml(); - final sameArtists = artistNames.split(", ").any( - (artist) => - trackArtistNames?.any((ar) => artist == ar) ?? false, - ); - if (weakMatch) { - final containsName = - s.name?.unescapeHtml().contains(track.name!) ?? false; - final containsPrimaryArtist = s.primaryArtists - .unescapeHtml() - .contains(trackArtistNames?.first ?? ""); - - return containsName && containsPrimaryArtist; - } - - return sameName && sameArtists; - }, - ) - .map(toSiblingType) - .toList(); - - if (weakMatch && matchedResults.isEmpty) { - return results.map(toSiblingType).toList(); - } - - return matchedResults; - } - - @override - Future copyWithSibling() async { - if (siblings.isNotEmpty) { - return this; - } - final fetchedSiblings = await fetchSiblings(ref: ref, track: this); - - return JioSaavnSourcedTrack( - ref: ref, - siblings: fetchedSiblings - .where((s) => s.info.id != sourceInfo.id) - .map((s) => s.info) - .toList(), - source: source, - sourceInfo: sourceInfo, - track: this, - ); - } - - @override - Future swapWithSibling(SourceInfo sibling) async { - if (sibling.id == sourceInfo.id) { - return null; - } - - // a sibling source that was fetched from the search results - final isStepSibling = siblings.none((s) => s.id == sibling.id); - - final newSourceInfo = isStepSibling - ? sibling - : siblings.firstWhere((s) => s.id == sibling.id); - final newSiblings = siblings.where((s) => s.id != sibling.id).toList() - ..insert(0, sourceInfo); - - final [item] = await jiosaavnClient.songs.detailsById([newSourceInfo.id]); - - final (:info, :source) = toSiblingType(item); - - final database = ref.read(databaseProvider); - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: id!, - sourceId: info.id, - sourceType: const Value(SourceType.jiosaavn), - // Because we're sorting by createdAt in the query - // we have to update it to indicate priority - createdAt: Value(DateTime.now()), - ), - mode: InsertMode.replace, - ); - - return JioSaavnSourcedTrack( - ref: ref, - siblings: newSiblings, - source: source!, - sourceInfo: info, - track: this, - ); - } -} diff --git a/lib/services/sourced_track/sources/piped.dart b/lib/services/sourced_track/sources/piped.dart deleted file mode 100644 index d24f110f..00000000 --- a/lib/services/sourced_track/sources/piped.dart +++ /dev/null @@ -1,301 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:drift/drift.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:piped_client/piped_client.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; - -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:spotube/services/sourced_track/exceptions.dart'; -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/source_map.dart'; -import 'package:spotube/services/sourced_track/models/video_info.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:spotube/services/sourced_track/sources/youtube.dart'; -import 'package:spotube/utils/service_utils.dart'; - -final pipedProvider = Provider( - (ref) { - final instance = - ref.watch(userPreferencesProvider.select((s) => s.pipedInstance)); - return PipedClient(instance: instance); - }, -); - -class PipedSourceInfo extends SourceInfo { - PipedSourceInfo({ - required super.id, - required super.title, - required super.artist, - required super.thumbnail, - required super.pageUrl, - required super.duration, - required super.artistUrl, - required super.album, - }); -} - -class PipedSourcedTrack extends SourcedTrack { - PipedSourcedTrack({ - required super.ref, - required super.source, - required super.siblings, - required super.sourceInfo, - required super.track, - }); - - static Future fetchFromTrack({ - required Track track, - required Ref ref, - }) async { - final database = ref.read(databaseProvider); - final cachedSource = await (database.select(database.sourceMatchTable) - ..where((s) => s.trackId.equals(track.id!)) - ..limit(1) - ..orderBy([ - (s) => - OrderingTerm(expression: s.createdAt, mode: OrderingMode.desc), - ])) - .getSingleOrNull(); - final preferences = ref.read(userPreferencesProvider); - final pipedClient = ref.read(pipedProvider); - - if (cachedSource == null) { - final siblings = await fetchSiblings(ref: ref, track: track); - if (siblings.isEmpty) { - throw TrackNotFoundError(track); - } - - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: track.id!, - sourceId: siblings.first.info.id, - sourceType: Value( - preferences.searchMode == SearchMode.youtube - ? SourceType.youtube - : SourceType.youtubeMusic, - ), - ), - ); - - return PipedSourcedTrack( - ref: ref, - siblings: siblings.map((s) => s.info).skip(1).toList(), - source: siblings.first.source as SourceMap, - sourceInfo: siblings.first.info, - track: track, - ); - } else { - final manifest = await pipedClient.streams(cachedSource.sourceId); - - return PipedSourcedTrack( - ref: ref, - siblings: [], - source: toSourceMap(manifest), - sourceInfo: PipedSourceInfo( - id: manifest.id, - artist: manifest.uploader, - artistUrl: manifest.uploaderUrl, - pageUrl: "https://www.youtube.com/watch?v=${manifest.id}", - thumbnail: manifest.thumbnailUrl, - title: manifest.title, - duration: manifest.duration, - album: null, - ), - track: track, - ); - } - } - - static SourceMap toSourceMap(PipedStreamResponse manifest) { - final m4a = manifest.audioStreams - .where((audio) => audio.format == PipedAudioStreamFormat.m4a) - .sorted((a, b) => a.bitrate.compareTo(b.bitrate)); - - final weba = manifest.audioStreams - .where((audio) => audio.format == PipedAudioStreamFormat.webm) - .sorted((a, b) => a.bitrate.compareTo(b.bitrate)); - - return SourceMap( - m4a: SourceQualityMap( - high: m4a.first.url.toString(), - medium: (m4a.elementAtOrNull(m4a.length ~/ 2) ?? m4a[1]).url.toString(), - low: m4a.last.url.toString(), - ), - weba: SourceQualityMap( - high: weba.first.url.toString(), - medium: - (weba.elementAtOrNull(weba.length ~/ 2) ?? weba[1]).url.toString(), - low: weba.last.url.toString(), - ), - ); - } - - static Future toSiblingType( - int index, - YoutubeVideoInfo item, - PipedClient pipedClient, - ) async { - SourceMap? sourceMap; - if (index == 0) { - final manifest = await pipedClient.streams(item.id); - sourceMap = toSourceMap(manifest); - } - - final SiblingType sibling = ( - info: PipedSourceInfo( - id: item.id, - artist: item.channelName, - artistUrl: "https://www.youtube.com/${item.channelId}", - pageUrl: "https://www.youtube.com/watch?v=${item.id}", - thumbnail: item.thumbnailUrl, - title: item.title, - duration: item.duration, - album: null, - ), - source: sourceMap, - ); - - return sibling; - } - - static Future> fetchSiblings({ - required Track track, - required Ref ref, - }) async { - final pipedClient = ref.read(pipedProvider); - final preference = ref.read(userPreferencesProvider); - - final query = SourcedTrack.getSearchTerm(track); - - final PipedSearchResult(items: searchResults) = await pipedClient.search( - query, - preference.searchMode == SearchMode.youtube - ? PipedFilter.video - : PipedFilter.musicSongs, - ); - - // when falling back to piped API make sure to use the YouTube mode - final isYouTubeMusic = preference.audioSource != AudioSource.piped - ? false - : preference.searchMode == SearchMode.youtubeMusic; - - if (isYouTubeMusic) { - final artists = (track.artists ?? []) - .map((ar) => ar.name) - .toList() - .whereNotNull() - .toList(); - - return await Future.wait( - searchResults - .map( - (result) => YoutubeVideoInfo.fromSearchItemStream( - result as PipedSearchItemStream, - preference.searchMode, - ), - ) - .sorted((a, b) => b.views.compareTo(a.views)) - .where( - (item) => artists.any( - (artist) => - artist.toLowerCase() == item.channelName.toLowerCase(), - ), - ) - .mapIndexed((i, r) => toSiblingType(i, r, pipedClient)), - ); - } - - if (ServiceUtils.onlyContainsEnglish(query)) { - return await Future.wait( - searchResults - .whereType() - .map( - (result) => YoutubeVideoInfo.fromSearchItemStream( - result, - preference.searchMode, - ), - ) - .mapIndexed((i, r) => toSiblingType(i, r, pipedClient)), - ); - } - - final rankedSiblings = YoutubeSourcedTrack.rankResults( - searchResults - .map( - (result) => YoutubeVideoInfo.fromSearchItemStream( - result as PipedSearchItemStream, - preference.searchMode, - ), - ) - .toList(), - track, - ); - - return await Future.wait( - rankedSiblings.mapIndexed((i, r) => toSiblingType(i, r, pipedClient)), - ); - } - - @override - Future copyWithSibling() async { - if (siblings.isNotEmpty) { - return this; - } - final fetchedSiblings = await fetchSiblings(ref: ref, track: this); - - return PipedSourcedTrack( - ref: ref, - siblings: fetchedSiblings - .where((s) => s.info.id != sourceInfo.id) - .map((s) => s.info) - .toList(), - source: source, - sourceInfo: sourceInfo, - track: this, - ); - } - - @override - Future swapWithSibling(SourceInfo sibling) async { - if (sibling.id == sourceInfo.id) { - return null; - } - - // a sibling source that was fetched from the search results - final isStepSibling = siblings.none((s) => s.id == sibling.id); - - final newSourceInfo = isStepSibling - ? sibling - : siblings.firstWhere((s) => s.id == sibling.id); - final newSiblings = siblings.where((s) => s.id != sibling.id).toList() - ..insert(0, sourceInfo); - - final pipedClient = ref.read(pipedProvider); - - final manifest = await pipedClient.streams(newSourceInfo.id); - - final database = ref.read(databaseProvider); - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: id!, - sourceId: newSourceInfo.id, - sourceType: const Value(SourceType.youtube), - // Because we're sorting by createdAt in the query - // we have to update it to indicate priority - createdAt: Value(DateTime.now()), - ), - mode: InsertMode.replace, - ); - - return PipedSourcedTrack( - ref: ref, - siblings: newSiblings, - source: toSourceMap(manifest), - sourceInfo: newSourceInfo, - track: this, - ); - } -} diff --git a/lib/services/sourced_track/sources/youtube.dart b/lib/services/sourced_track/sources/youtube.dart deleted file mode 100644 index 0b5ee71b..00000000 --- a/lib/services/sourced_track/sources/youtube.dart +++ /dev/null @@ -1,336 +0,0 @@ -import 'package:collection/collection.dart'; -import 'package:drift/drift.dart'; -import 'package:flutter_riverpod/flutter_riverpod.dart'; -import 'package:http/http.dart'; -import 'package:spotify/spotify.dart'; -import 'package:spotube/models/database/database.dart'; -import 'package:spotube/provider/database/database.dart'; -import 'package:spotube/services/logger/logger.dart'; -import 'package:spotube/services/song_link/song_link.dart'; -import 'package:spotube/services/sourced_track/enums.dart'; -import 'package:spotube/services/sourced_track/exceptions.dart'; -import 'package:spotube/services/sourced_track/models/source_info.dart'; -import 'package:spotube/services/sourced_track/models/source_map.dart'; -import 'package:spotube/services/sourced_track/models/video_info.dart'; -import 'package:spotube/services/sourced_track/sourced_track.dart'; -import 'package:spotube/utils/service_utils.dart'; -import 'package:youtube_explode_dart/youtube_explode_dart.dart'; - -final youtubeClient = YoutubeExplode(); -final officialMusicRegex = RegExp( - r"official\s(video|audio|music\svideo|lyric\svideo|visualizer)", - caseSensitive: false, -); - -class YoutubeSourceInfo extends SourceInfo { - YoutubeSourceInfo({ - required super.id, - required super.title, - required super.artist, - required super.thumbnail, - required super.pageUrl, - required super.duration, - required super.artistUrl, - required super.album, - }); -} - -class YoutubeSourcedTrack extends SourcedTrack { - YoutubeSourcedTrack({ - required super.source, - required super.siblings, - required super.sourceInfo, - required super.track, - required super.ref, - }); - - static Future fetchFromTrack({ - required Track track, - required Ref ref, - }) async { - final database = ref.read(databaseProvider); - final cachedSource = await (database.select(database.sourceMatchTable) - ..where((s) => s.trackId.equals(track.id!)) - ..limit(1) - ..orderBy([ - (s) => - OrderingTerm(expression: s.createdAt, mode: OrderingMode.desc), - ])) - .get() - .then((s) => s.firstOrNull); - - if (cachedSource == null || cachedSource.sourceType != SourceType.youtube) { - final siblings = await fetchSiblings(ref: ref, track: track); - if (siblings.isEmpty) { - throw TrackNotFoundError(track); - } - - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: track.id!, - sourceId: siblings.first.info.id, - sourceType: const Value(SourceType.youtube), - ), - ); - - return YoutubeSourcedTrack( - ref: ref, - siblings: siblings.map((s) => s.info).skip(1).toList(), - source: siblings.first.source as SourceMap, - sourceInfo: siblings.first.info, - track: track, - ); - } - final item = await youtubeClient.videos.get(cachedSource.sourceId); - final manifest = await youtubeClient.videos.streamsClient - .getManifest( - cachedSource.sourceId, - ) - .timeout( - const Duration(seconds: 5), - onTimeout: () => throw ClientException("Timeout"), - ); - return YoutubeSourcedTrack( - ref: ref, - siblings: [], - source: toSourceMap(manifest), - sourceInfo: YoutubeSourceInfo( - id: item.id.value, - artist: item.author, - artistUrl: "https://www.youtube.com/channel/${item.channelId}", - pageUrl: item.url, - thumbnail: item.thumbnails.highResUrl, - title: item.title, - duration: item.duration ?? Duration.zero, - album: null, - ), - track: track, - ); - } - - static SourceMap toSourceMap(StreamManifest manifest) { - var m4a = manifest.audioOnly - .where((audio) => audio.codec.mimeType == "audio/mp4") - .sortByBitrate(); - - var weba = manifest.audioOnly - .where((audio) => audio.codec.mimeType == "audio/webm") - .sortByBitrate(); - - m4a = m4a.isEmpty ? weba.toList() : m4a; - weba = weba.isEmpty ? m4a.toList() : weba; - - return SourceMap( - m4a: SourceQualityMap( - high: m4a.first.url.toString(), - medium: (m4a.elementAtOrNull(m4a.length ~/ 2) ?? m4a[1]).url.toString(), - low: m4a.last.url.toString(), - ), - weba: SourceQualityMap( - high: weba.first.url.toString(), - medium: - (weba.elementAtOrNull(weba.length ~/ 2) ?? weba[1]).url.toString(), - low: weba.last.url.toString(), - ), - ); - } - - static Future toSiblingType( - int index, - YoutubeVideoInfo item, - ) async { - SourceMap? sourceMap; - if (index == 0) { - final manifest = - await youtubeClient.videos.streamsClient.getManifest(item.id).timeout( - const Duration(seconds: 5), - onTimeout: () => throw ClientException("Timeout"), - ); - sourceMap = toSourceMap(manifest); - } - - final SiblingType sibling = ( - info: YoutubeSourceInfo( - id: item.id, - artist: item.channelName, - artistUrl: "https://www.youtube.com/channel/${item.channelId}", - pageUrl: "https://www.youtube.com/watch?v=${item.id}", - thumbnail: item.thumbnailUrl, - title: item.title, - duration: item.duration, - album: null, - ), - source: sourceMap, - ); - - return sibling; - } - - static List rankResults( - List results, Track track) { - final artists = (track.artists ?? []) - .map((ar) => ar.name) - .toList() - .whereNotNull() - .toList(); - - return results - .sorted((a, b) => b.views.compareTo(a.views)) - .map((sibling) { - int score = 0; - - for (final artist in artists) { - final isSameChannelArtist = - sibling.channelName.toLowerCase() == artist.toLowerCase(); - final channelContainsArtist = sibling.channelName - .toLowerCase() - .contains(artist.toLowerCase()); - - if (isSameChannelArtist || channelContainsArtist) { - score += 1; - } - - final titleContainsArtist = - sibling.title.toLowerCase().contains(artist.toLowerCase()); - - if (titleContainsArtist) { - score += 1; - } - } - - final titleContainsTrackName = - sibling.title.toLowerCase().contains(track.name!.toLowerCase()); - - final hasOfficialFlag = - officialMusicRegex.hasMatch(sibling.title.toLowerCase()); - - if (titleContainsTrackName) { - score += 3; - } - - if (hasOfficialFlag) { - score += 1; - } - - if (hasOfficialFlag && titleContainsTrackName) { - score += 2; - } - - return (sibling: sibling, score: score); - }) - .sorted((a, b) => b.score.compareTo(a.score)) - .map((e) => e.sibling) - .toList(); - } - - static Future> fetchSiblings({ - required Track track, - required Ref ref, - }) async { - final links = await SongLinkService.links(track.id!); - final ytLink = links.firstWhereOrNull((link) => link.platform == "youtube"); - - if (ytLink?.url != null - // allows to fetch siblings more results for already sourced track - && - track is! SourcedTrack) { - try { - return [ - await toSiblingType( - 0, - YoutubeVideoInfo.fromVideo( - await youtubeClient.videos.get(ytLink!.url!), - ), - ) - ]; - } on VideoUnplayableException catch (e, stack) { - // Ignore this error and continue with the search - AppLogger.reportError(e, stack); - } - } - - final query = SourcedTrack.getSearchTerm(track); - - final searchResults = await youtubeClient.search.search( - "$query - Topic", - filter: TypeFilters.video, - ); - - if (ServiceUtils.onlyContainsEnglish(query)) { - return await Future.wait(searchResults - .map(YoutubeVideoInfo.fromVideo) - .mapIndexed(toSiblingType)); - } - - final rankedSiblings = rankResults( - searchResults.map(YoutubeVideoInfo.fromVideo).toList(), - track, - ); - - return await Future.wait(rankedSiblings.mapIndexed(toSiblingType)); - } - - @override - Future swapWithSibling(SourceInfo sibling) async { - if (sibling.id == sourceInfo.id) { - return null; - } - - // a sibling source that was fetched from the search results - final isStepSibling = siblings.none((s) => s.id == sibling.id); - - final newSourceInfo = isStepSibling - ? sibling - : siblings.firstWhere((s) => s.id == sibling.id); - final newSiblings = siblings.where((s) => s.id != sibling.id).toList() - ..insert(0, sourceInfo); - - final manifest = await youtubeClient.videos.streamsClient - .getManifest(newSourceInfo.id) - .timeout( - const Duration(seconds: 5), - onTimeout: () => throw ClientException("Timeout"), - ); - - final database = ref.read(databaseProvider); - - await database.into(database.sourceMatchTable).insert( - SourceMatchTableCompanion.insert( - trackId: id!, - sourceId: newSourceInfo.id, - sourceType: const Value(SourceType.youtube), - // Because we're sorting by createdAt in the query - // we have to update it to indicate priority - createdAt: Value(DateTime.now()), - ), - mode: InsertMode.replace, - ); - - return YoutubeSourcedTrack( - ref: ref, - siblings: newSiblings, - source: toSourceMap(manifest), - sourceInfo: newSourceInfo, - track: this, - ); - } - - @override - Future copyWithSibling() async { - if (siblings.isNotEmpty) { - return this; - } - final fetchedSiblings = await fetchSiblings(ref: ref, track: this); - - return YoutubeSourcedTrack( - ref: ref, - siblings: fetchedSiblings - .where((s) => s.info.id != sourceInfo.id) - .map((s) => s.info) - .toList(), - source: source, - sourceInfo: sourceInfo, - track: this, - ); - } -} diff --git a/lib/services/wm_tools/wm_tools.dart b/lib/services/wm_tools/wm_tools.dart index 920e09b5..f60b4ac9 100644 --- a/lib/services/wm_tools/wm_tools.dart +++ b/lib/services/wm_tools/wm_tools.dart @@ -1,4 +1,4 @@ -import 'package:flutter/material.dart'; +import 'package:shadcn_flutter/shadcn_flutter.dart'; import 'package:spotube/services/kv_store/kv_store.dart'; import 'package:spotube/utils/platform.dart'; import 'package:window_manager/window_manager.dart'; diff --git a/lib/services/youtube_engine/newpipe_engine.dart b/lib/services/youtube_engine/newpipe_engine.dart new file mode 100644 index 00000000..d6445a19 --- /dev/null +++ b/lib/services/youtube_engine/newpipe_engine.dart @@ -0,0 +1,116 @@ +import 'package:flutter_new_pipe_extractor/flutter_new_pipe_extractor.dart' + hide Engagement; +import 'package:spotube/services/youtube_engine/youtube_engine.dart'; +import 'package:spotube/utils/platform.dart'; +import 'package:youtube_explode_dart/youtube_explode_dart.dart'; +import 'package:http_parser/http_parser.dart'; + +class NewPipeEngine implements YouTubeEngine { + static bool get isAvailableForPlatform => kIsAndroid || kIsDesktop; + + AudioOnlyStreamInfo _parseAudioStream(AudioStream stream, String videoId) { + return AudioOnlyStreamInfo( + VideoId(videoId), + stream.itag, + Uri.parse(stream.content), + StreamContainer.parse(stream.mediaFormat!.mimeType.split("/").last), + FileSize.unknown, + Bitrate(stream.bitrate), + stream.codec, + switch (stream.bitrate) { + > 130 * 1024 => "high", + > 64 * 1024 => "medium", + _ => "low", + }, + [], + MediaType.parse(stream.mediaFormat!.mimeType), + null, + ); + } + + Video _parseVideo(VideoInfo info) { + return Video( + VideoId(info.id), + info.name, + info.uploaderName, + ChannelId(info.uploaderUrl), + info.uploadDate.offsetDateTime, + info.uploadDate.offsetDateTime.toString(), + info.uploadDate.offsetDateTime, + info.description.content ?? "", + Duration(seconds: info.duration), + ThumbnailSet(info.id), + info.tags, + Engagement( + info.viewCount, + info.likeCount, + info.dislikeCount, + ), + !info.streamType.name.toLowerCase().contains("live"), + ); + } + + Video _parseVideoResult(VideoSearchResultItem info) { + final id = Uri.parse(info.url).queryParameters["v"]!; + return Video( + VideoId(id), + info.name, + info.uploaderName, + ChannelId(info.uploaderUrl), + info.uploadDate?.offsetDateTime, + info.uploadDate?.offsetDateTime.toString(), + info.uploadDate?.offsetDateTime, + info.shortDescription ?? "", + Duration(seconds: info.duration), + ThumbnailSet(id), + [], + Engagement(info.viewCount, null, null), + !info.streamType.name.toLowerCase().contains("live"), + ); + } + + @override + Future getStreamManifest(String videoId) async { + final video = await NewPipeExtractor.getVideoInfo(videoId); + + final streams = + video.audioStreams.map((stream) => _parseAudioStream(stream, videoId)); + + return StreamManifest(streams); + } + + @override + Future