diff --git a/.circleci/config.yml b/.circleci/config.yml
deleted file mode 100644
index a55310ce..00000000
--- a/.circleci/config.yml
+++ /dev/null
@@ -1,177 +0,0 @@
-version: 2.1
-
-orbs:
- gh: circleci/github-cli@2.2.0
-
-jobs:
- flutter_linux_arm:
- machine:
- image: ubuntu-2204:current
- resource_class: arm.medium
- parameters:
- version:
- type: string
- default: 3.1.1
- channel:
- type: enum
- enum:
- - release
- - nightly
- default: release
- github_run_number:
- type: string
- default: "0"
- dry_run:
- type: boolean
- default: true
- steps:
- - checkout
- - gh/setup
-
- - run:
- name: Get current date
- command: |
- echo "export CURRENT_DATE=$(date +%Y-%m-%d)" >> $BASH_ENV
-
- - run:
- name: Install dependencies
- command: |
- 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 zip rpm
-
- - run:
- name: Install Flutter
- command: |
- git clone https://github.com/flutter/flutter.git
- cd flutter && git checkout stable && cd ..
- export PATH="$PATH:`pwd`/flutter/bin"
- flutter precache
- flutter doctor -v
-
- - run:
- name: Install AppImageTool
- command: |
- wget -O appimagetool "https://github.com/AppImage/AppImageKit/releases/download/continuous/appimagetool-aarch64.AppImage"
- chmod +x appimagetool
- mv appimagetool flutter/bin
-
- - persist_to_workspace:
- root: flutter
- paths:
- - .
-
- - when:
- condition:
- equal: [<< parameters.channel >>, nightly]
- steps:
- - run:
- name: Replace pubspec version and BUILD_VERSION Env (nightly)
- command: |
- curl -sS https://webi.sh/yq | sh
- yq -i '.version |= sub("\+\d+", "+<< parameters.channel >>.")' pubspec.yaml
- yq -i '.version += strenv(GITHUB_RUN_NUMBER)' pubspec.yaml
- echo 'export BUILD_VERSION="<< parameters.version >>+<< parameters.channel >>.<< parameters.github_run_number >>"' >> $BASH_ENV
-
- - when:
- condition:
- equal: [<< parameters.channel >>, release]
- steps:
- - run: echo 'export BUILD_VERSION="<< parameters.version >>"' >> $BASH_ENV
-
- - run:
- name: Generate .env file
- command: |
- echo "SPOTIFY_SECRETS=${SPOTIFY_SECRETS}" >> .env
-
- - run:
- name: Replace Version in files
- command: |
- sed -i 's|%{{APPDATA_RELEASE}}%| |' linux/com.github.KRTirtho.Spotube.appdata.xml
- echo "build_arch: aarch64" >> linux/packaging/rpm/make_config.yaml
-
- - run:
- name: Build secrets
- command: |
- export PATH="$PATH:`pwd`/flutter/bin"
- flutter config --enable-linux-desktop
- flutter pub get
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
-
- - run:
- name: Build Flutter app
- command: |
- export PATH="$PATH:`pwd`/flutter/bin"
- export PATH="$PATH":"$HOME/.pub-cache/bin"
- dart pub global activate flutter_distributor
- alias dpkg-deb="dpkg-deb --Zxz"
- flutter_distributor package --platform=linux --targets=deb
- flutter_distributor package --platform=linux --targets=appimage
- flutter_distributor package --platform=linux --targets=rpm
-
- - when:
- condition:
- equal: [<< parameters.channel >>, nightly]
- steps:
- - run: make tar VERSION=nightly ARCH=arm64 PKG_ARCH=aarch64
-
- - when:
- condition:
- equal: [<< parameters.channel >>, release]
- steps:
- - run: make tar VERSION=${BUILD_VERSION} ARCH=arm64 PKG_ARCH=aarch64
-
- - run:
- name: Move artifacts
- command: |
- mkdir bundle
- mv build/spotube-linux-*-aarch64.tar.xz bundle/
- mv dist/**/spotube-*-linux.deb bundle/Spotube-linux-aarch64.deb
- mv dist/**/spotube-*-linux.rpm bundle/Spotube-linux-aarch64.rpm
- mv dist/**/spotube-*-linux.AppImage bundle/Spotube-linux-aarch64.AppImage
- zip -r Spotube-linux-aarch64.zip bundle
-
- - store_artifacts:
- path: Spotube-linux-aarch64.zip
-
- - when:
- condition:
- and:
- - equal: [<< parameters.dry_run >>, false]
- - equal: [<< parameters.channel >>, release]
- steps:
- - run:
- name: Upload to release (release)
- command: gh release upload v<< parameters.version >> bundle/* --clobber
-
- - when:
- condition:
- and:
- - equal: [<< parameters.dry_run >>, false]
- - equal: [<< parameters.channel >>, nightly]
- steps:
- - run:
- name: Upload to release (nightly)
- command: gh release upload nightly bundle/* --clobber
-
-parameters:
- GHA_Actor:
- type: string
- default: ""
- GHA_Action:
- type: string
- default: ""
- GHA_Event:
- type: string
- default: ""
- GHA_Meta:
- type: string
- default: ""
-
-workflows:
- build_flutter_for_arm_workflow:
- when: << pipeline.parameters.GHA_Action >>
- jobs:
- - flutter_linux_arm:
- context:
- - org-global
- - GITHUB_CREDS
diff --git a/.dockerignore b/.dockerignore
new file mode 100644
index 00000000..ddfd1517
--- /dev/null
+++ b/.dockerignore
@@ -0,0 +1,6 @@
+build
+dist
+.dart_tool
+.idea
+.github
+.git
\ No newline at end of file
diff --git a/.env.example b/.env.example
index 22abd24b..6a88cb99 100644
--- a/.env.example
+++ b/.env.example
@@ -9,3 +9,8 @@ ENABLE_UPDATE_CHECK=
LASTFM_API_KEY=
LASTFM_API_SECRET=
+
+# Release channel. Can be: nightly, stable
+RELEASE_CHANNEL=
+
+HIDE_DONATIONS=
diff --git a/.fvm/fvm_config.json b/.fvm/fvm_config.json
index 7ca74200..305f34df 100644
--- a/.fvm/fvm_config.json
+++ b/.fvm/fvm_config.json
@@ -1,4 +1,3 @@
{
- "flutterSdkVersion": "3.19.1",
- "flavors": {}
+ "flutterSdkVersion": "3.24.3"
}
\ No newline at end of file
diff --git a/.fvmrc b/.fvmrc
new file mode 100644
index 00000000..c62692b4
--- /dev/null
+++ b/.fvmrc
@@ -0,0 +1,4 @@
+{
+ "flutter": "3.24.3",
+ "flavors": {}
+}
\ No newline at end of file
diff --git a/.github/Dockerfile b/.github/Dockerfile
new file mode 100644
index 00000000..f6a9f538
--- /dev/null
+++ b/.github/Dockerfile
@@ -0,0 +1,25 @@
+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 64ee89d2..a9c57836 100644
--- a/.github/ISSUE_TEMPLATE/bug_report.yml
+++ b/.github/ISSUE_TEMPLATE/bug_report.yml
@@ -7,8 +7,12 @@ labels:
body:
- type: checkboxes
attributes:
- label: Is there an existing issue for this?
- description: Make sure to check if this issue is a duplicate.
+ label: Is there an existing issue for this? (Please read the description)
+ description: |
+ 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.
options:
- label: I have searched the existing issues
required: true
@@ -16,23 +20,41 @@ body:
attributes:
label: Current Behavior
description: Write what you are experiencing currently.
+ placeholder: |
+ The app isn't working as expected. It crashes when I do this...
validations:
required: true
- type: textarea
attributes:
label: Expected Behavior
description: Write what you expected to happen.
+ placeholder: |
+ The app should do this when I do that...
validations:
required: true
- type: textarea
attributes:
label: Steps to reproduce
- description: Steps to reproduce the issue. A not well written description might delay the resolve of it.
+ description: Steps to reproduce the issue. A not well written description might lead to the delay in fixing the issue.
placeholder: |
1. I opened the app
2. I did this
3. And that
4. Then this happened
+ - type: textarea
+ attributes:
+ label: Logs
+ description: |
+ If you have any logs, paste them here. Make sure to remove any sensitive information.
+ You can find the logs in the app's Settings > Developers > Logs page.
+ value: |
+
+ Logs
+
+ ```
+
+ ```
+
validations:
required: true
- type: input
@@ -53,7 +75,7 @@ body:
description: Where did you install Spotube from?
multiple: true
options:
- - "Website (spotube.netlify.app) or (spotube.krtirtho.dev)"
+ - "Website (spotube.krtirtho.dev)"
- "GitHub Releases (Binary)"
- "GitHub Actions (Nightly Binary)"
- "Play Store (Android)"
@@ -74,7 +96,7 @@ 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. We welcome contributions!
+ 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!
options:
- label: I'm ready to work on this issue!
- required: false
\ No newline at end of file
+ required: false
diff --git a/.github/workflows/pr-lint.yml b/.github/workflows/pr-lint.yml
index e4fb55c5..db158029 100644
--- a/.github/workflows/pr-lint.yml
+++ b/.github/workflows/pr-lint.yml
@@ -4,13 +4,15 @@ on:
pull_request:
env:
- FLUTTER_VERSION: '3.16.0'
+ FLUTTER_VERSION: 3.22.2
jobs:
lint:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
+ with:
+ ref: ${{ github.event.pull_request.head.sha }}
- uses: subosito/flutter-action@v2
with:
flutter-version: ${{ env.FLUTTER_VERSION }}
diff --git a/.github/workflows/spotube-publish-binary.yml b/.github/workflows/spotube-publish-binary.yml
index 12a2f99b..3a456bda 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.1.0
+ default: 3.8.3
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)
+ description: Jobs to run (flathub,aur,winget,chocolatey,playstore)
required: true
type: string
- default: "flathub,aur,winget,chocolatey"
+ default: "flathub,aur,winget,chocolatey,playstore"
jobs:
flathub:
@@ -66,7 +66,7 @@ jobs:
- name: Release to AUR
if: ${{ !inputs.dry_run }}
- uses: KSXGitHub/github-actions-deploy-aur@v2.7.0
+ uses: KSXGitHub/github-actions-deploy-aur@v2.7.2
with:
pkgname: spotube-bin
pkgbuild: aur-struct/PKGBUILD
@@ -76,12 +76,12 @@ jobs:
commit_message: Updated to v${{ inputs.version }}
winget:
- runs-on: windows-latest
+ runs-on: ubuntu-latest
if: contains(inputs.jobs, 'winget')
steps:
- name: Release winget package
if: ${{ !inputs.dry_run }}
- uses: vedantmgoyal2009/winget-releaser@v2
+ uses: vedantmgoyal9/winget-releaser@main
with:
version: ${{ inputs.version }}
release-tag: v${{ inputs.version }}
@@ -104,3 +104,34 @@ jobs:
- name: Publish to Chocolatey Repository
if: ${{ !inputs.dry_run }}
run: choco push Spotube-windows-x86_64.nupkg --source https://push.chocolatey.org/
+
+ playstore:
+ runs-on: ubuntu-latest
+ if: contains(inputs.jobs, 'playstore')
+ 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 }}
diff --git a/.github/workflows/spotube-release-binary.yml b/.github/workflows/spotube-release-binary.yml
index adb99003..d059a067 100644
--- a/.github/workflows/spotube-release-binary.yml
+++ b/.github/workflows/spotube-release-binary.yml
@@ -2,279 +2,126 @@ name: Spotube Release Binary
on:
workflow_dispatch:
inputs:
- version:
- description: Version to release (x.x.x)
- default: 3.4.1
- required: true
channel:
type: choice
- description: Release Channel
- required: true
options:
- stable
- nightly
default: nightly
+ description: The release channel
debug:
- description: Debug on failed when channel is nightly
- required: true
type: boolean
default: false
+ description: Debug with SSH toggle
+ required: false
dry_run:
- description: Dry run
- required: true
type: boolean
- default: true
+ default: false
+ description: Dry run without uploading to release
env:
- FLUTTER_VERSION: '3.19.1'
+ FLUTTER_VERSION: 3.24.3
+
+permissions:
+ contents: write
jobs:
- windows:
- runs-on: windows-latest
+ build_platform:
+ strategy:
+ matrix:
+ include:
+ - os: ubuntu-latest
+ platform: linux
+ files: |
+ dist/Spotube-linux-x86_64.deb
+ dist/Spotube-linux-x86_64.rpm
+ dist/spotube-linux-*-x86_64.tar.xz
+ - os: ubuntu-latest
+ platform: linux_arm
+ files: |
+ dist/Spotube-linux-aarch64.deb
+ dist/spotube-linux-*-aarch64.tar.xz
+ - os: ubuntu-latest
+ platform: android
+ files: |
+ build/Spotube-android-all-arch.apk
+ build/Spotube-playstore-all-arch.aab
+ - os: windows-latest
+ platform: windows
+ files: |
+ dist/Spotube-windows-x86_64.nupkg
+ dist/Spotube-windows-x86_64-setup.exe
+ - os: macos-latest
+ platform: ios
+ files: |
+ Spotube-iOS.ipa
+ - os: macos-14
+ platform: macos
+ 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
with:
cache: true
+ cache-key: ${{ runner.os }}-flutter-${{ hashFiles('**/pubspec.yaml') }}
flutter-version: ${{ env.FLUTTER_VERSION }}
-
- - name: Replace pubspec version and BUILD_VERSION Env (nightly)
- if: ${{ inputs.channel == 'nightly' }}
- run: |
- choco install sed make yq -y
- yq -i '.version |= sub("\+\d+", "+${{ inputs.channel }}.")' pubspec.yaml
- yq -i '.version += strenv(GITHUB_RUN_NUMBER)' pubspec.yaml
- "BUILD_VERSION=${{ inputs.version }}+${{ inputs.channel }}.${{ github.run_number }}" >> $env:GITHUB_ENV
-
- - name: BUILD_VERSION Env (stable)
- if: ${{ inputs.channel == 'stable' }}
- run: |
- "BUILD_VERSION=${{ inputs.version }}" >> $env:GITHUB_ENV
-
- - name: Replace version in files
- run: |
- choco install sed make -y
- sed -i "s/%{{SPOTUBE_VERSION}}%/${{ env.BUILD_VERSION }}/" windows/runner/Runner.rc
- sed -i "s/%{{SPOTUBE_VERSION}}%/${{ env.BUILD_VERSION }}/" choco-struct/tools/VERIFICATION.txt
- sed -i "s/%{{SPOTUBE_VERSION}}%/${{ env.BUILD_VERSION }}/" choco-struct/spotube.nuspec
-
- - name: Create Stable .env
- if: ${{ inputs.channel == 'stable' }}
- run: echo '${{ secrets.DOTENV_RELEASE }}' > .env
-
- - name: Create Nightly .env
- if: ${{ inputs.channel == 'nightly' }}
- run: echo '${{ secrets.DOTENV_NIGHTLY }}' > .env
-
- - name: Generating Secrets
- run: |
- flutter config --enable-windows-desktop
- flutter pub get
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
-
- - name: Build Windows Executable
- run: |
- dart pub global activate flutter_distributor
- make innoinstall
- flutter_distributor package --platform=windows --targets=exe --skip-clean
- mv dist/**/spotube-*-windows-setup.exe dist/Spotube-windows-x86_64-setup.exe
-
- - name: Create Chocolatey Package and set hash
- if: ${{ inputs.channel == 'stable' }}
- run: |
- Set-Variable -Name HASH -Value (Get-FileHash dist\Spotube-windows-x86_64-setup.exe).Hash
- sed -i "s/%{{WIN_SHA256}}%/$HASH/" choco-struct/tools/VERIFICATION.txt
- make choco
- mv dist/spotube.*.nupkg dist/Spotube-windows-x86_64.nupkg
-
-
- - name: Upload Artifact
- uses: actions/upload-artifact@v3
+ - name: Setup Java
+ if: ${{matrix.platform == 'android'}}
+ uses: actions/setup-java@v4
with:
- if-no-files-found: error
- name: Spotube-Release-Binaries
- path: |
- dist/Spotube-windows-x86_64.nupkg
- dist/Spotube-windows-x86_64-setup.exe
-
- - name: Debug With SSH When fails
- if: ${{ failure() && inputs.debug && inputs.channel == 'nightly' }}
- uses: mxschmitt/action-tmate@v3
+ 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:
- limit-access-to-actor: true
+ toolchain: stable
- linux:
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v4
- - uses: subosito/flutter-action@v2.12.0
- with:
- cache: true
- flutter-version: ${{ env.FLUTTER_VERSION }}
-
- - name: Get current date
- id: date
- run: echo "::set-output name=date::$(date +'%Y-%m-%d')"
-
- - name: Install Dependencies
- 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
-
- - name: Install AppImage Tool
- run: |
- wget -O appimagetool "https://github.com/AppImage/AppImageKit/releases/download/continuous/appimagetool-x86_64.AppImage"
- chmod +x appimagetool
- mv appimagetool /usr/local/bin/
-
- - name: Replace pubspec version and BUILD_VERSION Env (nightly)
- if: ${{ inputs.channel == 'nightly' }}
- run: |
- curl -sS https://webi.sh/yq | sh
- yq -i '.version |= sub("\+\d+", "+${{ inputs.channel }}.")' pubspec.yaml
- yq -i '.version += strenv(GITHUB_RUN_NUMBER)' pubspec.yaml
- echo "BUILD_VERSION=${{ inputs.version }}+${{ inputs.channel }}.${{ github.run_number }}" >> $GITHUB_ENV
-
- - name: BUILD_VERSION Env (stable)
- if: ${{ inputs.channel == 'stable' }}
- run: |
- echo "BUILD_VERSION=${{ inputs.version }}" >> $GITHUB_ENV
-
- - name: Create Stable .env
- if: ${{ inputs.channel == 'stable' }}
- run: echo '${{ secrets.DOTENV_RELEASE }}' > .env
-
- - name: Create Nightly .env
- if: ${{ inputs.channel == 'nightly' }}
- run: echo '${{ secrets.DOTENV_NIGHTLY }}' > .env
-
- - name: Replace Version in files
- run: |
- sed -i 's|%{{APPDATA_RELEASE}}%| |' linux/com.github.KRTirtho.Spotube.appdata.xml
-
- - name: Generate Secrets
- run: |
- flutter config --enable-linux-desktop
- flutter pub get
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
-
- - name: Build Linux Packages
- run: |
- dart pub global activate flutter_distributor
- alias dpkg-deb="dpkg-deb --Zxz"
- flutter_distributor package --platform=linux --targets=deb
- flutter_distributor package --platform=linux --targets=rpm
-
- - name: Create tar.xz (stable)
- if: ${{ inputs.channel == 'stable' }}
- run: make tar VERSION=${{ env.BUILD_VERSION }} ARCH=x64 PKG_ARCH=x86_64
-
- - name: Create tar.xz (nightly)
- if: ${{ inputs.channel == 'nightly' }}
- run: make tar VERSION=nightly ARCH=x64 PKG_ARCH=x86_64
-
- - name: Move Files to dist
- run: |
- mv build/spotube-linux-*-x86_64.tar.xz dist/
- mv dist/**/spotube-*-linux.deb dist/Spotube-linux-x86_64.deb
- mv dist/**/spotube-*-linux.rpm dist/Spotube-linux-x86_64.rpm
-
-
- - uses: actions/upload-artifact@v3
- if: ${{ inputs.channel == 'release' }}
- with:
- if-no-files-found: error
- name: Spotube-Release-Binaries
- path: |
- dist/Spotube-linux-x86_64.deb
- dist/Spotube-linux-x86_64.rpm
- dist/spotube-linux-${{ env.BUILD_VERSION }}-x86_64.tar.xz
-
- - uses: actions/upload-artifact@v3
- if: ${{ inputs.channel == 'nightly' }}
- with:
- if-no-files-found: error
- name: Spotube-Release-Binaries
- path: |
- dist/Spotube-linux-x86_64.deb
- dist/Spotube-linux-x86_64.rpm
- dist/spotube-linux-nightly-x86_64.tar.xz
-
- - name: Debug With SSH When fails
- if: ${{ failure() && inputs.debug && inputs.channel == 'nightly' }}
- uses: mxschmitt/action-tmate@v3
- with:
- limit-access-to-actor: true
-
-
- android:
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v4
- - uses: subosito/flutter-action@v2.12.0
- with:
- cache: true
- flutter-version: ${{ env.FLUTTER_VERSION }}
-
- - name: Install Dependencies
- run: |
- sudo apt-get update -y
- sudo apt-get install -y clang cmake ninja-build pkg-config libgtk-3-dev make python3-pip python3-setuptools patchelf desktop-file-utils libgdk-pixbuf2.0-dev fakeroot strace fuse xmlstarlet
-
- - name: Replace pubspec version and BUILD_VERSION Env (nightly)
- if: ${{ inputs.channel == 'nightly' }}
- run: |
- curl -sS https://webi.sh/yq | sh
- yq -i '.version |= sub("\+\d+", "+${{ inputs.channel }}.")' pubspec.yaml
- yq -i '.version += strenv(GITHUB_RUN_NUMBER)' pubspec.yaml
- echo "BUILD_VERSION=${{ inputs.version }}+${{ inputs.channel }}.${{ github.run_number }}" >> $GITHUB_ENV
-
- - name: BUILD_VERSION Env (stable)
- if: ${{ inputs.channel == 'stable' }}
- run: |
- echo "BUILD_VERSION=${{ inputs.version }}" >> $GITHUB_ENV
-
- - name: Create Stable .env
- if: ${{ inputs.channel == 'stable' }}
- run: echo '${{ secrets.DOTENV_RELEASE }}' > .env
-
- - name: Create Nightly .env
- if: ${{ inputs.channel == 'nightly' }}
- run: echo '${{ secrets.DOTENV_NIGHTLY }}' > .env
-
- - name: Generate Secrets
+ - name: Install ${{matrix.platform}} dependencies
run: |
flutter pub get
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
+ dart cli/cli.dart install-dependencies --platform=${{matrix.platform}}
- 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: Build Apk
- run: |
- flutter build apk --flavor ${{ inputs.channel }}
- mv build/app/outputs/flutter-apk/app-${{ inputs.channel }}-release.apk build/Spotube-android-all-arch.apk
-
- - name: Build Playstore AppBundle
- run: |
- echo 'ENABLE_UPDATE_CHECK=0' >> .env
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
- export MANIFEST=android/app/src/main/AndroidManifest.xml
- xmlstarlet ed -d '//meta-data[@android:name="com.google.android.gms.car.application"]' $MANIFEST > $MANIFEST.tmp
- mv $MANIFEST.tmp $MANIFEST
- flutter build appbundle --flavor ${{ inputs.channel }}
- mv build/app/outputs/bundle/${{ inputs.channel }}Release/app-${{ inputs.channel }}-release.aab build/Spotube-playstore-all-arch.aab
-
-
+
+ - 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}}
+ env:
+ CHANNEL: ${{inputs.channel}}
+ DOTENV: ${{secrets.DOTENV_RELEASE}}
+
- uses: actions/upload-artifact@v3
with:
if-no-files-found: error
name: Spotube-Release-Binaries
- path: |
- build/Spotube-android-all-arch.apk
- build/Spotube-playstore-all-arch.aab
+ path: ${{matrix.files}}
- name: Debug With SSH When fails
if: ${{ failure() && inputs.debug && inputs.channel == 'nightly' }}
@@ -282,136 +129,12 @@ jobs:
with:
limit-access-to-actor: true
- macos:
-
- runs-on: macos-12
- steps:
- - uses: actions/checkout@v4
- - uses: subosito/flutter-action@v2.12.0
- with:
- cache: true
- flutter-version: ${{ env.FLUTTER_VERSION }}
-
- - name: Replace pubspec version and BUILD_VERSION Env (nightly)
- if: ${{ inputs.channel == 'nightly' }}
- run: |
- brew install yq
- yq -i '.version |= sub("\+\d+", "+${{ inputs.channel }}.")' pubspec.yaml
- yq -i '.version += strenv(GITHUB_RUN_NUMBER)' pubspec.yaml
- echo "BUILD_VERSION=${{ inputs.version }}+${{ inputs.channel }}.${{ github.run_number }}" >> $GITHUB_ENV
-
- - name: BUILD_VERSION Env (stable)
- if: ${{ inputs.channel == 'stable' }}
- run: |
- echo "BUILD_VERSION=${{ inputs.version }}" >> $GITHUB_ENV
-
- - name: Create Stable .env
- if: ${{ inputs.channel == 'stable' }}
- run: echo '${{ secrets.DOTENV_RELEASE }}' > .env
-
- - name: Create Nightly .env
- if: ${{ inputs.channel == 'nightly' }}
- run: echo '${{ secrets.DOTENV_NIGHTLY }}' > .env
-
- - name: Generate Secrets
- run: |
- dart pub global activate flutter_distributor
- flutter pub get
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
-
- - name: Build Macos App
- run: |
- flutter config --enable-macos-desktop
- flutter build macos
- du -sh build/macos/Build/Products/Release/spotube.app
-
- - name: Package Macos App
- run: |
- python3 -m pip install setuptools
- npm install -g appdmg
- mkdir -p build/${{ env.BUILD_VERSION }}
- appdmg appdmg.json build/Spotube-macos-universal.dmg
- flutter_distributor package --platform=macos --targets pkg --skip-clean
- mv dist/**/spotube-*-macos.pkg build/Spotube-macos-universal.pkg
-
- - uses: actions/upload-artifact@v3
- with:
- if-no-files-found: error
- name: Spotube-Release-Binaries
- path: |
- build/Spotube-macos-universal.dmg
- build/Spotube-macos-universal.pkg
-
- - name: Debug With SSH When fails
- if: ${{ failure() && inputs.debug && inputs.channel == 'nightly' }}
- uses: mxschmitt/action-tmate@v3
- with:
- limit-access-to-actor: true
-
- iOS:
- runs-on: macos-latest
- steps:
- - uses: actions/checkout@v4
- - uses: subosito/flutter-action@v2.10.0
- with:
- cache: true
- flutter-version: ${{ env.FLUTTER_VERSION }}
-
- - name: Replace pubspec version and BUILD_VERSION Env (nightly)
- if: ${{ inputs.channel == 'nightly' }}
- run: |
- brew install yq
- yq -i '.version |= sub("\+\d+", "+${{ inputs.channel }}.")' pubspec.yaml
- yq -i '.version += strenv(GITHUB_RUN_NUMBER)' pubspec.yaml
- echo "BUILD_VERSION=${{ inputs.version }}+${{ inputs.channel }}.${{ github.run_number }}" >> $GITHUB_ENV
-
- - name: BUILD_VERSION Env (stable)
- if: ${{ inputs.channel == 'stable' }}
- run: |
- echo "BUILD_VERSION=${{ inputs.version }}" >> $GITHUB_ENV
-
- - name: Create Stable .env
- if: ${{ inputs.channel == 'stable' }}
- run: echo '${{ secrets.DOTENV_RELEASE }}' > .env
-
- - name: Create Nightly .env
- if: ${{ inputs.channel == 'nightly' }}
- run: echo '${{ secrets.DOTENV_NIGHTLY }}' > .env
-
- - name: Generate Secrets
- run: |
- flutter pub get
- dart run build_runner build --delete-conflicting-outputs --enable-experiment=records,patterns
-
- - name: Build iOS iPA
- run: |
- flutter build ios --release --no-codesign --flavor ${{ inputs.channel }}
- ln -sf ./build/ios/iphoneos Payload
- zip -r9 Spotube-iOS.ipa Payload/${{ inputs.channel }}.app
-
- - uses: actions/upload-artifact@v3
- with:
- if-no-files-found: error
- name: Spotube-Release-Binaries
- path: |
- Spotube-iOS.ipa
-
- - name: Debug With SSH When fails
- if: ${{ failure() && inputs.debug && inputs.channel == 'nightly' }}
- uses: mxschmitt/action-tmate@v3
- with:
- limit-access-to-actor: true
-
upload:
runs-on: ubuntu-latest
-
needs:
- - windows
- - linux
- - android
- - macos
- - iOS
+ - build_platform
steps:
+ - uses: actions/checkout@v4
- uses: actions/download-artifact@v3
with:
name: Spotube-Release-Binaries
@@ -426,6 +149,10 @@ jobs:
md5sum Spotube-Release-Binaries/* >> RELEASE.md5sum
sha256sum Spotube-Release-Binaries/* >> RELEASE.sha256sum
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
with:
@@ -435,17 +162,12 @@ jobs:
RELEASE.md5sum
RELEASE.sha256sum
- - name: Debug With SSH
- uses: mxschmitt/action-tmate@v3
- with:
- limit-access-to-actor: true
-
- name: Upload Release Binaries (stable)
if: ${{ !inputs.dry_run && inputs.channel == 'stable' }}
uses: ncipollo/release-action@v1
with:
token: ${{ secrets.GITHUB_TOKEN }}
- tag: v${{ inputs.version }} # mind the "v" prefix
+ tag: v${{ env.PUBSPEC_VERSION }} # mind the "v" prefix
omitBodyDuringUpdate: true
omitNameDuringUpdate: true
omitPrereleaseDuringUpdate: true
@@ -463,3 +185,8 @@ jobs:
omitPrereleaseDuringUpdate: true
allowUpdates: true
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
diff --git a/.gitignore b/.gitignore
index 96d81087..f9bd15f8 100644
--- a/.gitignore
+++ b/.gitignore
@@ -73,6 +73,10 @@ dist
appimage-build
android/key.properties
-.fvm/flutter_sdk
**/pb_data
+
+tm.json
+
+# FVM Version Cache
+.fvm/
\ No newline at end of file
diff --git a/.metadata b/.metadata
index 082985ad..828f2c0a 100644
--- a/.metadata
+++ b/.metadata
@@ -1,11 +1,11 @@
# This file tracks properties of this Flutter project.
# Used by Flutter tool to assess capabilities and perform upgrades etc.
#
-# This file should be version controlled.
+# This file should be version controlled and should not be manually edited.
version:
- revision: eb6d86ee27deecba4a83536aa20f366a6044895c
- channel: stable
+ revision: "300451adae589accbece3490f4396f10bdf15e6e"
+ channel: "stable"
project_type: app
@@ -13,11 +13,11 @@ project_type: app
migration:
platforms:
- platform: root
- create_revision: eb6d86ee27deecba4a83536aa20f366a6044895c
- base_revision: eb6d86ee27deecba4a83536aa20f366a6044895c
- - platform: macos
- create_revision: eb6d86ee27deecba4a83536aa20f366a6044895c
- base_revision: eb6d86ee27deecba4a83536aa20f366a6044895c
+ create_revision: 300451adae589accbece3490f4396f10bdf15e6e
+ base_revision: 300451adae589accbece3490f4396f10bdf15e6e
+ - platform: windows
+ create_revision: 300451adae589accbece3490f4396f10bdf15e6e
+ base_revision: 300451adae589accbece3490f4396f10bdf15e6e
# User provided section
diff --git a/.vscode/settings.json b/.vscode/settings.json
index 0e6a4294..11fae610 100644
--- a/.vscode/settings.json
+++ b/.vscode/settings.json
@@ -2,13 +2,22 @@
"cmake.configureOnOpen": false,
"cSpell.words": [
"acousticness",
+ "ambiguate",
+ "Amoled",
+ "Buildless",
"danceability",
+ "fuzzywuzzy",
+ "gapless",
"instrumentalness",
"Mpris",
+ "RGBO",
"riverpod",
"Scrobblenaut",
+ "skeletonizer",
+ "songlink",
"speechiness",
"Spotube",
+ "titlebar",
"winget"
],
"editor.formatOnSave": true,
@@ -16,5 +25,7 @@
"explorer.fileNesting.patterns": {
"pubspec.yaml": "pubspec.lock,analysis_options.yaml,.packages,.flutter-plugins,.flutter-plugins-dependencies,flutter_launcher_icons*.yaml,flutter_native_splash*.yaml",
"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"
}
\ No newline at end of file
diff --git a/.vscode/snippets.code-snippets b/.vscode/snippets.code-snippets
new file mode 100644
index 00000000..9a18929b
--- /dev/null
+++ b/.vscode/snippets.code-snippets
@@ -0,0 +1,170 @@
+{
+ "PaginatedState": {
+ "scope": "dart",
+ "prefix": "paginatedState",
+ "description": "Generate a PaginatedState",
+ "body": [
+ "class ${1:Model}State extends PaginatedState<${2:Model}> {",
+ " ${1:Model}State({",
+ " required super.items,",
+ " required super.offset,",
+ " required super.limit,",
+ " required super.hasMore,",
+ " });",
+ " ",
+ " @override",
+ " ${1:Model}State copyWith({",
+ " List<${2:Model}>? items,",
+ " int? offset,",
+ " int? limit,",
+ " bool? hasMore,",
+ " }) {",
+ " return ${1:Model}State(",
+ " items: items ?? this.items,",
+ " offset: offset ?? this.offset,",
+ " limit: limit ?? this.limit,",
+ " hasMore: hasMore ?? this.hasMore,",
+ " );",
+ " }",
+ "}"
+ ]
+ },
+ "PaginatedAsyncNotifier": {
+ "scope": "dart",
+ "prefix": "paginatedAsyncNotifier",
+ "description": "Generate a PaginatedAsyncNotifier",
+ "body": [
+ "class ${1:NotifierName}Notifier extends PaginatedAsyncNotifier<${3:Item}, ${2:Model}State> {",
+ " ${1:NotifierName}Notifier() : super();",
+ " ",
+ " @override",
+ " fetch(int offset, int limit) async {",
+ " throw UnimplementedError();",
+ " }",
+ " ",
+ " @override",
+ " build() async {",
+ " throw UnimplementedError();",
+ " }",
+ "}"
+ ]
+ },
+ "PaginaitedNotifierWithState": {
+ "scope": "dart",
+ "prefix": "paginatedNotifierWithState",
+ "description": "Generate a PaginatedNotifier with PaginatedState",
+ "body": [
+ "class $1State extends PaginatedState<$2> {",
+ " $1State({",
+ " required super.items,",
+ " required super.offset,",
+ " required super.limit,",
+ " required super.hasMore,",
+ " });",
+ " ",
+ " @override",
+ " $1State copyWith({",
+ " List<$2>? items,",
+ " int? offset,",
+ " int? limit,",
+ " bool? hasMore,",
+ " }) {",
+ " return $1State(",
+ " items: items ?? this.items,",
+ " offset: offset ?? this.offset,",
+ " limit: limit ?? this.limit,",
+ " hasMore: hasMore ?? this.hasMore,",
+ " );",
+ " }",
+ "}",
+ " ",
+ "class $1Notifier",
+ " extends PaginatedAsyncNotifier<$2, $1State> {",
+ " $1Notifier() : super();",
+ " ",
+ " @override",
+ " fetch(int offset, int limit) async {",
+ " throw UnimplementedError();",
+ " }",
+ " ",
+ " @override",
+ " build() async {",
+ " throw UnimplementedError();",
+ " }",
+ "}",
+ " ",
+ "final ${1/(.*)/${1:/camelcase}/}Provider = AsyncNotifierProvider<$1Notifier, $1State>(",
+ " ()=> $1Notifier(),",
+ ");"
+ ]
+ },
+ "FamilyPaginatedAsyncNotifier": {
+ "scope": "dart",
+ "prefix": "familyPaginatedAsyncNotifier",
+ "description": "Generate a FamilyPaginatedAsyncNotifier",
+ "body": [
+ "class ${1:NotifierName}Notifier extends FamilyPaginatedAsyncNotifier<${3:Item}, ${2:Model}State, {$4:Arg}> {",
+ " ${1:NotifierName}Notifier() : super();",
+ " ",
+ " @override",
+ " fetch(arg, offset, limit) async {",
+ " throw UnimplementedError();",
+ " }",
+ " ",
+ " @override",
+ " build(arg) async {",
+ " throw UnimplementedError();",
+ " }",
+ "}"
+ ]
+ },
+ "FamilyPaginaitedNotifierWithState": {
+ "scope": "dart",
+ "prefix": "familyPaginatedNotifierWithState",
+ "description": "Generate a FamilyPaginatedAsyncNotifier with PaginatedState",
+ "body": [
+ "class $1State extends PaginatedState<$2> {",
+ " $1State({",
+ " required super.items,",
+ " required super.offset,",
+ " required super.limit,",
+ " required super.hasMore,",
+ " });",
+ " ",
+ " @override",
+ " $1State copyWith({",
+ " List<$2>? items,",
+ " int? offset,",
+ " int? limit,",
+ " bool? hasMore,",
+ " }) {",
+ " return $1State(",
+ " items: items ?? this.items,",
+ " offset: offset ?? this.offset,",
+ " limit: limit ?? this.limit,",
+ " hasMore: hasMore ?? this.hasMore,",
+ " );",
+ " }",
+ "}",
+ " ",
+ "class $1Notifier",
+ " extends FamilyPaginatedAsyncNotifier<$2, $1State, $3> {",
+ " $1Notifier() : super();",
+ " ",
+ " @override",
+ " fetch(arg, offset, limit) async {",
+ " throw UnimplementedError();",
+ " }",
+ " ",
+ " @override",
+ " build(arg) async {",
+ " throw UnimplementedError();",
+ " }",
+ "}",
+ " ",
+ "final ${1/(.*)/${1:/camelcase}/}Provider = AsyncNotifierProviderFamily<$1Notifier, $1State, $3>(",
+ " ()=> $1Notifier(),",
+ ");"
+ ]
+ },
+}
\ No newline at end of file
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 8f48b39e..11b06ed0 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -2,6 +2,191 @@
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.
+## [3.8.3](https://github.com/krtirtho/spotube/compare/v3.8.2...v3.8.3) (2024-10-09)
+
+## Changes
+
+### Bug Fixes
+
+- update youtube_explode_dart to 2.2.3 to fix no playback (#1980)
+
+### Features
+
+- **macos**: enable same window webview support
+
+## [3.8.2](https://github.com/krtirtho/spotube/compare/v3.8.1...v3.8.2) (2024-09-30)
+
+## Changes
+
+### Bug Fixes
+
+- 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
+
+### Bug Fixes
+
+- **translations**: correct some basque incorrect translations (#1815)
+- **lyrics**: LRCLIB lyrics should be usable without logging in #1803
+- playlist displaying descriptions unescaped html #1784
+- **android**: pressing back while the player is open doesn't take to previous page
+- handle dublicated items in playback queue correctly #1852
+- **desktop**: scrollbar overlapping with more options of tracks and playlists
+- **discord**: stop discord rpc from try update presence when not connected
+- **stats**: minutes page shows plays and streams page shows minutes which should be the opposite #1880
+- **android**: clears queue upon swiping away notification
+- **player**: shuffle button state resets after closing page #1657
+- getting started page login page exception #1800
+- **mobile**: queue doesn't persist
+- local tracks takes time to load
+- start radio not working #1629
+
+### Features
+
+- **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
+
+- translations: make state page's hard coded strings translatable (#1719)
+- discord: add listening activity type
+- discord: album art, playing time and play pause support (#1765)
+- linux: Use XDG_STATE_HOME to storage logs (#1675)
+- discord rpc for macOS, windows-arm64 and linux-arm64 (#1713)
+- desktop: implement webview based login
+- stats: add lazy loading support
+
+### Bug Fixes
+
+- translations: fix Russian translations (#1696)
+- ios: permission exception
+- linux: tray icon wrong name for flatpak
+- windows: app crashes when no internet
+- windows: local tracks plays but disabled playback controls
+- go to track album shows up for local tracks
+- local track metadata timeout
+- windows: window stretching #1553
+- android: app getting killed from background
+- linux: OS Media control not working for Flatpak #1627
+- incorrect datatype used for MPRIS position property #1521
+- Too many artists for a track causing overflows
+- playlist share button does not work #1639
+- unescape html escape values #1300
+- lyrics page doesn't scroll to top after song ends #885
+- changed source doesn't get saved and uses the wrong once again
+- null exception in album page navigated from /home
+- 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))
+
+## [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))
+
+
+### 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))
+
+## [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)
+
+
+### 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))
+
+## [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))
+
+
+### 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))
+
## [3.4.1](https://personal.github.com/krtirtho/spotube/compare/v3.4.0...v3.4.1) (2024-01-27)
diff --git a/CONTRIBUTION.md b/CONTRIBUTION.md
index 13996cea..d4746a1a 100644
--- a/CONTRIBUTION.md
+++ b/CONTRIBUTION.md
@@ -25,14 +25,14 @@ All types of contributions are encouraged and valued. See the [Table of Contents
- [Before Submitting an Enhancement](#before-submitting-an-enhancement)
- [How Do I Submit a Good Enhancement Suggestion?](#how-do-i-submit-a-good-enhancement-suggestion)
- [Your First Code Contribution](#your-first-code-contribution)
- - [Submit translations](#submit-translations)
+ - [Submit Translations](#submit-translations)
## Code of Conduct
This project and everyone participating in it is governed by the
[Spotube Code of Conduct](https://github.com/KRTirtho/spotube/blob/master/CODE_OF_CONDUCT.md).
By participating, you are expected to uphold this code. Please report unacceptable behavior
-to <>.
+to krtirtho@gmail.com.
## I Have a Question
@@ -123,16 +123,16 @@ Do the following:
- Install Development dependencies in linux
- Debian (>=12/Bookworm)/Ubuntu
```bash
- $ apt-get install mpv libmpv-dev libappindicator3-1 gir1.2-appindicator3-0.1 libappindicator3-dev libsecret-1-0 libjsoncpp25 libsecret-1-dev libjsoncpp-dev libnotify-bin libnotify-dev
+ $ apt-get install mpv libmpv-dev libappindicator3-1 gir1.2-appindicator3-0.1 libappindicator3-dev libsecret-1-0 libjsoncpp25 libsecret-1-dev libjsoncpp-dev libnotify-bin libnotify-dev avahi-daemon avahi-discover avahi-utils libnss-mdns mdns-scan libwebkit2gtk-4.1-0 libwebkit2gtk-4.1-dev libsoup-3.0-0 libsoup-3.0-dev
```
- Use `libjsoncpp1` instead of `libjsoncpp25` (for Ubuntu < 22.04)
- Arch/Manjaro
```bash
- yay -S mpv libappindicator-gtk3 libsecret jsoncpp libnotify
+ yay -S mpv libappindicator-gtk3 libsecret jsoncpp libnotify avahi nss-mdns mdns-scan webkit2gtk-4.1 libsoup3
```
- Fedora
```bash
- dnf install mpv mpv-devel libappindicator-gtk3 libappindicator-gtk3-devel libsecret libsecret-devel jsoncpp jsoncpp-devel libnotify libnotify-devel
+ dnf install mpv mpv-devel libappindicator-gtk3 libappindicator-gtk3-devel libsecret libsecret-devel jsoncpp jsoncpp-devel libnotify libnotify-devel avahi mdns-scan nss-mdns webkit2gtk4.1 webkit2gtk4.1-devel libsoup3 libsoup3-devel
```
- Clone the Repo
- Create a `.env` in root of the project following the `.env.example` template
diff --git a/README.md b/README.md
index 469d03ac..71c879ba 100644
--- a/README.md
+++ b/README.md
@@ -7,7 +7,7 @@ eliminating the need for Spotify Premium
Btw it's not just another Electron app 😉
-
+
@@ -97,12 +97,7 @@ This handy table lists all the methods you can use to install Spotube:
AppImage
-
-
-
-
- Note: AppimageLauncher is required!
-
+ AppImage's lacking stability led to it's temporal removal. More information at https://github.com/KRTirtho/spotube/issues/1082
Debian/Ubuntu
@@ -204,6 +199,8 @@ If you are concerned, you can [read the reason of choosing this license](https:/
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. [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
@@ -213,106 +210,116 @@ If you are concerned, you can [read the reason of choosing this license](https:/
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_mpris](https://github.com/bdrazhzhov/audio-service-mpris) - audio_service platform interface supporting Media Player Remote Interfacing Specification.
1. [audio_service](https://pub.dev/packages/audio_service) - Flutter plugin to play audio in the background while the screen is off.
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_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. [build_runner](https://pub.dev/packages/build_runner) - A build system for Dart code generation and modular compilation.
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. [catcher_2](https://github.com/ThexXTURBOXx/catcher_2) - Plugin for error catching which provides multiple handlers for dealing with errors when they are not caught by the developer.
1. [collection](https://pub.dev/packages/collection) - Collections and utilities functions and classes related to collections.
-1. [cupertino_icons](https://pub.dev/packages/cupertino_icons) - Default icons asset for Cupertino widgets based on Apple styled icons
+1. [crypto](https://pub.dev/packages/crypto) - Implementations of SHA, MD5, and HMAC cryptographic functions.
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. [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. [dart_discord_rpc](https://github.com/alexmercerind/dart_discord_rpc) - Discord Rich Presence for Flutter & Dart apps & games.
1. [dbus](https://github.com/canonical/dbus.dart) - A native Dart implementation of the D-Bus message bus client. This package allows Dart applications to directly access services on the Linux desktop.
1. [device_info_plus](https://plus.fluttercommunity.dev/) - Flutter plugin providing detailed information about the device (make, model, etc.), and Android or iOS version the app is running on.
-1. [device_preview](https://github.com/aloisdeniel/flutter_device_preview) - Approximate how your Flutter app looks and performs on another device.
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. [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. [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. [envied_generator](https://github.com/petercinibulk/envied) - Generator for the Envied package. See https://pub.dev/packages/envied.
1. [envied](https://github.com/petercinibulk/envied) - Explicitly reads environment variables into a dart file from a .env file for more security and faster start up times.
+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. [fl_query](https://fl-query.krtirtho.dev) - Asynchronous data caching, refetching & invalidation library for Flutter
-1. [fl_query_hooks](https://fl-query.krtirtho.dev) - Elite flutter_hooks compatible library for fl_query, the Asynchronous data caching, refetching & invalidation library for Flutter
-1. [fl_query_devtools](https://fl-query.krtirtho.dev) - Devtools support for Fl-Query
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_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_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_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_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. [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 simple way to access state from anywhere in your application while robust and testable.
+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. [freezed_annotation](https://pub.dev/packages/freezed_annotation) - Annotations for the freezed code-generator. This package does nothing without freezed too.
+1. [freezed](https://pub.dev/packages/freezed) - Code generation for immutable classes that has a simple syntax/API without compromising on the features.
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. [hooks_riverpod](https://riverpod.dev) - A simple way to access state from anywhere in your application while robust and testable.
+1. [hive_generator](https://github.com/hivedb/hive/tree/master/hive_generator) - Extension for Hive. Automatically generates TypeAdapters to store any class.
+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. [hooks_riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze.
+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. [html](https://pub.dev/packages/html) - APIs for parsing and manipulating HTML content outside the browser.
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. [introduction_screen](https://github.com/pyozer/introduction_screen) - Introduction/Onboarding package for flutter app with some customizations possibilities
+1. [introduction_screen](https://pub.dev/packages/introduction_screen) - Introduction/Onboarding package for flutter app with some customizations possibilities
+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. [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. [json_serializable](https://pub.dev/packages/json_serializable) - Automatically generate code for converting to and from JSON by annotating Dart classes.
+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. [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. [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_libs_audio](https://github.com/media-kit/media-kit.git) - package:media_kit audio (only) playback native libraries for all platforms.
+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. [metadata_god](https://github.com/KRTirtho/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. [package_info_plus](https://plus.fluttercommunity.dev/) - Flutter plugin for querying information about the application package, such as CFBundleVersion on iOS or versionCode on Android.
1. [palette_generator](https://pub.dev/packages/palette_generator) - Flutter package for generating palette colors from a source image.
-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. [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. [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. [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. [pub_api_client](https://github.com/leoafarias/pub_api_client) - An API Client for Pub to interact with public package information.
+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. [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. [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. [scroll_to_index](https://github.com/quire-io/scroll-to-index) - Scroll to a specific child of any scrollable widget in Flutter
-1. [sidebarx](https://github.com/Frezyx/sidebarx) - flutter multiplatform navigation sidebar / side navigationbar / drawer widget
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_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. [shelf](https://pub.dev/packages/shelf) - A model for web server middleware that encourages composition and easy reuse.
+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. [skeleton_text](https://github.com/101Loop/Skeleton-Text) - A package that provides an easy way to add skeleton text loading animation in Flutter project. This project is a part of 101Loop community.
+1. [skeletonizer](https://github.com/Milad-Akarie/skeletonizer) - Converts already built widgets into skeleton loaders with no extra effort.
+1. [sliver_tools](https://github.com/Kavantix) - A set of useful sliver tools that are missing from the flutter framework
1. [smtc_windows](https://github.com/KRTirtho/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. [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://pub.dev/packages/system_theme) - A plugin to get the current system theme info. Supports Android, Web, Windows, Linux and macOS
+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.
1. [url_launcher](https://pub.dev/packages/url_launcher) - Flutter plugin for launching a URL. Supports web, phone, SMS, and email schemes.
1. [uuid](https://pub.dev/packages/uuid) - RFC4122 (v1, v4, v5, v6, v7, v8) UUID Generator and Parser for Dart
1. [version](https://github.com/dartninja/version) - Provides a simple class for parsing and comparing semantic versions as defined by http://semver.org/
-1. [visibility_detector](https://pub.dev/packages/visibility_detector) - A widget that detects the visibility of its child and notifies a callback.
-1. [window_manager](https://github.com/leanflutter/window_manager) - 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. [simple_icons](https://teavelopment.com/) - The Simple Icon pack available as Flutter Icons. Provides over 1500 Free SVG icons for popular brands.
-1. [audio_service_mpris](https://github.com/bdrazhzhov/audio-service-mpris) - audio_service platform interface supporting Media Player Remote Interfacing Specification.
-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. [jiosaavn](https://github.com/KRTirtho/jiosaavn) - Unofficial API client for jiosaavn.com
1. [very_good_infinite_list](https://github.com/VeryGoodOpenSource/very_good_infinite_list) - A library for easily displaying paginated data, created by Very Good Ventures. Great for activity feeds, news feeds, and more.
-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. [sliver_tools](https://github.com/Kavantix) - A set of useful sliver tools that are missing from the flutter framework
-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. [visibility_detector](https://pub.dev/packages/visibility_detector) - A widget that detects the visibility of its child and notifies a callback.
+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. [skeletonizer](https://github.com/Milad-Akarie/skeletonizer) - Converts already built widgets into skeleton loaders with no extra effort.
-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. [win32_registry](https://win32.pub) - A package that provides a friendly Dart API for accessing the Windows Registry.
-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. [build_runner](https://pub.dev/packages/build_runner) - A build system for Dart code generation and modular compilation.
-1. [envied_generator](https://github.com/petercinibulk/envied) - Generator for the Envied package. See https://pub.dev/packages/envied.
-1. [flutter_distributor](https://distributor.leanflutter.org) - A complete tool for packaging and publishing your Flutter apps.
-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. [pub_api_client](https://github.com/leoafarias/pub_api_client) - An API Client for Pub to interact with public package information.
-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. [flutter_desktop_tools](https://github.com/KRTirtho/flutter_desktop_tools) - Essential collection of tools for flutter desktop app development
-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. [window_size](https://github.com/google/flutter-desktop-embedding.git) - Allows resizing and repositioning the window containing Flutter.
-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. [dart_discord_rpc](https://github.com/alexmercerind/dart_discord_rpc) - Discord Rich Presence for Flutter & Dart apps & games.
+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. [xml](https://github.com/renggli/dart-xml) - A lightweight library for parsing, traversing, querying, transforming and building XML documents.
+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.
© Copyright Spotube 2024
diff --git a/analysis_options.yaml b/analysis_options.yaml
index 5f2cbbe1..d5b904cc 100644
--- a/analysis_options.yaml
+++ b/analysis_options.yaml
@@ -25,12 +25,17 @@ linter:
# avoid_print: false # Uncomment to disable the `avoid_print` rule
# prefer_single_quotes: true # Uncomment to enable the `prefer_single_quotes` rule
file_names: false
+ avoid_renaming_method_parameters: false
# Additional information about this file can be found at
# https://dart.dev/guides/language/analysis-options
analyzer:
- enable-experiment:
- - records
- - patterns
errors:
invalid_annotation_target: ignore
+ plugins:
+ - custom_lint
+ exclude:
+ - "**.freezed.dart"
+ - "**.g.dart"
+ - "**.gr.dart"
+ - "**/generated_plugin_registrant.dart"
diff --git a/android/app/build.gradle b/android/app/build.gradle
index 2f85cdeb..8ec1872e 100644
--- a/android/app/build.gradle
+++ b/android/app/build.gradle
@@ -1,3 +1,9 @@
+plugins {
+ id "com.android.application"
+ id "kotlin-android"
+ id "dev.flutter.flutter-gradle-plugin"
+}
+
def localProperties = new Properties()
def localPropertiesFile = rootProject.file('local.properties')
if (localPropertiesFile.exists()) {
@@ -6,11 +12,6 @@ if (localPropertiesFile.exists()) {
}
}
-def flutterRoot = localProperties.getProperty('flutter.sdk')
-if (flutterRoot == null) {
- throw new GradleException("Flutter SDK not found. Define location with flutter.sdk in the local.properties file.")
-}
-
def flutterVersionCode = localProperties.getProperty('flutter.versionCode')
if (flutterVersionCode == null) {
flutterVersionCode = '1'
@@ -21,10 +22,6 @@ if (flutterVersionName == null) {
flutterVersionName = '1.0'
}
-apply plugin: 'com.android.application'
-apply plugin: 'kotlin-android'
-apply from: "$flutterRoot/packages/flutter_tools/gradle/flutter.gradle"
-
def keystoreProperties = new Properties()
def keystorePropertiesFile = rootProject.file('key.properties')
if (keystorePropertiesFile.exists()) {
@@ -34,7 +31,7 @@ if (keystorePropertiesFile.exists()) {
android {
compileSdkVersion 34
- ndkVersion "21.4.7075529"
+ ndkVersion "25.1.8937393"
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
@@ -50,10 +47,9 @@ android {
}
defaultConfig {
- // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
applicationId "oss.krtirtho.spotube"
minSdkVersion 24
- targetSdkVersion flutter.targetSdkVersion
+ targetSdkVersion 34
versionCode flutterVersionCode.toInteger()
versionName flutterVersionName
multiDexEnabled true
@@ -71,6 +67,9 @@ android {
release {
signingConfig signingConfigs.release
}
+ debug {
+ signingConfig signingConfigs.release
+ }
}
flavorDimensions "default"
@@ -81,16 +80,19 @@ android {
resValue "string", "app_name_en", "Spotube Nightly"
applicationIdSuffix ".nightly"
versionNameSuffix "-nightly"
+ signingConfig signingConfigs.release
}
dev {
dimension "default"
resValue "string", "app_name_en", "Spotube Dev"
applicationIdSuffix ".dev"
versionNameSuffix "-dev"
+ signingConfig signingConfigs.release
}
stable {
dimension "default"
resValue "string", "app_name_en", "Spotube"
+ signingConfig signingConfigs.release
}
}
@@ -101,15 +103,6 @@ flutter {
}
dependencies {
- implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
- constraints {
- implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version") {
- because("kotlin-stdlib-jdk7 is now a part of kotlin-stdlib")
- }
- implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8:$kotlin_version") {
- because("kotlin-stdlib-jdk8 is now a part of kotlin-stdlib")
- }
- }
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.5.1'
// other deps so just ignore
diff --git a/android/app/src/main/AndroidManifest.xml b/android/app/src/main/AndroidManifest.xml
index 5ab7a0b5..64c32e28 100644
--- a/android/app/src/main/AndroidManifest.xml
+++ b/android/app/src/main/AndroidManifest.xml
@@ -3,6 +3,7 @@
+
@@ -24,6 +25,11 @@
android:usesCleartextTraffic="true"
android:requestLegacyExternalStorage="true"
>
+
+
+
properties.load(reader) }
+ repositories {
+ google()
+ mavenCentral()
+ gradlePluginPortal()
+ }
+}
-def flutterSdkPath = properties.getProperty("flutter.sdk")
-assert flutterSdkPath != null, "flutter.sdk not set in local.properties"
-apply from: "$flutterSdkPath/packages/flutter_tools/gradle/app_plugin_loader.gradle"
+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
+}
+
+include ":app"
\ No newline at end of file
diff --git a/assets/spotube-logo.bmp b/assets/spotube-logo.bmp
new file mode 100644
index 00000000..c3503e85
Binary files /dev/null and b/assets/spotube-logo.bmp differ
diff --git a/aur-struct/.SRCINFO b/aur-struct/.SRCINFO
index ae0b6d10..4c07a045 100644
--- a/aur-struct/.SRCINFO
+++ b/aur-struct/.SRCINFO
@@ -1,17 +1,18 @@
pkgbase = spotube-bin
- pkgdesc = Open source Spotify client that doesn't require Premium nor uses Electron! Available for both desktop & mobile!
- pkgver = 2.3.0
- pkgrel = 1
- url = https://github.com/KRTirtho/spotube/
- arch = x86_64
- license = BSD-4-Clause
- depends = mpv
- depends = libappindicator-gtk3
- depends = libsecret
- depends = jsoncpp
- depends = libnotify
- depends = xdg-user-dirs
- source = https://github.com/KRTirtho/spotube/releases/download/v2.3.0/Spotube-linux-x86_64.tar.xz
- md5sums = 8cd6a7385c5c75d203dccd762f1d63ec
+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/
+arch = x86_64
+license = BSD-4-Clause
+depends = mpv
+depends = libappindicator-gtk3
+depends = libsecret
+depends = jsoncpp
+depends = libnotify
+depends = xdg-user-dirs
+depends = webkit2gtk-4.1
+source = https://github.com/KRTirtho/spotube/releases/download/v3.7.1/spotube-linux-3.7.1-x86_64.tar.xz
+md5sums = 475b1ae9b08f27743a4d4749391ae3db
pkgname = spotube-bin
diff --git a/aur-struct/PKGBUILD b/aur-struct/PKGBUILD
index 4663c3ab..d7e1052b 100644
--- a/aur-struct/PKGBUILD
+++ b/aur-struct/PKGBUILD
@@ -8,7 +8,7 @@ arch=(x86_64)
url="https://github.com/KRTirtho/spotube/"
license=('BSD-4-Clause')
groups=()
-depends=('mpv' 'libappindicator-gtk3' 'libsecret' 'jsoncpp' 'libnotify' 'xdg-user-dirs')
+depends=('mpv' 'libappindicator-gtk3' 'libsecret' 'jsoncpp' 'libnotify' 'xdg-user-dirs' 'webkit2gtk-4.1')
makedepends=()
checkdepends=()
optdepends=()
diff --git a/bin/gen-credits.dart b/bin/gen-credits.dart
deleted file mode 100644
index f8975335..00000000
--- a/bin/gen-credits.dart
+++ /dev/null
@@ -1,103 +0,0 @@
-import 'dart:developer';
-import 'dart:io';
-
-import 'package:collection/collection.dart';
-import 'package:http/http.dart';
-import 'package:html/parser.dart';
-import 'package:pub_api_client/pub_api_client.dart';
-import 'package:pubspec_parse/pubspec_parse.dart';
-
-void main() async {
- final client = PubClient();
-
- final pubspec = Pubspec.parse(File('pubspec.yaml').readAsStringSync());
-
- final allDeps = [
- ...pubspec.dependencies.entries,
- ...pubspec.devDependencies.entries,
- ];
-
- final dependencies = allDeps
- .where((d) => d.value is HostedDependency)
- .map((d) => d.key)
- .toSet();
- final packageInfo = await Future.wait(dependencies.map(client.packageInfo));
-
- final gitDepsList = List.castFrom,
- MapEntry>(
- allDeps
- .where((d) => d.value is GitDependency)
- .map((d) => MapEntry(d.key, d.value as GitDependency))
- .toList(),
- );
-
- final gitDeps = gitDepsList.map(
- (d) {
- final uri = Uri.parse(
- d.value.url.toString().replaceAll('.git', ''),
- );
- return MapEntry(
- d.key,
- uri.replace(
- pathSegments: [
- ...uri.pathSegments,
- 'raw',
- d.value.ref ?? 'main',
- d.value.path ?? '',
- 'pubspec.yaml',
- ],
- ).toString(),
- );
- },
- ).toList();
-
- final gitPubspecs = await Future.wait(
- gitDeps.map(
- (d) {
- Pubspec parser(res) {
- try {
- return Pubspec.parse(res.body);
- } catch (e) {
- final document = parse(res.body);
- final pre = document.querySelector('pre');
- if (pre == null) {
- log(d.toString());
- rethrow;
- }
- return Pubspec.parse(pre.text);
- }
- }
-
- return get(Uri.parse(d.value)).then(parser).catchError(
- (_) => get(Uri.parse(d.value.replaceFirst('/main', '/master')))
- .then(parser),
- );
- },
- ),
- );
-
- // ignore: avoid_print
- print(
- packageInfo
- .map(
- (package) =>
- '1. [${package.name}](${package.latestPubspec.homepage ?? package.url}) - ${package.description.replaceAll('\n', '')}',
- )
- .join('\n'),
- );
- // ignore: avoid_print
- print(
- gitPubspecs.map(
- (package) {
- final packageUrl = package.homepage ??
- gitDepsList
- .firstWhereOrNull((dep) => dep.key == package.name)
- ?.value
- .url
- .toString();
- return '1. [${package.name}]($packageUrl) - ${package.description?.replaceAll('\n', '')}';
- },
- ).join('\n'),
- );
- exit(0);
-}
diff --git a/bin/untranslated_messages.dart b/bin/untranslated_messages.dart
deleted file mode 100644
index e19f9a07..00000000
--- a/bin/untranslated_messages.dart
+++ /dev/null
@@ -1,49 +0,0 @@
-import 'dart:convert';
-import 'dart:io';
-
-/// Generate JSON output for untranslated messages with English values
-/// for quick translation in ChatGPT
-///
-/// Usage: dart bin/untranslated_messages.dart [locale?]
-///
-/// Example: dart bin/untranslated_messages.dart
-///
-/// or with specific locale (e.g. bn (Bengali))
-///
-/// Example: dart bin/untranslated_messages.dart bn
-
-void main(List args) {
- final file = jsonDecode(
- File('untranslated_messages.json').readAsStringSync(),
- ) as Map;
-
- final englishMessages =
- jsonDecode(File('lib/l10n/app_en.arb').readAsStringSync())
- as Map;
-
- final messagesWithValues = {};
-
- for (final MapEntry(key: locale, value: messages) in file.entries) {
- messagesWithValues[locale] = Map.fromEntries(
- messages
- .map(
- (message) =>
- MapEntry(message, englishMessages[message]),
- )
- .toList()
- .cast>(),
- );
- }
-
- print(
- "Prompt:\n"
- "Translate following to their appropriate locale for flutter arb translations files."
- " Put the respective new translations in a map of their corresponding locale.",
- );
- // ignore: avoid_print
- print(
- const JsonEncoder.withIndent(' ').convert(
- args.isNotEmpty ? messagesWithValues[args.first] : messagesWithValues,
- ),
- );
-}
diff --git a/bin/verify-pkgbuild.dart b/bin/verify-pkgbuild.dart
deleted file mode 100644
index 587e63d0..00000000
--- a/bin/verify-pkgbuild.dart
+++ /dev/null
@@ -1,22 +0,0 @@
-import 'dart:convert';
-import 'dart:io';
-
-void main() {
- Process.run("sh", ["-c", '"./scripts/pkgbuild2json.sh aur-struct/PKGBUILD"'])
- .then((result) {
- try {
- final pkgbuild = jsonDecode(result.stdout);
- if (pkgbuild["version"] !=
- Platform.environment["RELEASE_VERSION"]?.substring(1)) {
- throw Exception(
- "PKGBUILD version doesn't match current RELEASE_VERSION");
- }
- if (pkgbuild["release"] != "1") {
- throw Exception("In new releases pkgrel should be 1");
- }
- } catch (e) {
- // ignore: avoid_print
- print("[Failed to parse PKGBUILD] $e");
- }
- });
-}
diff --git a/build.yaml b/build.yaml
index f074d6e1..8dbfe45d 100644
--- a/build.yaml
+++ b/build.yaml
@@ -2,4 +2,16 @@ targets:
$default:
sources:
exclude:
- - bin/*.dart
\ No newline at end of file
+ - bin/*.dart
+ builders:
+ json_serializable:
+ options:
+ any_map: true
+ explicit_to_json: true
+ drift_dev:
+ options:
+ sql:
+ dialect: sqlite
+ options:
+ modules:
+ - json1
diff --git a/cli/README.md b/cli/README.md
new file mode 100644
index 00000000..b2ba8ebd
--- /dev/null
+++ b/cli/README.md
@@ -0,0 +1,4 @@
+## Spotube Configuration CLI
+
+This is used for building the project for multiple platforms and having utilities specific for the project.
+Written in Dart
diff --git a/cli/cli.dart b/cli/cli.dart
new file mode 100644
index 00000000..26190d4c
--- /dev/null
+++ b/cli/cli.dart
@@ -0,0 +1,22 @@
+import 'package:args/command_runner.dart';
+
+import 'commands/build.dart';
+import 'commands/credits.dart';
+import 'commands/install-dependencies.dart';
+import 'commands/translated.dart';
+import 'commands/untranslated.dart';
+
+void main(List args) {
+ final commandRunner = CommandRunner(
+ "cli",
+ "Configuration CLI for Spotube",
+ );
+
+ commandRunner.addCommand(InstallDependenciesCommand());
+ commandRunner.addCommand(BuildCommand());
+ commandRunner.addCommand(CreditsCommand());
+ commandRunner.addCommand(TranslatedCommand());
+ commandRunner.addCommand(UntranslatedCommand());
+
+ commandRunner.run(args);
+}
diff --git a/cli/commands/build.dart b/cli/commands/build.dart
new file mode 100644
index 00000000..fdf35a95
--- /dev/null
+++ b/cli/commands/build.dart
@@ -0,0 +1,25 @@
+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';
+
+class BuildCommand extends Command {
+ @override
+ String get description => "Build for different platforms";
+
+ @override
+ String get name => "build";
+
+ BuildCommand() {
+ addSubcommand(AndroidBuildCommand());
+ addSubcommand(IosBuildCommand());
+ addSubcommand(LinuxBuildCommand());
+ addSubcommand(LinuxArmBuildCommand());
+ addSubcommand(MacosBuildCommand());
+ addSubcommand(WindowsBuildCommand());
+ }
+}
diff --git a/cli/commands/build/android.dart b/cli/commands/build/android.dart
new file mode 100644
index 00000000..4216553a
--- /dev/null
+++ b/cli/commands/build/android.dart
@@ -0,0 +1,92 @@
+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';
+
+class AndroidBuildCommand extends Command with BuildCommandCommonSteps {
+ @override
+ String get description => "Build for android";
+
+ @override
+ String get name => "android";
+
+ @override
+ FutureOr? run() async {
+ await bootstrap();
+
+ await shell.run(
+ "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",
+ "app",
+ "outputs",
+ "flutter-apk",
+ "app-${CliEnv.channel.name}-release.apk",
+ ),
+ );
+
+ await ogApkFile.copy(
+ 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
new file mode 100644
index 00000000..4c7e3e51
--- /dev/null
+++ b/cli/commands/build/common.dart
@@ -0,0 +1,66 @@
+import 'dart:io';
+
+import 'package:args/command_runner.dart';
+import 'package:path/path.dart';
+import 'package:process_run/shell_run.dart';
+import 'package:pubspec_parse/pubspec_parse.dart';
+
+import '../../core/env.dart';
+
+mixin BuildCommandCommonSteps on Command {
+ final shell = Shell();
+ Directory get cwd => Directory.current;
+
+ Pubspec? _pubspec;
+
+ Pubspec get pubspec {
+ if (_pubspec != null) {
+ return _pubspec!;
+ }
+
+ final pubspecFile = File(join(cwd.path, "pubspec.yaml"));
+ _pubspec = Pubspec.parse(pubspecFile.readAsStringSync());
+
+ return _pubspec!;
+ }
+
+ String get versionWithoutBuildNumber {
+ return "${pubspec.version!.major}.${pubspec.version!.minor}.${pubspec.version!.patch}";
+ }
+
+ RegExp get versionVarRegExp =>
+ RegExp(r"\%\{\{SPOTUBE_VERSION\}\}\%", multiLine: true);
+
+ File get dotEnvFile => File(join(cwd.path, ".env"));
+
+ Future bootstrap() async {
+ await dotEnvFile.create(recursive: true);
+
+ await dotEnvFile.writeAsString(
+ "${CliEnv.dotenv}\n"
+ "RELEASE_CHANNEL=${CliEnv.channel.name}\n",
+ );
+
+ if (CliEnv.channel == BuildChannel.nightly) {
+ final pubspecFile = File(join(cwd.path, "pubspec.yaml"));
+
+ pubspecFile.writeAsStringSync(
+ pubspecFile.readAsStringSync().replaceAll(
+ "version: ${pubspec.version!.canonicalizedVersion}",
+ "version: $versionWithoutBuildNumber+${CliEnv.ghRunNumber}",
+ ),
+ );
+
+ _pubspec = null;
+ pubspec;
+ }
+
+ await shell.run(
+ """
+ flutter pub get
+ dart run build_runner build --delete-conflicting-outputs
+ dart pub global activate flutter_distributor
+ """,
+ );
+ }
+}
diff --git a/cli/commands/build/ios.dart b/cli/commands/build/ios.dart
new file mode 100644
index 00000000..6460f9ed
--- /dev/null
+++ b/cli/commands/build/ios.dart
@@ -0,0 +1,29 @@
+import 'dart:async';
+
+import 'package:args/command_runner.dart';
+import 'package:path/path.dart';
+
+import '../../core/env.dart';
+import 'common.dart';
+
+class IosBuildCommand extends Command with BuildCommandCommonSteps {
+ @override
+ String get description => "iOS build command";
+
+ @override
+ String get name => "ios";
+
+ @override
+ FutureOr? run() async {
+ await bootstrap();
+
+ final buildDirPath = join(cwd.path, "build", "ios", "iphoneos");
+ await shell.run(
+ """
+ flutter build ios --release --no-codesign --flavor ${CliEnv.channel.name}
+ ln -sf $buildDirPath Payload
+ zip -r9 Spotube-iOS.ipa ${join("Payload", "${CliEnv.channel.name}.app")}
+ """,
+ );
+ }
+}
diff --git a/cli/commands/build/linux.dart b/cli/commands/build/linux.dart
new file mode 100644
index 00000000..a218720c
--- /dev/null
+++ b/cli/commands/build/linux.dart
@@ -0,0 +1,106 @@
+import 'dart:async';
+import 'dart:io';
+
+import 'package:io/io.dart';
+import 'package:args/command_runner.dart';
+import 'package:intl/intl.dart';
+import 'package:path/path.dart';
+
+import '../../core/env.dart';
+import 'common.dart';
+
+class LinuxBuildCommand extends Command with BuildCommandCommonSteps {
+ @override
+ String get description => "Linux build command";
+
+ @override
+ String get name => "linux";
+
+ @override
+ FutureOr? run() async {
+ stdout.writeln("Replacing versions");
+
+ final appDataFile = File(
+ join(cwd.path, "linux", "com.github.KRTirtho.Spotube.appdata.xml"),
+ );
+
+ appDataFile.writeAsStringSync(
+ appDataFile.readAsStringSync().replaceAll(
+ versionVarRegExp,
+ ' ',
+ ),
+ );
+
+ await bootstrap();
+
+ await shell.run(
+ """
+ flutter_distributor package --platform=linux --targets=deb
+ flutter_distributor package --platform=linux --targets=rpm
+ """,
+ );
+
+ final tempDir = join(Directory.systemTemp.path, "spotube-tar");
+
+ final bundleDirPath =
+ join(cwd.path, "build", "linux", "x64", "release", "bundle");
+
+ final tarFile = File(join(
+ cwd.path,
+ "dist",
+ "spotube-linux-"
+ "${CliEnv.channel == BuildChannel.nightly ? "nightly" : versionWithoutBuildNumber}"
+ "-x86_64.tar.xz",
+ ));
+
+ await copyPath(bundleDirPath, tempDir);
+ await File(join(cwd.path, "linux", "spotube.desktop")).copy(
+ join(tempDir, "spotube.desktop"),
+ );
+ await File(
+ join(cwd.path, "linux", "com.github.KRTirtho.Spotube.appdata.xml"),
+ ).copy(
+ join(tempDir, "com.github.KRTirtho.Spotube.appdata.xml"),
+ );
+ await File(join(cwd.path, "assets", "spotube-logo.png")).copy(
+ join(tempDir, "spotube-logo.png"),
+ );
+
+ await shell.run(
+ "tar -cJf ${tarFile.path} -C $tempDir .",
+ );
+
+ final ogDeb = File(
+ join(
+ cwd.path,
+ "dist",
+ pubspec.version.toString(),
+ "spotube-${pubspec.version}-linux.deb",
+ ),
+ );
+
+ final ogRpm = File(
+ join(
+ cwd.path,
+ "dist",
+ pubspec.version.toString(),
+ "spotube-${pubspec.version}-linux.rpm",
+ ),
+ );
+
+ await ogDeb.copy(
+ join(cwd.path, "dist", "Spotube-linux-x86_64.deb"),
+ );
+ await ogRpm.copy(
+ join(cwd.path, "dist", "Spotube-linux-x86_64.rpm"),
+ );
+
+ await ogDeb.delete();
+ await ogRpm.delete();
+
+ stdout.writeln("✅ Linux building done");
+ }
+}
diff --git a/cli/commands/build/linux_arm.dart b/cli/commands/build/linux_arm.dart
new file mode 100644
index 00000000..a09f0980
--- /dev/null
+++ b/cli/commands/build/linux_arm.dart
@@ -0,0 +1,37 @@
+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
new file mode 100644
index 00000000..e8f34b77
--- /dev/null
+++ b/cli/commands/build/macos.dart
@@ -0,0 +1,42 @@
+import 'dart:async';
+import 'dart:io';
+
+import 'package:args/command_runner.dart';
+import 'package:path/path.dart';
+
+import 'common.dart';
+
+class MacosBuildCommand extends Command with BuildCommandCommonSteps {
+ @override
+ String get description => "Macos Build command";
+
+ @override
+ String get name => "macos";
+
+ @override
+ FutureOr? run() async {
+ await bootstrap();
+
+ await shell.run(
+ """
+ flutter build macos
+ appdmg appdmg.json ${join(cwd.path, "build", "Spotube-macos-universal.dmg")}
+ flutter_distributor package --platform=macos --targets pkg --skip-clean
+ """,
+ );
+
+ final ogPkg = File(
+ join(
+ cwd.path,
+ "dist",
+ pubspec.version.toString(),
+ "spotube-${pubspec.version}-macos.pkg",
+ ),
+ );
+
+ await ogPkg.copy(
+ join(cwd.path, "build", "Spotube-macos-universal.pkg"),
+ );
+ await ogPkg.delete();
+ }
+}
diff --git a/cli/commands/build/windows.dart b/cli/commands/build/windows.dart
new file mode 100644
index 00000000..c44ed52f
--- /dev/null
+++ b/cli/commands/build/windows.dart
@@ -0,0 +1,119 @@
+import 'dart:io';
+
+import 'package:args/command_runner.dart';
+import 'package:path/path.dart';
+import 'package:crypto/crypto.dart';
+import 'common.dart';
+
+class WindowsBuildCommand extends Command with BuildCommandCommonSteps {
+ @override
+ String get description => "Build Windows exe";
+
+ @override
+ String get name => "windows";
+
+ Future innoDependInstall() async {
+ final innoDependencyPath = join(cwd.path, "build", "inno-depend");
+
+ await shell.run(
+ "git clone https://github.com/DomGries/InnoDependencyInstaller.git $innoDependencyPath",
+ );
+ }
+
+ @override
+ void run() async {
+ stdout.writeln("Replace versions");
+
+ final chocoFiles = [
+ join(cwd.path, "choco-struct", "tools", "VERIFICATION.txt"),
+ join(cwd.path, "choco-struct", "spotube.nuspec"),
+ ];
+
+ for (final filePath in chocoFiles) {
+ final file = File(filePath);
+ final content = file.readAsStringSync();
+ final newContent =
+ content.replaceAll(versionVarRegExp, versionWithoutBuildNumber);
+
+ file.writeAsStringSync(newContent);
+ }
+
+ await bootstrap();
+ await innoDependInstall();
+
+ final runnerRCFile = File(
+ join(cwd.path, "windows", "runner", "Runner.rc"),
+ );
+
+ runnerRCFile.writeAsStringSync(
+ runnerRCFile
+ .readAsStringSync()
+ .replaceAll("%{{SPOTUBE_VERSION}}%", versionWithoutBuildNumber)
+ .replaceAll(
+ "%{{SPOTUBE_VERSION_AS_NUMBER}}%",
+ [
+ pubspec.version!.major,
+ pubspec.version!.minor,
+ pubspec.version!.patch,
+ 0
+ ].join(","),
+ ),
+ );
+
+ await shell.run(
+ "flutter_distributor package --platform=windows --targets=exe --skip-clean",
+ );
+
+ final ogExe = File(
+ join(
+ cwd.path,
+ "dist",
+ pubspec.version.toString(),
+ "spotube-${pubspec.version}-windows-setup.exe",
+ ),
+ );
+
+ final exePath = join(cwd.path, "dist", "Spotube-windows-x86_64-setup.exe");
+
+ await ogExe.copy(exePath);
+ await ogExe.delete();
+
+ stdout.writeln("✅ Windows exe built at $exePath");
+
+ final exeFile = File(exePath);
+
+ final hash = sha256.convert(await exeFile.readAsBytes()).toString();
+
+ final chocoVerificationFile = File(chocoFiles.first);
+
+ chocoVerificationFile.writeAsStringSync(
+ chocoVerificationFile.readAsStringSync().replaceAll(
+ RegExp(r"\%\{\{WIN_SHA256\}\}\%"),
+ hash,
+ ),
+ );
+
+ await exeFile.copy(
+ join(cwd.path, "choco-struct", "tools", basename(exeFile.path)),
+ );
+
+ await shell.run(
+ "choco pack ${chocoFiles[1]} --outputdirectory ${join(cwd.path, "dist")}",
+ );
+
+ final chocoNupkg = File(
+ join(cwd.path, "dist", "spotube.$versionWithoutBuildNumber.nupkg"),
+ );
+
+ final distNupkgPath = join(
+ cwd.path,
+ "dist",
+ "Spotube-windows-x86_64.nupkg",
+ );
+
+ await chocoNupkg.copy(distNupkgPath);
+ await chocoNupkg.delete();
+
+ stdout.writeln("✅ Windows nupkg built at $distNupkgPath");
+ }
+}
diff --git a/cli/commands/credits.dart b/cli/commands/credits.dart
new file mode 100644
index 00000000..6bad7a44
--- /dev/null
+++ b/cli/commands/credits.dart
@@ -0,0 +1,121 @@
+import 'dart:io';
+
+import 'package:args/command_runner.dart';
+import 'package:collection/collection.dart';
+import 'package:dio/dio.dart';
+import 'package:html/parser.dart';
+import 'package:path/path.dart';
+import 'package:pub_api_client/pub_api_client.dart';
+import 'package:pubspec_parse/pubspec_parse.dart';
+
+class CreditsCommand extends Command {
+ final dio = Dio(
+ BaseOptions(
+ responseType: ResponseType.plain,
+ ),
+ );
+
+ @override
+ String get description => "Generate credits for used Library's authors";
+
+ @override
+ String get name => "credits";
+
+ @override
+ run() async {
+ final client = PubClient();
+ final cwd = Directory.current;
+
+ final pubspec = Pubspec.parse(
+ File(join(cwd.path, 'pubspec.yaml')).readAsStringSync(),
+ );
+
+ final allDeps = [
+ ...pubspec.dependencies.entries,
+ ...pubspec.devDependencies.entries,
+ ];
+
+ final dependencies = allDeps
+ .where((d) => d.value is HostedDependency)
+ .map((d) => d.key)
+ .toSet();
+ final packageInfo = await Future.wait(dependencies.map(client.packageInfo));
+
+ final gitDepsList = List.castFrom,
+ MapEntry>(
+ allDeps
+ .where((d) => d.value is GitDependency)
+ .map((d) => MapEntry(d.key, d.value as GitDependency))
+ .toList(),
+ );
+
+ final gitDeps = gitDepsList.map(
+ (d) {
+ final uri = Uri.parse(
+ d.value.url.toString().replaceAll('.git', ''),
+ );
+ return MapEntry(
+ d.key,
+ uri.replace(
+ pathSegments: [
+ ...uri.pathSegments,
+ 'raw',
+ d.value.ref ?? 'main',
+ d.value.path ?? '',
+ 'pubspec.yaml',
+ ],
+ ).toString(),
+ );
+ },
+ ).toList();
+
+ final gitPubspecs = await Future.wait(
+ gitDeps.map(
+ (d) {
+ Pubspec parser(Response res) {
+ try {
+ return Pubspec.parse(res.data);
+ } catch (e) {
+ final document = parse(res.data);
+ final pre = document.querySelector('pre');
+ if (pre == null) {
+ stdout.writeln(d.toString());
+ rethrow;
+ }
+ return Pubspec.parse(pre.text);
+ }
+ }
+
+ return dio.get(d.value).then(parser).catchError(
+ (_) => dio
+ .get(d.value.replaceFirst('/main', '/master'))
+ .then(parser),
+ );
+ },
+ ),
+ );
+
+ stdout.writeln(
+ packageInfo
+ .map(
+ (package) =>
+ '1. [${package.name}](${package.latestPubspec.homepage ?? package.url}) - ${package.description.replaceAll('\n', '')}',
+ )
+ .join('\n'),
+ );
+
+ stdout.writeln(
+ gitPubspecs.map(
+ (package) {
+ final packageUrl = package.homepage ??
+ gitDepsList
+ .firstWhereOrNull((dep) => dep.key == package.name)
+ ?.value
+ .url
+ .toString();
+ return '1. [${package.name}]($packageUrl) - ${package.description?.replaceAll('\n', '')}';
+ },
+ ).join('\n'),
+ );
+ }
+}
diff --git a/cli/commands/install-dependencies.dart b/cli/commands/install-dependencies.dart
new file mode 100644
index 00000000..dc519cc6
--- /dev/null
+++ b/cli/commands/install-dependencies.dart
@@ -0,0 +1,79 @@
+import 'dart:async';
+
+import 'package:args/command_runner.dart';
+import 'package:process_run/shell_run.dart';
+
+class InstallDependenciesCommand extends Command {
+ @override
+ String get description => "Install platform dependencies";
+
+ @override
+ String get name => "install-dependencies";
+
+ InstallDependenciesCommand() {
+ argParser.addOption(
+ "platform",
+ abbr: "p",
+ allowed: [
+ "windows",
+ "linux",
+ "linux_arm",
+ "macos",
+ "ios",
+ "android",
+ ],
+ mandatory: true,
+ );
+ }
+
+ @override
+ FutureOr? run() async {
+ final shell = Shell();
+
+ switch (argResults!.option("platform")) {
+ case "windows":
+ 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
+ """,
+ );
+ break;
+ case "macos":
+ await shell.run(
+ """
+ brew install python-setuptools
+ npm install -g appdmg
+ """,
+ );
+ break;
+ case "ios":
+ await shell.run(
+ """
+ rustup target add aarch64-apple-ios
+ """,
+ );
+ break;
+ case "android":
+ await shell.run(
+ """
+ sudo apt-get update -y
+ sudo apt-get install -y clang cmake ninja-build pkg-config libgtk-3-dev make python3-pip python3-setuptools patchelf desktop-file-utils libgdk-pixbuf2.0-dev fakeroot strace fuse
+ """,
+ );
+ break;
+ default:
+ break;
+ }
+ }
+}
diff --git a/cli/commands/translated.dart b/cli/commands/translated.dart
new file mode 100644
index 00000000..43c4ea49
--- /dev/null
+++ b/cli/commands/translated.dart
@@ -0,0 +1,39 @@
+import 'dart:async';
+
+import 'dart:convert';
+import 'dart:io';
+import 'package:args/command_runner.dart';
+import 'package:path/path.dart';
+
+class TranslatedCommand extends Command {
+ @override
+ String get description =>
+ "Update translation based on generated translated messages";
+
+ @override
+ String get name => "translated";
+
+ @override
+ FutureOr? run() async {
+ final cwd = Directory.current;
+ final translatedFile = jsonDecode(
+ await File(join(cwd.path, 'tm.json')).readAsString(),
+ ) as Map;
+
+ for (final MapEntry(:key, :value) in translatedFile.entries) {
+ stdout.writeln('Updating locale: $key');
+ final file = File(join(cwd.path, 'lib', 'l10n', 'app_$key.arb'));
+
+ final fileContent =
+ jsonDecode(await file.readAsString()) as Map;
+
+ final newContent = {...fileContent, ...value};
+
+ await file.writeAsString(
+ const JsonEncoder.withIndent(' ').convert(newContent),
+ );
+
+ stdout.writeln('✅ Updated locale: $key');
+ }
+ }
+}
diff --git a/cli/commands/untranslated.dart b/cli/commands/untranslated.dart
new file mode 100644
index 00000000..dadcd8b5
--- /dev/null
+++ b/cli/commands/untranslated.dart
@@ -0,0 +1,48 @@
+import 'package:args/command_runner.dart';
+import 'dart:convert';
+import 'dart:io';
+
+import 'package:path/path.dart';
+
+class UntranslatedCommand extends Command {
+ @override
+ get name => "untranslated";
+ @override
+ get description =>
+ "Generate Untranslated Messages for ChatGPT based Translation";
+
+ @override
+ run() async {
+ final cwd = Directory.current;
+ final file = jsonDecode(
+ File(join(cwd.path, 'untranslated_messages.json')).readAsStringSync(),
+ ) as Map;
+
+ final englishMessages = jsonDecode(
+ File(join(cwd.path, 'lib', 'l10n', 'app_en.arb')).readAsStringSync(),
+ ) as Map;
+
+ final messagesWithValues = {};
+
+ for (final MapEntry(key: locale, value: messages) in file.entries) {
+ messagesWithValues[locale] = Map.fromEntries(
+ messages
+ .map(
+ (message) =>
+ MapEntry(message, englishMessages[message]),
+ )
+ .toList()
+ .cast>(),
+ );
+ }
+
+ stdout.writeln(
+ "Prompt:\n"
+ "Translate following to their appropriate locale for flutter arb translations files."
+ " Put the respective new translations in a map of their corresponding locale.",
+ );
+ stdout.writeln(
+ const JsonEncoder.withIndent(' ').convert(messagesWithValues),
+ );
+ }
+}
diff --git a/cli/core/env.dart b/cli/core/env.dart
new file mode 100644
index 00000000..33cc5df1
--- /dev/null
+++ b/cli/core/env.dart
@@ -0,0 +1,24 @@
+import 'dart:io';
+
+enum BuildChannel {
+ stable,
+ nightly;
+
+ factory BuildChannel.fromEnvironment(String name) {
+ final channel = Platform.environment[name]!;
+ if (channel == "stable") {
+ return BuildChannel.stable;
+ } else if (channel == "nightly") {
+ return BuildChannel.nightly;
+ } else {
+ throw Exception("Invalid channel: $channel");
+ }
+ }
+}
+
+class CliEnv {
+ static final channel = BuildChannel.fromEnvironment("CHANNEL");
+ static final dotenv = Platform.environment["DOTENV"]!;
+ static final ghRunNumber = Platform.environment["GITHUB_RUN_NUMBER"];
+ static final flutterVersion = Platform.environment["FLUTTER_VERSION"]!;
+}
diff --git a/devtools_options.yaml b/devtools_options.yaml
new file mode 100644
index 00000000..7e7e7f67
--- /dev/null
+++ b/devtools_options.yaml
@@ -0,0 +1 @@
+extensions:
diff --git a/ios/Podfile b/ios/Podfile
index bc3dcaa6..7235f482 100644
--- a/ios/Podfile
+++ b/ios/Podfile
@@ -1,5 +1,5 @@
# Uncomment this line to define a global platform for your project
-# platform :ios, '12.0'
+platform :ios, '13.0'
# CocoaPods analytics sends network stats synchronously affecting flutter build latency.
ENV['COCOAPODS_DISABLE_STATS'] = 'true'
diff --git a/ios/Podfile.lock b/ios/Podfile.lock
index 0b75217f..2d570cbc 100644
--- a/ios/Podfile.lock
+++ b/ios/Podfile.lock
@@ -1,10 +1,13 @@
PODS:
- - app_links (0.0.1):
+ - app_links (0.0.2):
- Flutter
- audio_service (0.0.1):
- Flutter
- audio_session (0.0.1):
- Flutter
+ - bonsoir_darwin (0.0.1):
+ - Flutter
+ - FlutterMacOS
- device_info_plus (0.0.1):
- Flutter
- DKImagePickerController/Core (4.3.4):
@@ -44,29 +47,23 @@ PODS:
- file_selector_ios (0.0.1):
- Flutter
- Flutter (1.0.0)
- - flutter_inappwebview (0.0.1):
+ - flutter_broadcasts (0.0.1):
- Flutter
- - flutter_inappwebview/Core (= 0.0.1)
- - OrderedSet (~> 5.0)
- - flutter_inappwebview/Core (0.0.1):
+ - flutter_discord_rpc (0.0.1):
- Flutter
- - OrderedSet (~> 5.0)
- - flutter_keyboard_visibility (0.0.1):
+ - flutter_inappwebview_ios (0.0.1):
- Flutter
- - flutter_mailer (0.0.1):
+ - flutter_inappwebview_ios/Core (= 0.0.1)
+ - OrderedSet (~> 6.0.3)
+ - flutter_inappwebview_ios/Core (0.0.1):
- Flutter
+ - OrderedSet (~> 6.0.3)
- flutter_native_splash (0.0.1):
- Flutter
- flutter_secure_storage (6.0.0):
- Flutter
- flutter_sharing_intent (0.0.1):
- Flutter
- - fluttertoast (0.0.2):
- - Flutter
- - Toast
- - FMDB (2.7.5):
- - FMDB/standard (= 2.7.5)
- - FMDB/standard (2.7.5)
- image_picker_ios (0.0.1):
- Flutter
- integration_test (0.0.1):
@@ -75,14 +72,15 @@ PODS:
- Flutter
- media_kit_native_event_loop (1.0.0):
- Flutter
- - metadata_god (0.0.1)
- - OrderedSet (5.0.0)
+ - metadata_god (0.0.1):
+ - Flutter
+ - OrderedSet (6.0.3)
- package_info_plus (0.4.5):
- Flutter
- path_provider_foundation (0.0.1):
- Flutter
- FlutterMacOS
- - permission_handler_apple (9.1.1):
+ - permission_handler_apple (9.3.0):
- Flutter
- SDWebImage (5.18.8):
- SDWebImage/Core (= 5.18.8)
@@ -92,9 +90,23 @@ PODS:
- FlutterMacOS
- sqflite (0.0.3):
- Flutter
- - FMDB (>= 2.7.5)
+ - FlutterMacOS
+ - "sqlite3 (3.46.0+1)":
+ - "sqlite3/common (= 3.46.0+1)"
+ - "sqlite3/common (3.46.0+1)"
+ - "sqlite3/fts5 (3.46.0+1)":
+ - sqlite3/common
+ - "sqlite3/perf-threadsafe (3.46.0+1)":
+ - sqlite3/common
+ - "sqlite3/rtree (3.46.0+1)":
+ - sqlite3/common
+ - sqlite3_flutter_libs (0.0.1):
+ - Flutter
+ - sqlite3 (~> 3.46.0)
+ - sqlite3/fts5
+ - sqlite3/perf-threadsafe
+ - sqlite3/rtree
- SwiftyGif (5.4.4)
- - Toast (4.0.0)
- url_launcher_ios (0.0.1):
- Flutter
@@ -102,17 +114,17 @@ DEPENDENCIES:
- app_links (from `.symlinks/plugins/app_links/ios`)
- audio_service (from `.symlinks/plugins/audio_service/ios`)
- audio_session (from `.symlinks/plugins/audio_session/ios`)
+ - bonsoir_darwin (from `.symlinks/plugins/bonsoir_darwin/darwin`)
- 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`)
- Flutter (from `Flutter`)
- - flutter_inappwebview (from `.symlinks/plugins/flutter_inappwebview/ios`)
- - flutter_keyboard_visibility (from `.symlinks/plugins/flutter_keyboard_visibility/ios`)
- - flutter_mailer (from `.symlinks/plugins/flutter_mailer/ios`)
+ - flutter_broadcasts (from `.symlinks/plugins/flutter_broadcasts/ios`)
+ - flutter_discord_rpc (from `.symlinks/plugins/flutter_discord_rpc/ios`)
+ - flutter_inappwebview_ios (from `.symlinks/plugins/flutter_inappwebview_ios/ios`)
- 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`)
- - fluttertoast (from `.symlinks/plugins/fluttertoast/ios`)
- image_picker_ios (from `.symlinks/plugins/image_picker_ios/ios`)
- integration_test (from `.symlinks/plugins/integration_test/ios`)
- media_kit_libs_ios_audio (from `.symlinks/plugins/media_kit_libs_ios_audio/ios`)
@@ -122,18 +134,18 @@ DEPENDENCIES:
- path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`)
- permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`)
- shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`)
- - sqflite (from `.symlinks/plugins/sqflite/ios`)
+ - sqflite (from `.symlinks/plugins/sqflite/darwin`)
+ - sqlite3_flutter_libs (from `.symlinks/plugins/sqlite3_flutter_libs/ios`)
- url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`)
SPEC REPOS:
trunk:
- DKImagePickerController
- DKPhotoGallery
- - FMDB
- OrderedSet
- SDWebImage
+ - sqlite3
- SwiftyGif
- - Toast
EXTERNAL SOURCES:
app_links:
@@ -142,6 +154,8 @@ EXTERNAL SOURCES:
:path: ".symlinks/plugins/audio_service/ios"
audio_session:
:path: ".symlinks/plugins/audio_session/ios"
+ bonsoir_darwin:
+ :path: ".symlinks/plugins/bonsoir_darwin/darwin"
device_info_plus:
:path: ".symlinks/plugins/device_info_plus/ios"
file_picker:
@@ -150,20 +164,18 @@ EXTERNAL SOURCES:
:path: ".symlinks/plugins/file_selector_ios/ios"
Flutter:
:path: Flutter
- flutter_inappwebview:
- :path: ".symlinks/plugins/flutter_inappwebview/ios"
- flutter_keyboard_visibility:
- :path: ".symlinks/plugins/flutter_keyboard_visibility/ios"
- flutter_mailer:
- :path: ".symlinks/plugins/flutter_mailer/ios"
+ flutter_broadcasts:
+ :path: ".symlinks/plugins/flutter_broadcasts/ios"
+ flutter_discord_rpc:
+ :path: ".symlinks/plugins/flutter_discord_rpc/ios"
+ flutter_inappwebview_ios:
+ :path: ".symlinks/plugins/flutter_inappwebview_ios/ios"
flutter_native_splash:
:path: ".symlinks/plugins/flutter_native_splash/ios"
flutter_secure_storage:
:path: ".symlinks/plugins/flutter_secure_storage/ios"
flutter_sharing_intent:
:path: ".symlinks/plugins/flutter_sharing_intent/ios"
- fluttertoast:
- :path: ".symlinks/plugins/fluttertoast/ios"
image_picker_ios:
:path: ".symlinks/plugins/image_picker_ios/ios"
integration_test:
@@ -183,44 +195,46 @@ EXTERNAL SOURCES:
shared_preferences_foundation:
:path: ".symlinks/plugins/shared_preferences_foundation/darwin"
sqflite:
- :path: ".symlinks/plugins/sqflite/ios"
+ :path: ".symlinks/plugins/sqflite/darwin"
+ sqlite3_flutter_libs:
+ :path: ".symlinks/plugins/sqlite3_flutter_libs/ios"
url_launcher_ios:
:path: ".symlinks/plugins/url_launcher_ios/ios"
SPEC CHECKSUMS:
- app_links: 5ef33d0d295a89d9d16bb81b0e3b0d5f70d6c875
+ app_links: e7a6750a915a9e161c58d91bc610e8cd1d4d0ad0
audio_service: f509d65da41b9521a61f1c404dd58651f265a567
- audio_session: 4f3e461722055d21515cf3261b64c973c062f345
- device_info_plus: 7545d84d8d1b896cb16a4ff98c19f07ec4b298ea
+ audio_session: 088d2483ebd1dc43f51d253d4a1c517d9a2e7207
+ bonsoir_darwin: e3b8526c42ca46a885142df84229131dfabea842
+ device_info_plus: 97af1d7e84681a90d0693e63169a5d50e0839a0d
DKImagePickerController: b512c28220a2b8ac7419f21c491fc8534b7601ac
DKPhotoGallery: fdfad5125a9fdda9cc57df834d49df790dbb4179
- file_picker: 15fd9539e4eb735dc54bae8c0534a7a9511a03de
- file_selector_ios: 8c25d700d625e1dcdd6599f2d927072f2254647b
+ file_picker: 09aa5ec1ab24135ccd7a1621c46c84134bfd6655
+ file_selector_ios: 78baf21d03f1e37a7df97bb2494f9cd86de8fa5d
Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7
- flutter_inappwebview: acd4fc0f012cefd09015000c241137d82f01ba62
- flutter_keyboard_visibility: 0339d06371254c3eb25eeb90ba8d17dca8f9c069
- flutter_mailer: 2ef5a67087bc8c6c4cefd04a178bf1ae2c94cd83
- flutter_native_splash: 52501b97d1c0a5f898d687f1646226c1f93c56ef
+ flutter_broadcasts: 3ece15b27d8ccbe2132c3df303e7c3401feab882
+ flutter_discord_rpc: e1c342f29ceb9dd76cdc01db59a70c93bb4d9ec5
+ flutter_inappwebview_ios: 6f63631e2c62a7c350263b13fa5427aedefe81d4
+ flutter_native_splash: edf599c81f74d093a4daf8e17bd7a018854bc778
flutter_secure_storage: 23fc622d89d073675f2eaa109381aefbcf5a49be
flutter_sharing_intent: e35380d0e1501d7111dbb7e46d5ac6339da6da98
- fluttertoast: 31b00dabfa7fb7bacd9e7dbee580d7a2ff4bf265
- FMDB: 2ce00b547f966261cd18927a3ddb07cb6f3db82a
- image_picker_ios: 4a8aadfbb6dc30ad5141a2ce3832af9214a705b5
- integration_test: 13825b8a9334a850581300559b8839134b124670
+ image_picker_ios: b545a5f16c0fa88e3ecbbce3ed4de45567a8ec18
+ integration_test: 252f60fa39af5e17c3aa9899d35d908a0721b573
media_kit_libs_ios_audio: 8f39d96a9c630685dfb844c289bd1d114c486fb3
media_kit_native_event_loop: 99111eded5acbdc9c2738021ea6550dd36ca8837
- metadata_god: eceae399d0020475069a5cebc35943ce8562b5d7
- OrderedSet: aaeb196f7fef5a9edf55d89760da9176ad40b93c
- package_info_plus: fd030dabf36271f146f1f3beacd48f564b0f17f7
- path_provider_foundation: 29f094ae23ebbca9d3d0cec13889cd9060c0e943
- permission_handler_apple: e76247795d700c14ea09e3a2d8855d41ee80a2e6
+ metadata_god: 4bbd8523cdb5d42c5e59d2fabad01ff8f4bc53f9
+ OrderedSet: e539b66b644ff081c73a262d24ad552a69be3a94
+ package_info_plus: 58f0028419748fad15bf008b270aaa8e54380b1c
+ path_provider_foundation: 3784922295ac71e43754bd15e0653ccfd36a147c
+ permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2
SDWebImage: a81bbb3ba4ea5f810f4069c68727cb118467a04a
- shared_preferences_foundation: 5b919d13b803cadd15ed2dc053125c68730e5126
- sqflite: 31f7eba61e3074736dff8807a9b41581e4f7f15a
+ shared_preferences_foundation: b4c3b4cddf1c21f02770737f147a3f5da9d39695
+ sqflite: 673a0e54cc04b7d6dba8d24fb8095b31c3a99eec
+ sqlite3: 292c3e1bfe89f64e51ea7fc7dab9182a017c8630
+ sqlite3_flutter_libs: 0d611efdf6d1c9297d5ab03dab21b75aeebdae31
SwiftyGif: 93a1cc87bf3a51916001cf8f3d63835fb64c819f
- Toast: 91b396c56ee72a5790816f40d3a94dd357abc196
- url_launcher_ios: 08a3dfac5fb39e8759aeb0abbd5d9480f30fc8b4
+ url_launcher_ios: 6116280ddcfe98ab8820085d8d76ae7449447586
-PODFILE CHECKSUM: 5129d2e80ab0dfc533f262cedf032011b1dfe4fd
+PODFILE CHECKSUM: 0659b64ac6e9e96b61d8550decffa8bff51a957e
COCOAPODS: 1.15.2
diff --git a/ios/Runner.xcodeproj/project.pbxproj b/ios/Runner.xcodeproj/project.pbxproj
index 13f624a4..34793f68 100644
--- a/ios/Runner.xcodeproj/project.pbxproj
+++ b/ios/Runner.xcodeproj/project.pbxproj
@@ -324,6 +324,7 @@
9705A1C41CF9048500538489 /* Embed Frameworks */,
3B06AD1E1E4923F5004D2608 /* Thin Binary */,
6E9FEF583EA597C8B76255B2 /* [CP] Embed Pods Frameworks */,
+ 46F6EB27C31C41D86428A28B /* [CP] Copy Pods Resources */,
);
buildRules = (
);
@@ -346,6 +347,7 @@
B536BD992B405DB1009B3CE4 /* Embed Frameworks */,
B536BD9A2B405DB1009B3CE4 /* Thin Binary */,
A6D446F111DE4C4A202BE7F7 /* [CP] Embed Pods Frameworks */,
+ 2DEF3CF18D30E819C0FF4BCE /* [CP] Copy Pods Resources */,
);
buildRules = (
);
@@ -368,6 +370,7 @@
B536BDB62B405FDE009B3CE4 /* Embed Frameworks */,
B536BDB72B405FDE009B3CE4 /* Thin Binary */,
244D41CE80E4BC0FFD63F8C6 /* [CP] Embed Pods Frameworks */,
+ 4DD66E9E53D92195290872BE /* [CP] Copy Pods Resources */,
);
buildRules = (
);
@@ -390,6 +393,7 @@
B536BDD82B4060B3009B3CE4 /* Embed Frameworks */,
B536BDD92B4060B3009B3CE4 /* Thin Binary */,
D566C841A84D807A607F6DE5 /* [CP] Embed Pods Frameworks */,
+ 5C9D945D6569D9C3AC420285 /* [CP] Copy Pods Resources */,
);
buildRules = (
);
@@ -523,6 +527,23 @@
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
showEnvVarsInLog = 0;
};
+ 2DEF3CF18D30E819C0FF4BCE /* [CP] Copy Pods Resources */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-stable/Pods-stable-resources-${CONFIGURATION}-input-files.xcfilelist",
+ );
+ name = "[CP] Copy Pods Resources";
+ outputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-stable/Pods-stable-resources-${CONFIGURATION}-output-files.xcfilelist",
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-stable/Pods-stable-resources.sh\"\n";
+ showEnvVarsInLog = 0;
+ };
3B06AD1E1E4923F5004D2608 /* Thin Binary */ = {
isa = PBXShellScriptBuildPhase;
alwaysOutOfDate = 1;
@@ -539,6 +560,57 @@
shellPath = /bin/sh;
shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" embed_and_thin";
};
+ 46F6EB27C31C41D86428A28B /* [CP] Copy Pods Resources */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-input-files.xcfilelist",
+ );
+ name = "[CP] Copy Pods Resources";
+ outputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-output-files.xcfilelist",
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources.sh\"\n";
+ showEnvVarsInLog = 0;
+ };
+ 4DD66E9E53D92195290872BE /* [CP] Copy Pods Resources */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-dev/Pods-dev-resources-${CONFIGURATION}-input-files.xcfilelist",
+ );
+ name = "[CP] Copy Pods Resources";
+ outputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-dev/Pods-dev-resources-${CONFIGURATION}-output-files.xcfilelist",
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-dev/Pods-dev-resources.sh\"\n";
+ showEnvVarsInLog = 0;
+ };
+ 5C9D945D6569D9C3AC420285 /* [CP] Copy Pods Resources */ = {
+ isa = PBXShellScriptBuildPhase;
+ buildActionMask = 2147483647;
+ files = (
+ );
+ inputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-nightly/Pods-nightly-resources-${CONFIGURATION}-input-files.xcfilelist",
+ );
+ name = "[CP] Copy Pods Resources";
+ outputFileListPaths = (
+ "${PODS_ROOT}/Target Support Files/Pods-nightly/Pods-nightly-resources-${CONFIGURATION}-output-files.xcfilelist",
+ );
+ runOnlyForDeploymentPostprocessing = 0;
+ shellPath = /bin/sh;
+ shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-nightly/Pods-nightly-resources.sh\"\n";
+ showEnvVarsInLog = 0;
+ };
5CD4405E93760FBD048E36E2 /* [CP] Check Pods Manifest.lock */ = {
isa = PBXShellScriptBuildPhase;
buildActionMask = 2147483647;
diff --git a/ios/Runner/Info.plist b/ios/Runner/Info.plist
index 8e103cfa..ffd511a4 100644
--- a/ios/Runner/Info.plist
+++ b/ios/Runner/Info.plist
@@ -66,5 +66,11 @@
UIViewControllerBasedStatusBarAppearance
+ NSLocalNetworkUsageDescription
+ To allow other devices on the network control playback of Spotube securely.
+ NSBonjourServices
+
+ _spotube._tcp
+
\ No newline at end of file
diff --git a/lib/collections/assets.gen.dart b/lib/collections/assets.gen.dart
index 8a2950fb..cff5b74f 100644
--- a/lib/collections/assets.gen.dart
+++ b/lib/collections/assets.gen.dart
@@ -59,6 +59,8 @@ class Assets {
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');
@@ -88,7 +90,7 @@ class Assets {
AssetGenImage('assets/user-placeholder.png');
/// List of all assets
- List get values => [
+ static List get values => [
albumPlaceholder,
bengaliPatternsBg,
branding,
@@ -98,6 +100,7 @@ class Assets {
placeholder,
spotubeHeroBanner,
spotubeLogoForeground,
+ spotubeLogoBmp,
spotubeLogoIco,
spotubeLogoPng,
spotubeLogoSvg,
diff --git a/lib/collections/cache_keys.dart b/lib/collections/cache_keys.dart
deleted file mode 100644
index bca13322..00000000
--- a/lib/collections/cache_keys.dart
+++ /dev/null
@@ -1,21 +0,0 @@
-abstract class LocalStorageKeys {
- static String saveTrackLyrics = 'save_track_lyrics';
- static String recommendationMarket = 'recommendation_market';
- static String ytSearchFormate = 'youtube_search_format';
-
- static String clientId = 'clientId';
- static String clientSecret = 'clientSecret';
- static String accessToken = 'accessToken';
- static String refreshToken = 'refreshToken';
- static String expiration = "expiration";
- static String geniusAccessToken = "genius_access_token";
-
- static String themeMode = "theme_mode";
- static String nextTrackHotKey = "next_track_hot_key";
- static String prevTrackHotKey = "prev_track_hot_key";
- static String playPauseHotKey = "play_pause_hot_key";
-
- static String volume = "volume";
-
- static String windowSizeInfo = "window_size_info";
-}
diff --git a/lib/collections/env.dart b/lib/collections/env.dart
index 50fe1e6a..eb60851f 100644
--- a/lib/collections/env.dart
+++ b/lib/collections/env.dart
@@ -1,8 +1,13 @@
import 'package:envied/envied.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
+import 'package:spotube/utils/platform.dart';
part 'env.g.dart';
+enum ReleaseChannel {
+ nightly,
+ stable,
+}
+
@Envied(obfuscate: true, requireEnvFile: true, path: ".env")
abstract class Env {
@EnviedField(varName: 'SPOTIFY_SECRETS')
@@ -14,6 +19,11 @@ abstract class Env {
@EnviedField(varName: 'LASTFM_API_SECRET')
static final String lastFmApiSecret = _Env.lastFmApiSecret;
+ @EnviedField(varName: 'HIDE_DONATIONS', defaultValue: "0")
+ static final int _hideDonations = _Env._hideDonations;
+
+ static bool get hideDonations => _hideDonations == 1;
+
static final spotifySecrets = rawSpotifySecrets.split(',').map((e) {
final secrets = e.trim().split(":").map((e) => e.trim());
return {
@@ -25,8 +35,15 @@ abstract class Env {
@EnviedField(varName: 'ENABLE_UPDATE_CHECK', defaultValue: "1")
static final String _enableUpdateChecker = _Env._enableUpdateChecker;
+ @EnviedField(varName: "RELEASE_CHANNEL", defaultValue: "nightly")
+ static final String _releaseChannel = _Env._releaseChannel;
+
+ static ReleaseChannel get releaseChannel => _releaseChannel == "stable"
+ ? ReleaseChannel.stable
+ : ReleaseChannel.nightly;
+
static bool get enableUpdateChecker =>
- DesktopTools.platform.isFlatpak || _enableUpdateChecker == "1";
+ kIsFlatpak || _enableUpdateChecker == "1";
static String discordAppId = "1176718791388975124";
}
diff --git a/lib/collections/fake.dart b/lib/collections/fake.dart
index 8f5f9e8b..31f97e0c 100644
--- a/lib/collections/fake.dart
+++ b/lib/collections/fake.dart
@@ -1,12 +1,14 @@
import 'package:spotify/spotify.dart';
-import 'package:spotube/extensions/track.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/provider/history/summary.dart';
abstract class FakeData {
static final Image image = Image()
..height = 1
..width = 1
- ..url = "url";
+ ..url = "https://dummyimage.com/100x100/cfcfcf/cfcfcf.jpg";
static final Followers followers = Followers()
..href = "text"
@@ -196,4 +198,62 @@ abstract class FakeData {
),
],
);
+
+ 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 const historySummary = PlaybackHistorySummary(
+ albums: 1,
+ artists: 1,
+ duration: Duration(seconds: 1),
+ playlists: 1,
+ tracks: 1,
+ fees: 1,
+ );
+
+ static final historyRecentlyPlayedPlaylist = HistoryTableData(
+ id: 0,
+ type: HistoryEntryType.track,
+ createdAt: DateTime.now(),
+ itemId: "1",
+ data: playlist.toJson(),
+ );
+
+ static final historyRecentlyPlayedAlbum = HistoryTableData(
+ id: 0,
+ type: HistoryEntryType.track,
+ createdAt: DateTime.now(),
+ itemId: "1",
+ data: album.toJson(),
+ );
+
+ static final historyRecentlyPlayedItems = List.generate(
+ 10,
+ (index) => index % 2 == 0
+ ? historyRecentlyPlayedPlaylist
+ : historyRecentlyPlayedAlbum,
+ );
}
diff --git a/lib/collections/formatters.dart b/lib/collections/formatters.dart
new file mode 100644
index 00000000..0aed9e9f
--- /dev/null
+++ b/lib/collections/formatters.dart
@@ -0,0 +1,8 @@
+import 'package:intl/intl.dart';
+
+final compactNumberFormatter = NumberFormat.compact();
+final usdFormatter = NumberFormat.compactCurrency(
+ locale: 'en-US',
+ symbol: r"$",
+ decimalDigits: 2,
+);
diff --git a/lib/collections/initializers.dart b/lib/collections/initializers.dart
index 9627de1c..976661fc 100644
--- a/lib/collections/initializers.dart
+++ b/lib/collections/initializers.dart
@@ -1,9 +1,10 @@
import 'dart:io';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
+
+import 'package:spotube/utils/platform.dart';
import 'package:win32_registry/win32_registry.dart';
Future registerWindowsScheme(String scheme) async {
- if (!DesktopTools.platform.isWindows) return;
+ if (!kIsWindows) return;
String appPath = Platform.resolvedExecutable;
String protocolRegKey = 'Software\\Classes\\$scheme';
diff --git a/lib/collections/intents.dart b/lib/collections/intents.dart
index 6f42113c..4f446831 100644
--- a/lib/collections/intents.dart
+++ b/lib/collections/intents.dart
@@ -5,9 +5,12 @@ 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/components/player/player_controls.dart';
-import 'package:spotube/models/logger.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.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';
@@ -17,8 +20,6 @@ class PlayPauseIntent extends Intent {
}
class PlayPauseAction extends Action {
- final logger = getLogger(PlayPauseAction);
-
@override
invoke(intent) async {
if (PlayerControls.focusNode.canRequestFocus) {
@@ -67,16 +68,16 @@ class HomeTabAction extends Action {
final router = intent.ref.read(routerProvider);
switch (intent.tab) {
case HomeTabs.browse:
- router.go("/");
+ router.goNamed(HomePage.name);
break;
case HomeTabs.search:
- router.go("/search");
+ router.goNamed(SearchPage.name);
break;
case HomeTabs.library:
- router.go("/library");
+ router.goNamed(LibraryPage.name);
break;
case HomeTabs.lyrics:
- router.go("/lyrics");
+ router.goNamed(LyricsPage.name);
break;
}
return null;
@@ -92,8 +93,8 @@ class SeekIntent extends Intent {
class SeekAction extends Action {
@override
invoke(intent) async {
- final playlist = intent.ref.read(ProxyPlaylistNotifier.provider);
- if (playlist.isFetching) {
+ final isFetchingActiveTrack = intent.ref.read(queryingTrackInfoProvider);
+ if (isFetchingActiveTrack) {
DirectionalFocusAction().invoke(
DirectionalFocusIntent(
intent.forward ? TraversalDirection.right : TraversalDirection.left,
@@ -101,7 +102,7 @@ class SeekAction extends Action {
);
return null;
}
- final position = (await audioPlayer.position ?? Duration.zero).inSeconds;
+ final position = audioPlayer.position.inSeconds;
await audioPlayer.seek(
Duration(
seconds: intent.forward ? position + 5 : position - 5,
diff --git a/lib/collections/language_codes.dart b/lib/collections/language_codes.dart
index 4b7a3a90..44da6ee6 100644
--- a/lib/collections/language_codes.dart
+++ b/lib/collections/language_codes.dart
@@ -81,10 +81,10 @@ abstract class LanguageLocals {
// name: "Bashkir",
// nativeName: "башҡорт теле",
// ),
- // "eu": const ISOLanguageName(
- // name: "Basque",
- // nativeName: "euskara,",
- // ),
+ "eu": const ISOLanguageName(
+ name: "Basque",
+ nativeName: "Euskara",
+ ),
// "be": const ISOLanguageName(
// name: "Belarusian",
// nativeName: "Беларуская",
@@ -157,10 +157,10 @@ abstract class LanguageLocals {
// name: "Croatian",
// nativeName: "hrvatski",
// ),
- // "cs": const ISOLanguageName(
- // name: "Czech",
- // nativeName: "česky, čeština",
- // ),
+ "cs": const ISOLanguageName(
+ name: "Czech",
+ nativeName: "česky, čeština",
+ ),
// "da": const ISOLanguageName(
// name: "Danish",
// nativeName: "dansk",
@@ -197,10 +197,10 @@ abstract class LanguageLocals {
// name: "Fijian",
// nativeName: "vosa Vakaviti",
// ),
- // "fi": const ISOLanguageName(
- // name: "Finnish",
- // nativeName: "suomi",
- // ),
+ "fi": const ISOLanguageName(
+ name: "Finnish",
+ nativeName: "suomi",
+ ),
"fr": const ISOLanguageName(
name: "French",
nativeName: "français",
@@ -213,10 +213,10 @@ abstract class LanguageLocals {
// name: "Galician",
// nativeName: "Galego",
// ),
- // "ka": const ISOLanguageName(
- // name: "Georgian",
- // nativeName: "ქართული",
- // ),
+ "ka": const ISOLanguageName(
+ name: "Georgian",
+ nativeName: "ქართული",
+ ),
"de": const ISOLanguageName(
name: "German",
nativeName: "Deutsch",
@@ -265,10 +265,10 @@ abstract class LanguageLocals {
// name: "Interlingua",
// nativeName: "Interlingua",
// ),
- // "id": const ISOLanguageName(
- // name: "Indonesian",
- // nativeName: "Bahasa Indonesia",
- // ),
+ "id": const ISOLanguageName(
+ name: "Indonesian",
+ nativeName: "Bahasa Indonesia",
+ ),
// "ie": const ISOLanguageName(
// name: "Interlingue",
// nativeName: "Occidental",
@@ -354,8 +354,8 @@ abstract class LanguageLocals {
// nativeName: "KiKongo",
// ),
"ko": const ISOLanguageName(
- name: "Korean",
- nativeName: "한국어 (韓國語), 조선말 (朝鮮語)",
+ name: "Korean",
+ nativeName: "한국어 (韓國語), 조선말 (朝鮮語)",
),
// "ku": const ISOLanguageName(
// name: "Kurdish",
@@ -637,10 +637,10 @@ abstract class LanguageLocals {
// name: "Tajik",
// nativeName: "тоҷикӣ, toğikī, تاجیکی",
// ),
- // "th": const ISOLanguageName(
- // name: "Thai",
- // nativeName: "ไทย",
- // ),
+ "th": const ISOLanguageName(
+ name: "Thai",
+ nativeName: "ไทย",
+ ),
// "ti": const ISOLanguageName(
// name: "Tigrinya",
// nativeName: "ትግርኛ",
diff --git a/lib/collections/routes.dart b/lib/collections/routes.dart
index 43d0cf2e..3bf1d883 100644
--- a/lib/collections/routes.dart
+++ b/lib/collections/routes.dart
@@ -1,39 +1,48 @@
-import 'package:catcher_2/catcher_2.dart';
import 'package:flutter/foundation.dart' hide Category;
import 'package:flutter/widgets.dart';
import 'package:go_router/go_router.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_provider.dart';
+import 'package:spotube/provider/authentication/authentication.dart';
import 'package:spotube/services/kv_store/kv_store.dart';
-import 'package:spotube/utils/platform.dart';
-import 'package:spotube/components/shared/spotube_page_route.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/desktop_login/login_tutorial.dart';
-import 'package:spotube/pages/desktop_login/desktop_login.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 = Catcher2.navigatorKey;
+final rootNavigatorKey = GlobalKey();
final shellRouteNavigatorKey = GlobalKey();
final routerProvider = Provider((ref) {
return GoRouter(
@@ -45,13 +54,11 @@ final routerProvider = Provider((ref) {
routes: [
GoRoute(
path: "/",
+ name: HomePage.name,
redirect: (context, state) async {
- final authNotifier =
- ref.read(AuthenticationNotifier.provider.notifier);
- final json = await authNotifier.box.get(authNotifier.cacheKey);
+ final auth = await ref.read(authenticationProvider.future);
- if (json?["cookie"] == null &&
- !KVStoreService.doneGettingStarted) {
+ if (auth == null && !KVStoreService.doneGettingStarted) {
return "/getting-started";
}
@@ -62,61 +69,88 @@ final routerProvider = Provider((ref) {
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: "Search",
+ name: SearchPage.name,
pageBuilder: (context, state) =>
const SpotubePage(child: SearchPage()),
),
GoRoute(
path: "/library",
- name: "Library",
+ name: LibraryPage.name,
pageBuilder: (context, state) =>
const SpotubePage(child: LibraryPage()),
routes: [
GoRoute(
- path: "generate",
- pageBuilder: (context, state) =>
- const SpotubePage(child: PlaylistGeneratorPage()),
- routes: [
- GoRoute(
- path: "result",
- pageBuilder: (context, state) => SpotubePage(
- child: PlaylistGenerateResultPage(
- state:
- state.extra as PlaylistGenerateResultRouteState,
- ),
+ 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),
+ );
+ },
+ ),
]),
GoRoute(
path: "/lyrics",
- name: "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(),
),
@@ -124,12 +158,14 @@ final routerProvider = Provider((ref) {
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(),
),
@@ -138,6 +174,7 @@ final routerProvider = Provider((ref) {
),
GoRoute(
path: "/album/:id",
+ name: AlbumPage.name,
pageBuilder: (context, state) {
assert(state.extra is AlbumSimple);
return SpotubePage(
@@ -147,6 +184,7 @@ final routerProvider = Provider((ref) {
),
GoRoute(
path: "/artist/:id",
+ name: ArtistPage.name,
pageBuilder: (context, state) {
assert(state.pathParameters["id"] != null);
return SpotubePage(
@@ -155,6 +193,7 @@ final routerProvider = Provider((ref) {
),
GoRoute(
path: "/playlist/:id",
+ name: PlaylistPage.name,
pageBuilder: (context, state) {
assert(state.extra is PlaylistSimple);
return SpotubePage(
@@ -166,6 +205,7 @@ final routerProvider = Provider((ref) {
),
GoRoute(
path: "/track/:id",
+ name: TrackPage.name,
pageBuilder: (context, state) {
final id = state.pathParameters["id"]!;
return SpotubePage(
@@ -173,10 +213,86 @@ final routerProvider = Provider((ref) {
);
},
),
+ GoRoute(
+ path: "/connect",
+ name: ConnectPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: ConnectPage(),
+ ),
+ 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(),
+ ),
+ routes: [
+ GoRoute(
+ path: "minutes",
+ name: StatsMinutesPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: StatsMinutesPage(),
+ ),
+ ),
+ GoRoute(
+ path: "streams",
+ name: StatsStreamsPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: StatsStreamsPage(),
+ ),
+ ),
+ GoRoute(
+ path: "fees",
+ name: StatsStreamFeesPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: StatsStreamFeesPage(),
+ ),
+ ),
+ GoRoute(
+ path: "artists",
+ name: StatsArtistsPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: StatsArtistsPage(),
+ ),
+ ),
+ GoRoute(
+ path: "albums",
+ name: StatsAlbumsPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: StatsAlbumsPage(),
+ ),
+ ),
+ GoRoute(
+ path: "playlists",
+ name: StatsPlaylistsPage.name,
+ pageBuilder: (context, state) => const SpotubePage(
+ child: StatsPlaylistsPage(),
+ ),
+ ),
+ ],
+ )
],
),
GoRoute(
path: "/mini-player",
+ name: MiniLyricsPage.name,
parentNavigatorKey: rootNavigatorKey,
pageBuilder: (context, state) => SpotubePage(
child: MiniLyricsPage(prevSize: state.extra as Size),
@@ -184,6 +300,7 @@ final routerProvider = Provider((ref) {
),
GoRoute(
path: "/getting-started",
+ name: GettingStarting.name,
parentNavigatorKey: rootNavigatorKey,
pageBuilder: (context, state) => const SpotubePage(
child: GettingStarting(),
@@ -191,20 +308,15 @@ final routerProvider = Provider((ref) {
),
GoRoute(
path: "/login",
- parentNavigatorKey: rootNavigatorKey,
- pageBuilder: (context, state) => SpotubePage(
- child: kIsMobile ? const WebViewLogin() : const DesktopLoginPage(),
- ),
- ),
- GoRoute(
- path: "/login-tutorial",
+ name: WebViewLogin.name,
parentNavigatorKey: rootNavigatorKey,
pageBuilder: (context, state) => const SpotubePage(
- child: LoginTutorial(),
+ child: WebViewLogin(),
),
),
GoRoute(
path: "/lastfm-login",
+ name: LastFMLoginPage.name,
parentNavigatorKey: rootNavigatorKey,
pageBuilder: (context, state) =>
const SpotubePage(child: LastFMLoginPage()),
diff --git a/lib/collections/side_bar_tiles.dart b/lib/collections/side_bar_tiles.dart
index 551d70d7..4f23c049 100644
--- a/lib/collections/side_bar_tiles.dart
+++ b/lib/collections/side_bar_tiles.dart
@@ -1,33 +1,82 @@
import 'package:flutter/material.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';
class SideBarTiles {
final IconData icon;
final String title;
final String id;
- SideBarTiles({required this.icon, required this.title, required this.id});
+ final String name;
+
+ SideBarTiles({
+ required this.icon,
+ required this.title,
+ required this.id,
+ required this.name,
+ });
}
List getSidebarTileList(AppLocalizations l10n) => [
- SideBarTiles(id: "browse", icon: SpotubeIcons.home, title: l10n.browse),
- SideBarTiles(id: "search", icon: SpotubeIcons.search, title: l10n.search),
SideBarTiles(
- id: "library", icon: SpotubeIcons.library, title: l10n.library),
- SideBarTiles(id: "lyrics", icon: SpotubeIcons.music, title: l10n.lyrics),
- ];
-
-List getNavbarTileList(AppLocalizations l10n) => [
- SideBarTiles(id: "browse", icon: SpotubeIcons.home, title: l10n.browse),
- SideBarTiles(id: "search", icon: SpotubeIcons.search, title: l10n.search),
+ id: "browse",
+ name: HomePage.name,
+ icon: SpotubeIcons.home,
+ title: l10n.browse,
+ ),
+ SideBarTiles(
+ id: "search",
+ name: SearchPage.name,
+ icon: SpotubeIcons.search,
+ title: l10n.search,
+ ),
SideBarTiles(
id: "library",
+ name: LibraryPage.name,
icon: SpotubeIcons.library,
title: l10n.library,
),
SideBarTiles(
- id: "settings",
- icon: SpotubeIcons.settings,
- title: l10n.settings,
- )
+ id: "lyrics",
+ name: LyricsPage.name,
+ icon: SpotubeIcons.music,
+ title: l10n.lyrics,
+ ),
+ SideBarTiles(
+ id: "stats",
+ name: StatsPage.name,
+ icon: SpotubeIcons.chart,
+ title: l10n.stats,
+ ),
+ ];
+
+List getNavbarTileList(AppLocalizations l10n) => [
+ SideBarTiles(
+ id: "browse",
+ name: HomePage.name,
+ icon: SpotubeIcons.home,
+ title: l10n.browse,
+ ),
+ SideBarTiles(
+ id: "search",
+ name: SearchPage.name,
+ icon: SpotubeIcons.search,
+ title: l10n.search,
+ ),
+ SideBarTiles(
+ id: "library",
+ name: LibraryPage.name,
+ icon: SpotubeIcons.library,
+ title: l10n.library,
+ ),
+ SideBarTiles(
+ id: "stats",
+ name: StatsPage.name,
+ icon: SpotubeIcons.chart,
+ title: l10n.stats,
+ ),
];
diff --git a/lib/collections/spotube_icons.dart b/lib/collections/spotube_icons.dart
index 6cf92085..a45e581e 100644
--- a/lib/collections/spotube_icons.dart
+++ b/lib/collections/spotube_icons.dart
@@ -115,4 +115,13 @@ abstract class SpotubeIcons {
static const github = SimpleIcons.github;
static const openCollective = SimpleIcons.opencollective;
static const anonymous = FeatherIcons.user;
+ static const history = FeatherIcons.clock;
+ static const connect = FeatherIcons.link;
+ static const speaker = FeatherIcons.speaker;
+ static const monitor = FeatherIcons.monitor;
+ static const power = FeatherIcons.power;
+ static const bluetooth = FeatherIcons.bluetooth;
+ static const chart = FeatherIcons.barChart2;
+ static const folderAdd = FeatherIcons.folderPlus;
+ static const folderRemove = FeatherIcons.folderMinus;
}
diff --git a/lib/components/shared/adaptive/adaptive_list_tile.dart b/lib/components/adaptive/adaptive_list_tile.dart
similarity index 100%
rename from lib/components/shared/adaptive/adaptive_list_tile.dart
rename to lib/components/adaptive/adaptive_list_tile.dart
diff --git a/lib/components/shared/adaptive/adaptive_pop_sheet_list.dart b/lib/components/adaptive/adaptive_pop_sheet_list.dart
similarity index 97%
rename from lib/components/shared/adaptive/adaptive_pop_sheet_list.dart
rename to lib/components/adaptive/adaptive_pop_sheet_list.dart
index 21f56a22..97dc6132 100644
--- a/lib/components/shared/adaptive/adaptive_pop_sheet_list.dart
+++ b/lib/components/adaptive/adaptive_pop_sheet_list.dart
@@ -187,7 +187,7 @@ class AdaptivePopSheetList extends StatelessWidget {
icon: icon ?? const Icon(SpotubeIcons.moreVertical),
tooltip: tooltip,
style: theme.iconButtonTheme.style?.copyWith(
- shape: MaterialStatePropertyAll(
+ shape: WidgetStatePropertyAll(
RoundedRectangleBorder(
borderRadius: borderRadius,
),
@@ -226,7 +226,10 @@ class _AdaptivePopSheetListItem extends StatelessWidget {
},
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 8),
- child: IgnorePointer(child: item),
+ child: IconTheme.merge(
+ data: const IconThemeData(opacity: 1),
+ child: IgnorePointer(child: item),
+ ),
),
);
}
diff --git a/lib/components/shared/adaptive/adaptive_popup_menu_button.dart b/lib/components/adaptive/adaptive_popup_menu_button.dart
similarity index 98%
rename from lib/components/shared/adaptive/adaptive_popup_menu_button.dart
rename to lib/components/adaptive/adaptive_popup_menu_button.dart
index 45f22825..02fced52 100644
--- a/lib/components/shared/adaptive/adaptive_popup_menu_button.dart
+++ b/lib/components/adaptive/adaptive_popup_menu_button.dart
@@ -12,13 +12,13 @@ class Action extends StatelessWidget {
final bool isExpanded;
final Color? backgroundColor;
const Action({
- Key? key,
+ super.key,
required this.icon,
required this.text,
required this.onPressed,
this.isExpanded = true,
this.backgroundColor,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/adaptive/adaptive_select_tile.dart b/lib/components/adaptive/adaptive_select_tile.dart
similarity index 83%
rename from lib/components/shared/adaptive/adaptive_select_tile.dart
rename to lib/components/adaptive/adaptive_select_tile.dart
index 58666e46..3f6d2700 100644
--- a/lib/components/shared/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_hooks/flutter_hooks.dart';
+import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/extensions/constrains.dart';
class AdaptiveSelectTile extends HookWidget {
@@ -38,11 +39,22 @@ class AdaptiveSelectTile extends HookWidget {
Widget build(BuildContext context) {
final theme = Theme.of(context);
final mediaQuery = MediaQuery.of(context);
- final rawControl = DropdownButton(
- items: options,
- value: value,
- onChanged: onChanged,
- menuMaxHeight: mediaQuery.size.height * 0.6,
+ 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
diff --git a/lib/components/shared/animated_gradient.dart b/lib/components/animated_gradient.dart
similarity index 98%
rename from lib/components/shared/animated_gradient.dart
rename to lib/components/animated_gradient.dart
index b6485f6b..aaba2ff9 100644
--- a/lib/components/shared/animated_gradient.dart
+++ b/lib/components/animated_gradient.dart
@@ -3,7 +3,7 @@ import 'package:flutter_hooks/flutter_hooks.dart';
class AnimateGradient extends HookWidget {
const AnimateGradient({
- Key? key,
+ super.key,
required this.primaryColors,
required this.secondaryColors,
this.child,
@@ -17,8 +17,7 @@ class AnimateGradient extends HookWidget {
this.reverse = true,
}) : assert(primaryColors.length >= 2),
assert(primaryColors.length == secondaryColors.length),
- _controller = controller,
- super(key: key);
+ _controller = controller;
/// [controller]: pass this to have a fine control over the [Animation]
final AnimationController? _controller;
diff --git a/lib/components/artist/artist_album_list.dart b/lib/components/artist/artist_album_list.dart
deleted file mode 100644
index 5114170c..00000000
--- a/lib/components/artist/artist_album_list.dart
+++ /dev/null
@@ -1,42 +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/shared/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/models/logger.dart';
-import 'package:spotube/services/queries/queries.dart';
-
-class ArtistAlbumList extends HookConsumerWidget {
- final String artistId;
- ArtistAlbumList(
- this.artistId, {
- Key? key,
- }) : super(key: key);
-
- final logger = getLogger(ArtistAlbumList);
-
- @override
- Widget build(BuildContext context, ref) {
- final albumsQuery = useQueries.artist.albumsOf(ref, artistId);
-
- final albums = useMemoized(() {
- return albumsQuery.pages
- .expand((page) => page.items ?? const Iterable.empty())
- .toList();
- }, [albumsQuery.pages]);
-
- final theme = Theme.of(context);
-
- return HorizontalPlaybuttonCardView(
- isLoadingNextPage: albumsQuery.isLoadingNextPage,
- hasNextPage: albumsQuery.hasNextPage,
- items: albums,
- onFetchMore: albumsQuery.fetchNext,
- title: Text(
- context.l10n.albums,
- style: theme.textTheme.headlineSmall,
- ),
- );
- }
-}
diff --git a/lib/components/shared/bordered_text.dart b/lib/components/bordered_text.dart
similarity index 97%
rename from lib/components/shared/bordered_text.dart
rename to lib/components/bordered_text.dart
index 627b2a3c..f25f2208 100644
--- a/lib/components/shared/bordered_text.dart
+++ b/lib/components/bordered_text.dart
@@ -79,7 +79,7 @@ class BorderedText extends StatelessWidget {
strutStyle: child.strutStyle,
textAlign: child.textAlign,
textDirection: child.textDirection,
- textScaleFactor: child.textScaleFactor,
+ textScaler: child.textScaler,
),
child,
],
diff --git a/lib/components/shared/compact_search.dart b/lib/components/compact_search.dart
similarity index 97%
rename from lib/components/shared/compact_search.dart
rename to lib/components/compact_search.dart
index 70815291..d37cb673 100644
--- a/lib/components/shared/compact_search.dart
+++ b/lib/components/compact_search.dart
@@ -11,12 +11,12 @@ class CompactSearch extends HookWidget {
final Color? iconColor;
const CompactSearch({
- Key? key,
+ super.key,
this.onChanged,
this.placeholder = "Search...",
this.icon = SpotubeIcons.search,
this.iconColor,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/desktop_login/login_form.dart b/lib/components/desktop_login/login_form.dart
deleted file mode 100644
index 5abb9524..00000000
--- a/lib/components/desktop_login/login_form.dart
+++ /dev/null
@@ -1,74 +0,0 @@
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotube/extensions/context.dart';
-
-import 'package:spotube/provider/authentication_provider.dart';
-
-class TokenLoginForm extends HookConsumerWidget {
- final void Function()? onDone;
- const TokenLoginForm({
- Key? key,
- this.onDone,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final authenticationNotifier =
- ref.watch(AuthenticationNotifier.provider.notifier);
- final directCodeController = useTextEditingController();
- final mounted = useIsMounted();
-
- final isLoading = useState(false);
-
- return ConstrainedBox(
- constraints: const BoxConstraints(
- maxWidth: 400,
- ),
- child: Column(
- children: [
- TextField(
- controller: directCodeController,
- decoration: InputDecoration(
- hintText: context.l10n.spotify_cookie("\"sp_dc\""),
- labelText: context.l10n.cookie_name_cookie("sp_dc"),
- ),
- keyboardType: TextInputType.visiblePassword,
- ),
- const SizedBox(height: 10),
- FilledButton(
- onPressed: isLoading.value
- ? null
- : () async {
- try {
- isLoading.value = true;
- if (directCodeController.text.isEmpty) {
- ScaffoldMessenger.of(context).showSnackBar(
- SnackBar(
- content: Text(context.l10n.fill_in_all_fields),
- behavior: SnackBarBehavior.floating,
- ),
- );
- return;
- }
- final cookieHeader =
- "sp_dc=${directCodeController.text.trim()}";
-
- authenticationNotifier.setCredentials(
- await AuthenticationCredentials.fromCookie(
- cookieHeader),
- );
- if (mounted()) {
- onDone?.call();
- }
- } finally {
- isLoading.value = false;
- }
- },
- child: Text(context.l10n.submit),
- )
- ],
- ),
- );
- }
-}
diff --git a/lib/components/shared/dialogs/confirm_download_dialog.dart b/lib/components/dialogs/confirm_download_dialog.dart
similarity index 93%
rename from lib/components/shared/dialogs/confirm_download_dialog.dart
rename to lib/components/dialogs/confirm_download_dialog.dart
index c371e803..897c64cb 100644
--- a/lib/components/shared/dialogs/confirm_download_dialog.dart
+++ b/lib/components/dialogs/confirm_download_dialog.dart
@@ -1,11 +1,11 @@
import 'package:flutter/material.dart';
-import 'package:spotube/components/shared/image/universal_image.dart';
+import 'package:spotube/components/image/universal_image.dart';
import 'package:spotube/extensions/constrains.dart';
import 'package:spotube/extensions/context.dart';
class ConfirmDownloadDialog extends StatelessWidget {
- const ConfirmDownloadDialog({Key? key}) : super(key: key);
+ const ConfirmDownloadDialog({super.key});
@override
Widget build(BuildContext context) {
@@ -82,7 +82,7 @@ class ConfirmDownloadDialog extends StatelessWidget {
class BulletPoint extends StatelessWidget {
final String text;
- const BulletPoint(this.text, {Key? key}) : super(key: key);
+ const BulletPoint(this.text, {super.key});
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/dialogs/piped_down_dialog.dart b/lib/components/dialogs/piped_down_dialog.dart
similarity index 96%
rename from lib/components/shared/dialogs/piped_down_dialog.dart
rename to lib/components/dialogs/piped_down_dialog.dart
index 6220adeb..b1717a2a 100644
--- a/lib/components/shared/dialogs/piped_down_dialog.dart
+++ b/lib/components/dialogs/piped_down_dialog.dart
@@ -5,7 +5,7 @@ import 'package:spotube/extensions/context.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
class PipedDownDialog extends HookConsumerWidget {
- const PipedDownDialog({Key? key}) : super(key: key);
+ const PipedDownDialog({super.key});
@override
Widget build(BuildContext context, ref) {
diff --git a/lib/components/shared/dialogs/playlist_add_track_dialog.dart b/lib/components/dialogs/playlist_add_track_dialog.dart
similarity index 72%
rename from lib/components/shared/dialogs/playlist_add_track_dialog.dart
rename to lib/components/dialogs/playlist_add_track_dialog.dart
index 51b77c76..5af9c9e4 100644
--- a/lib/components/shared/dialogs/playlist_add_track_dialog.dart
+++ b/lib/components/dialogs/playlist_add_track_dialog.dart
@@ -1,16 +1,14 @@
-import 'package:fl_query_hooks/fl_query_hooks.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:spotify/spotify.dart';
-import 'package:spotube/components/playlist/playlist_create_dialog.dart';
-import 'package:spotube/components/shared/image/universal_image.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/provider/spotify_provider.dart';
-import 'package:spotube/services/queries/queries.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
+import 'package:spotube/extensions/image.dart';
+import 'package:spotube/provider/spotify/spotify.dart';
class PlaylistAddTrackDialog extends HookConsumerWidget {
/// The id of the playlist this dialog was opened from
@@ -19,33 +17,40 @@ class PlaylistAddTrackDialog extends HookConsumerWidget {
const PlaylistAddTrackDialog({
required this.tracks,
required this.openFromPlaylist,
- Key? key,
- }) : super(key: key);
+ super.key,
+ });
@override
Widget build(BuildContext context, ref) {
final ThemeData(:textTheme) = Theme.of(context);
- final spotify = ref.watch(spotifyProvider);
- final userPlaylists = useQueries.playlist.ofMineAll(ref);
+ final userPlaylists = ref.watch(favoritePlaylistsProvider);
+ final favoritePlaylistsNotifier =
+ ref.watch(favoritePlaylistsProvider.notifier);
- final me = useQueries.user.me(ref);
+ final me = ref.watch(meProvider);
final filteredPlaylists = useMemoized(
() =>
- userPlaylists.data
- ?.where(
+ userPlaylists.asData?.value.items
+ .where(
(playlist) =>
playlist.owner?.id != null &&
- playlist.owner!.id == me.data?.id &&
+ playlist.owner!.id == me.asData?.value.id &&
playlist.id != openFromPlaylist,
)
.toList() ??
[],
- [userPlaylists.data, me.data?.id, openFromPlaylist],
+ [userPlaylists.asData?.value, me.asData?.value.id, openFromPlaylist],
);
final playlistsCheck = useState({});
- final queryClient = useQueryClient();
+
+ useEffect(() {
+ if (userPlaylists.asData?.value != null) {
+ favoritePlaylistsNotifier.fetchAll();
+ }
+ return null;
+ }, [userPlaylists.asData?.value]);
Future onAdd() async {
final selectedPlaylists = playlistsCheck.value.entries
@@ -54,21 +59,12 @@ class PlaylistAddTrackDialog extends HookConsumerWidget {
await Future.wait(
selectedPlaylists.map(
- (playlistId) => spotify.playlists.addTracks(
- tracks
- .map(
- (track) => track.uri!,
- )
- .toList(),
- playlistId),
+ (playlistId) => favoritePlaylistsNotifier.addTracks(
+ playlistId,
+ tracks.map((e) => e.id!).toList(),
+ ),
),
).then((_) => Navigator.pop(context, true));
-
- await queryClient.refreshQueries(
- selectedPlaylists
- .map((playlistId) => "playlist-tracks/$playlistId")
- .toList(),
- );
}
return AlertDialog(
@@ -109,8 +105,7 @@ class PlaylistAddTrackDialog extends HookConsumerWidget {
return CheckboxListTile(
secondary: CircleAvatar(
backgroundImage: UniversalImage.imageProvider(
- TypeConversionUtils.image_X_UrlString(
- playlist.images,
+ playlist.images.asUrlString(
placeholder: ImagePlaceholder.collection,
),
),
diff --git a/lib/components/shared/dialogs/prompt_dialog.dart b/lib/components/dialogs/prompt_dialog.dart
similarity index 100%
rename from lib/components/shared/dialogs/prompt_dialog.dart
rename to lib/components/dialogs/prompt_dialog.dart
diff --git a/lib/components/shared/dialogs/replace_downloaded_dialog.dart b/lib/components/dialogs/replace_downloaded_dialog.dart
similarity index 96%
rename from lib/components/shared/dialogs/replace_downloaded_dialog.dart
rename to lib/components/dialogs/replace_downloaded_dialog.dart
index 77721041..00461d34 100644
--- a/lib/components/shared/dialogs/replace_downloaded_dialog.dart
+++ b/lib/components/dialogs/replace_downloaded_dialog.dart
@@ -8,8 +8,7 @@ final replaceDownloadedFileState = StateProvider((ref) => null);
class ReplaceDownloadedDialog extends ConsumerWidget {
final Track track;
- const ReplaceDownloadedDialog({required this.track, Key? key})
- : super(key: key);
+ const ReplaceDownloadedDialog({required this.track, super.key});
@override
Widget build(BuildContext context, ref) {
diff --git a/lib/components/dialogs/select_device_dialog.dart b/lib/components/dialogs/select_device_dialog.dart
new file mode 100644
index 00000000..3a3bde60
--- /dev/null
+++ b/lib/components/dialogs/select_device_dialog.dart
@@ -0,0 +1,67 @@
+import 'package:flutter/material.dart';
+import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:spotube/extensions/context.dart';
+import 'package:spotube/provider/connect/clients.dart';
+
+class SelectDeviceDialog extends HookConsumerWidget {
+ const SelectDeviceDialog({super.key});
+
+ @override
+ Widget build(BuildContext context, ref) {
+ final isRemoteService = useState(false);
+
+ final connectClients = ref.watch(connectClientsProvider);
+ final remoteService = connectClients.asData!.value.resolvedService!;
+
+ 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!;
+ },
+ ),
+ ],
+ ),
+ actions: [
+ TextButton(
+ onPressed: () {
+ Navigator.of(context).pop(isRemoteService.value);
+ },
+ child: Text(context.l10n.select),
+ ),
+ ],
+ );
+ }
+}
+
+Future showSelectDeviceDialog(BuildContext context, WidgetRef ref) async {
+ final connectClients = ref.read(connectClientsProvider);
+
+ if (connectClients.asData?.value.resolvedService == null) {
+ return false;
+ }
+
+ final isRemote = await showDialog(
+ context: context,
+ builder: (context) => const SelectDeviceDialog(),
+ );
+
+ return isRemote ?? false;
+}
diff --git a/lib/components/shared/dialogs/track_details_dialog.dart b/lib/components/dialogs/track_details_dialog.dart
similarity index 93%
rename from lib/components/shared/dialogs/track_details_dialog.dart
rename to lib/components/dialogs/track_details_dialog.dart
index 8634776f..61bca7b1 100644
--- a/lib/components/shared/dialogs/track_details_dialog.dart
+++ b/lib/components/dialogs/track_details_dialog.dart
@@ -2,20 +2,20 @@ 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/shared/links/hyper_link.dart';
-import 'package:spotube/components/shared/links/link_text.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/utils/type_conversion_utils.dart';
import 'package:spotube/extensions/duration.dart';
class TrackDetailsDialog extends HookWidget {
final Track track;
const TrackDetailsDialog({
- Key? key,
+ super.key,
required this.track,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
@@ -24,10 +24,11 @@ class TrackDetailsDialog extends HookWidget {
final detailsMap = {
context.l10n.title: track.name!,
- context.l10n.artist: TypeConversionUtils.artists_X_ClickableArtists(
- track.artists ?? [],
+ context.l10n.artist: ArtistLink(
+ artists: track.artists ?? [],
mainAxisAlignment: WrapAlignment.start,
textStyle: const TextStyle(color: Colors.blue),
+ hideOverflowArtist: false,
),
context.l10n.album: LinkText(
track.album!.name!,
diff --git a/lib/components/shared/expandable_search/expandable_search.dart b/lib/components/expandable_search/expandable_search.dart
similarity index 97%
rename from lib/components/shared/expandable_search/expandable_search.dart
rename to lib/components/expandable_search/expandable_search.dart
index 75ac6841..157e180f 100644
--- a/lib/components/shared/expandable_search/expandable_search.dart
+++ b/lib/components/expandable_search/expandable_search.dart
@@ -10,12 +10,12 @@ class ExpandableSearchField extends StatelessWidget {
final FocusNode searchFocus;
const ExpandableSearchField({
- Key? key,
+ super.key,
required this.isFiltering,
required this.onChangeFiltering,
required this.searchController,
required this.searchFocus,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
@@ -60,12 +60,12 @@ class ExpandableSearchButton extends StatelessWidget {
final ValueChanged? onPressed;
const ExpandableSearchButton({
- Key? key,
+ super.key,
required this.isFiltering,
required this.searchFocus,
this.icon = const Icon(SpotubeIcons.filter),
this.onPressed,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/fallbacks/anonymous_fallback.dart b/lib/components/fallbacks/anonymous_fallback.dart
similarity index 60%
rename from lib/components/shared/fallbacks/anonymous_fallback.dart
rename to lib/components/fallbacks/anonymous_fallback.dart
index aea7bf38..62ed8ddd 100644
--- a/lib/components/shared/fallbacks/anonymous_fallback.dart
+++ b/lib/components/fallbacks/anonymous_fallback.dart
@@ -1,22 +1,27 @@
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:spotube/extensions/context.dart';
+import 'package:spotube/pages/settings/settings.dart';
-import 'package:spotube/provider/authentication_provider.dart';
+import 'package:spotube/provider/authentication/authentication.dart';
import 'package:spotube/utils/service_utils.dart';
class AnonymousFallback extends ConsumerWidget {
final Widget? child;
const AnonymousFallback({
- Key? key,
+ super.key,
this.child,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context, ref) {
- final isLoggedIn = ref.watch(AuthenticationNotifier.provider) != null;
+ final isLoggedIn = ref.watch(authenticationProvider);
- if (isLoggedIn && child != null) return child!;
+ if (isLoggedIn.isLoading) {
+ return const Center(child: CircularProgressIndicator());
+ }
+
+ if (isLoggedIn.asData?.value != null && child != null) return child!;
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
@@ -25,7 +30,7 @@ class AnonymousFallback extends ConsumerWidget {
const SizedBox(height: 10),
FilledButton(
child: Text(context.l10n.login_with_spotify),
- onPressed: () => ServiceUtils.push(context, "/settings"),
+ onPressed: () => ServiceUtils.pushNamed(context, SettingsPage.name),
)
],
),
diff --git a/lib/components/shared/fallbacks/not_found.dart b/lib/components/fallbacks/not_found.dart
similarity index 75%
rename from lib/components/shared/fallbacks/not_found.dart
rename to lib/components/fallbacks/not_found.dart
index f45573ad..ce168f17 100644
--- a/lib/components/shared/fallbacks/not_found.dart
+++ b/lib/components/fallbacks/not_found.dart
@@ -1,9 +1,10 @@
import 'package:flutter/material.dart';
import 'package:spotube/collections/assets.gen.dart';
+import 'package:spotube/extensions/context.dart';
class NotFound extends StatelessWidget {
final bool vertical;
- const NotFound({Key? key, this.vertical = false}) : super(key: key);
+ const NotFound({super.key, this.vertical = false});
@override
Widget build(BuildContext context) {
@@ -18,9 +19,9 @@ class NotFound extends StatelessWidget {
crossAxisAlignment: CrossAxisAlignment.start,
mainAxisAlignment: MainAxisAlignment.center,
children: [
- Text("Nothing found", style: theme.textTheme.titleLarge),
+ Text(context.l10n.nothing_found, style: theme.textTheme.titleLarge),
Text(
- "The box is empty",
+ context.l10n.the_box_is_empty,
style: theme.textTheme.titleMedium,
),
],
diff --git a/lib/components/framework/app_pop_scope.dart b/lib/components/framework/app_pop_scope.dart
new file mode 100644
index 00000000..b8e35767
--- /dev/null
+++ b/lib/components/framework/app_pop_scope.dart
@@ -0,0 +1,104 @@
+import 'dart:io';
+
+import 'package:flutter/material.dart';
+
+/// A temporary workaround for [WillPopScope] and [PopScope] not working in GoRouter
+/// https://github.com/flutter/flutter/issues/140869#issuecomment-2247181468
+class AppPopScope extends StatefulWidget {
+ final Widget child;
+
+ final PopInvokedCallback? onPopInvoked;
+
+ final bool canPop;
+
+ const AppPopScope({
+ super.key,
+ required this.child,
+ this.canPop = true,
+ this.onPopInvoked,
+ });
+
+ @override
+ State createState() => _AppPopScopeState();
+}
+
+class _AppPopScopeState extends State {
+ final bool _enable = Platform.isAndroid;
+ ModalRoute? _route;
+ BackButtonDispatcher? _parentBackBtnDispatcher;
+ ChildBackButtonDispatcher? _backBtnDispatcher;
+
+ @override
+ void didChangeDependencies() {
+ super.didChangeDependencies();
+ _route = ModalRoute.of(context);
+ _updateBackButtonDispatcher();
+ }
+
+ @override
+ void activate() {
+ super.activate();
+ _updateBackButtonDispatcher();
+ }
+
+ @override
+ void deactivate() {
+ super.deactivate();
+ _disposeBackBtnDispatcher();
+ }
+
+ @override
+ void dispose() {
+ _disposeBackBtnDispatcher();
+ super.dispose();
+ }
+
+ @override
+ Widget build(BuildContext context) {
+ return PopScope(
+ canPop: widget.canPop,
+ onPopInvoked: widget.onPopInvoked,
+ child: widget.child,
+ );
+ }
+
+ void _updateBackButtonDispatcher() {
+ if (!_enable) return;
+
+ var dispatcher = Router.maybeOf(context)?.backButtonDispatcher;
+ if (dispatcher != _parentBackBtnDispatcher) {
+ _disposeBackBtnDispatcher();
+ _parentBackBtnDispatcher = dispatcher;
+ if (dispatcher is BackButtonDispatcher &&
+ dispatcher is! ChildBackButtonDispatcher) {
+ dispatcher = dispatcher.createChildBackButtonDispatcher();
+ }
+ _backBtnDispatcher = dispatcher as ChildBackButtonDispatcher;
+ }
+ _backBtnDispatcher?.removeCallback(_handleBackButton);
+ _backBtnDispatcher?.addCallback(_handleBackButton);
+ _backBtnDispatcher?.takePriority();
+ }
+
+ void _disposeBackBtnDispatcher() {
+ _backBtnDispatcher?.removeCallback(_handleBackButton);
+ if (_backBtnDispatcher is ChildBackButtonDispatcher) {
+ final child = _backBtnDispatcher as ChildBackButtonDispatcher;
+ _parentBackBtnDispatcher?.forget(child);
+ }
+ _backBtnDispatcher = null;
+ _parentBackBtnDispatcher = null;
+ }
+
+ bool get _onlyRoute => _route != null && _route!.isFirst && _route!.isCurrent;
+
+ Future _handleBackButton() async {
+ if (_onlyRoute) {
+ widget.onPopInvoked?.call(widget.canPop);
+ if (!widget.canPop) {
+ return true;
+ }
+ }
+ return false;
+ }
+}
diff --git a/lib/components/heart_button/heart_button.dart b/lib/components/heart_button/heart_button.dart
new file mode 100644
index 00000000..fa4318cc
--- /dev/null
+++ b/lib/components/heart_button/heart_button.dart
@@ -0,0 +1,86 @@
+import 'package:flutter/material.dart';
+import 'package:hooks_riverpod/hooks_riverpod.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';
+
+class HeartButton extends HookConsumerWidget {
+ final bool isLiked;
+ final void Function()? onPressed;
+ final IconData? icon;
+ final Color? color;
+ final String? tooltip;
+ const HeartButton({
+ required this.isLiked,
+ required this.onPressed,
+ this.color,
+ this.tooltip,
+ this.icon,
+ super.key,
+ });
+
+ @override
+ Widget build(BuildContext context, ref) {
+ final auth = ref.watch(authenticationProvider);
+
+ if (auth.asData?.value == null) 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),
+ ),
+ ),
+ onPressed: onPressed,
+ );
+ }
+}
+
+class TrackHeartButton extends HookConsumerWidget {
+ final Track track;
+ const TrackHeartButton({
+ super.key,
+ required this.track,
+ });
+
+ @override
+ Widget build(BuildContext context, ref) {
+ final savedTracks = ref.watch(likedTracksProvider);
+ final me = ref.watch(meProvider);
+ final (:isLiked, :toggleTrackLike) = useTrackToggleLike(track, ref);
+
+ if (me.isLoading) {
+ return const CircularProgressIndicator();
+ }
+
+ return HeartButton(
+ tooltip: isLiked
+ ? context.l10n.remove_from_favorites
+ : context.l10n.save_as_favorite,
+ isLiked: isLiked,
+ onPressed: savedTracks.asData?.value != 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
new file mode 100644
index 00000000..ba5cbee1
--- /dev/null
+++ b/lib/components/heart_button/use_track_toggle_like.dart
@@ -0,0 +1,37 @@
+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';
+
+typedef UseTrackToggleLike = ({
+ bool isLiked,
+ Future Function(Track track) toggleTrackLike,
+});
+
+UseTrackToggleLike useTrackToggleLike(Track track, WidgetRef ref) {
+ final savedTracks = ref.watch(likedTracksProvider);
+ final savedTracksNotifier = ref.watch(likedTracksProvider.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);
+
+ return (
+ isLiked: isLiked,
+ toggleTrackLike: (track) async {
+ await savedTracksNotifier.toggleFavorite(track);
+
+ if (!isLiked) {
+ await scrobblerNotifier.love(track);
+ } else {
+ await scrobblerNotifier.unlove(track);
+ }
+ },
+ );
+}
diff --git a/lib/components/home/sections/featured.dart b/lib/components/home/sections/featured.dart
deleted file mode 100644
index 8a7c2c95..00000000
--- a/lib/components/home/sections/featured.dart
+++ /dev/null
@@ -1,36 +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:skeletonizer/skeletonizer.dart';
-import 'package:spotify/spotify.dart';
-import 'package:spotube/components/shared/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/services/queries/queries.dart';
-
-class HomeFeaturedSection extends HookConsumerWidget {
- const HomeFeaturedSection({Key? key}) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final featuredPlaylistsQuery = useQueries.playlist.featured(ref);
- final playlists = useMemoized(
- () => featuredPlaylistsQuery.pages
- .whereType>()
- .expand((page) => page.items ?? const []),
- [featuredPlaylistsQuery.pages],
- );
- final isLoadingFeaturedPlaylists = !featuredPlaylistsQuery.hasPageData &&
- !featuredPlaylistsQuery.isLoadingNextPage;
-
- return Skeletonizer(
- enabled: isLoadingFeaturedPlaylists,
- child: HorizontalPlaybuttonCardView(
- items: playlists.toList(),
- title: Text(context.l10n.featured),
- isLoadingNextPage: featuredPlaylistsQuery.isLoadingNextPage,
- hasNextPage: featuredPlaylistsQuery.hasNextPage,
- onFetchMore: featuredPlaylistsQuery.fetchNext,
- ),
- );
- }
-}
diff --git a/lib/components/home/sections/new_releases.dart b/lib/components/home/sections/new_releases.dart
deleted file mode 100644
index 0f4a046a..00000000
--- a/lib/components/home/sections/new_releases.dart
+++ /dev/null
@@ -1,56 +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/shared/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/provider/authentication_provider.dart';
-import 'package:spotube/services/queries/queries.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
-
-class HomeNewReleasesSection extends HookConsumerWidget {
- const HomeNewReleasesSection({Key? key}) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final auth = ref.watch(AuthenticationNotifier.provider);
-
- final newReleases = useQueries.album.newReleases(ref);
- final userArtistsQuery = useQueries.artist.followedByMeAll(ref);
- final userArtists =
- userArtistsQuery.data?.map((s) => s.id!).toList() ?? const [];
-
- final albums = useMemoized(
- () {
- final allReleases = newReleases.pages
- .whereType>()
- .expand((page) => page.items ?? const [])
- .map((album) => TypeConversionUtils.simpleAlbum_X_Album(album));
-
- final userArtistReleases = allReleases.where((album) {
- return album.artists
- ?.any((artist) => userArtists.contains(artist.id!)) ==
- true;
- }).toList();
-
- if (userArtistReleases.isEmpty) return allReleases.toList();
- return userArtistReleases;
- },
- [newReleases.pages],
- );
-
- final hasNewReleases = newReleases.hasPageData &&
- userArtistsQuery.hasData &&
- !newReleases.isLoadingNextPage;
-
- if (auth == null || !hasNewReleases) return const SizedBox.shrink();
-
- return HorizontalPlaybuttonCardView(
- items: albums,
- title: Text(context.l10n.new_releases),
- isLoadingNextPage: newReleases.isLoadingNextPage,
- hasNextPage: newReleases.hasNextPage,
- onFetchMore: newReleases.fetchNext,
- );
- }
-}
diff --git a/lib/components/shared/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart b/lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart
similarity index 76%
rename from lib/components/shared/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart
rename to lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart
index dc9d30da..16204952 100644
--- a/lib/components/shared/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart
+++ b/lib/components/horizontal_playbutton_card_view/horizontal_playbutton_card_view.dart
@@ -5,9 +5,9 @@ import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:skeletonizer/skeletonizer.dart';
import 'package:spotify/spotify.dart';
import 'package:spotube/collections/fake.dart';
-import 'package:spotube/components/album/album_card.dart';
-import 'package:spotube/components/artist/artist_card.dart';
-import 'package:spotube/components/playlist/playlist_card.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';
@@ -17,20 +17,22 @@ class HorizontalPlaybuttonCardView extends HookWidget {
final VoidCallback onFetchMore;
final bool isLoadingNextPage;
final bool hasNextPage;
+ final Widget? titleTrailing;
- const HorizontalPlaybuttonCardView({
+ HorizontalPlaybuttonCardView({
required this.title,
required this.items,
required this.hasNextPage,
required this.onFetchMore,
required this.isLoadingNextPage,
- Key? key,
- }) : assert(
- items is List ||
- items is List ||
- items is List,
- ),
- super(key: key);
+ this.titleTrailing,
+ super.key,
+ }) : assert(
+ items.every(
+ (item) =>
+ item is PlaylistSimple || item is Artist || item is AlbumSimple,
+ ),
+ );
@override
Widget build(BuildContext context) {
@@ -49,9 +51,15 @@ class HorizontalPlaybuttonCardView extends HookWidget {
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
- DefaultTextStyle(
- style: textTheme.titleMedium!,
- child: title,
+ Row(
+ mainAxisAlignment: MainAxisAlignment.spaceBetween,
+ children: [
+ DefaultTextStyle(
+ style: textTheme.titleMedium!,
+ child: title,
+ ),
+ if (titleTrailing != null) titleTrailing!,
+ ],
),
SizedBox(
height: height,
@@ -85,11 +93,11 @@ class HorizontalPlaybuttonCardView extends HookWidget {
itemBuilder: (context, index) {
final item = items[index];
- return switch (item.runtimeType) {
- PlaylistSimple =>
+ return switch (item) {
+ PlaylistSimple() =>
PlaylistCard(item as PlaylistSimple),
- Album => AlbumCard(item as Album),
- Artist => Padding(
+ AlbumSimple() => AlbumCard(item as AlbumSimple),
+ Artist() => Padding(
padding: const EdgeInsets.symmetric(
horizontal: 12.0),
child: ArtistCard(item as Artist),
diff --git a/lib/components/shared/hover_builder.dart b/lib/components/hover_builder.dart
similarity index 95%
rename from lib/components/shared/hover_builder.dart
rename to lib/components/hover_builder.dart
index ec60848e..7793e744 100644
--- a/lib/components/shared/hover_builder.dart
+++ b/lib/components/hover_builder.dart
@@ -7,8 +7,8 @@ class HoverBuilder extends HookWidget {
const HoverBuilder({
required this.builder,
this.permanentState,
- Key? key,
- }) : super(key: key);
+ super.key,
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/image/universal_image.dart b/lib/components/image/universal_image.dart
similarity index 98%
rename from lib/components/shared/image/universal_image.dart
rename to lib/components/image/universal_image.dart
index 04c62478..d8902e63 100644
--- a/lib/components/shared/image/universal_image.dart
+++ b/lib/components/image/universal_image.dart
@@ -20,8 +20,8 @@ class UniversalImage extends HookWidget {
this.placeholder,
this.fit,
this.scale = 1,
- Key? key,
- }) : super(key: key);
+ super.key,
+ });
static ImageProvider imageProvider(
String path, {
diff --git a/lib/components/shared/inter_scrollbar/inter_scrollbar.dart b/lib/components/inter_scrollbar/inter_scrollbar.dart
similarity index 80%
rename from lib/components/shared/inter_scrollbar/inter_scrollbar.dart
rename to lib/components/inter_scrollbar/inter_scrollbar.dart
index 2b3ce319..8a86b643 100644
--- a/lib/components/shared/inter_scrollbar/inter_scrollbar.dart
+++ b/lib/components/inter_scrollbar/inter_scrollbar.dart
@@ -1,7 +1,7 @@
import 'package:draggable_scrollbar/draggable_scrollbar.dart';
import 'package:flutter/material.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:spotube/utils/platform.dart';
class InterScrollbar extends HookWidget {
final Widget child;
@@ -15,7 +15,7 @@ class InterScrollbar extends HookWidget {
@override
Widget build(BuildContext context) {
- if (DesktopTools.platform.isDesktop) return child;
+ if (kIsDesktop) return child;
return DraggableScrollbar.semicircle(
controller: controller,
diff --git a/lib/components/library/user_albums.dart b/lib/components/library/user_albums.dart
deleted file mode 100644
index 200d1c59..00000000
--- a/lib/components/library/user_albums.dart
+++ /dev/null
@@ -1,128 +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: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/components/album/album_card.dart';
-import 'package:spotube/components/shared/fallbacks/not_found.dart';
-import 'package:spotube/components/shared/inter_scrollbar/inter_scrollbar.dart';
-import 'package:spotube/components/shared/fallbacks/anonymous_fallback.dart';
-import 'package:spotube/components/shared/waypoint.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/provider/authentication_provider.dart';
-import 'package:spotube/services/queries/queries.dart';
-
-import 'package:spotube/utils/type_conversion_utils.dart';
-
-class UserAlbums extends HookConsumerWidget {
- const UserAlbums({Key? key}) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final auth = ref.watch(AuthenticationNotifier.provider);
- final albumsQuery = useQueries.album.ofMine(ref);
-
- final controller = useScrollController();
-
- final searchText = useState('');
-
- final allAlbums = useMemoized(
- () => albumsQuery.pages
- .expand((element) => element.items ?? []),
- [albumsQuery.pages],
- );
-
- final albums = useMemoized(() {
- if (searchText.value.isEmpty) {
- return allAlbums;
- }
- return allAlbums
- .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();
- }, [allAlbums, searchText.value]);
-
- if (auth == null) {
- return const AnonymousFallback();
- }
-
- final theme = Theme.of(context);
-
- return RefreshIndicator(
- onRefresh: () async {
- await albumsQuery.refresh();
- },
- child: SafeArea(
- child: Scaffold(
- appBar: PreferredSize(
- preferredSize: const Size.fromHeight(50),
- child: Padding(
- padding: const EdgeInsets.symmetric(horizontal: 8.0),
- child: ColoredBox(
- color: theme.scaffoldBackgroundColor,
- child: SearchBar(
- onChanged: (value) => searchText.value = value,
- leading: const Icon(SpotubeIcons.filter),
- hintText: context.l10n.filter_albums,
- ),
- ),
- ),
- ),
- body: SizedBox.expand(
- child: InterScrollbar(
- controller: controller,
- child: SingleChildScrollView(
- padding: const EdgeInsets.all(8.0),
- controller: controller,
- child: Skeletonizer(
- enabled: albumsQuery.pages.isEmpty,
- child: Center(
- child: Wrap(
- runSpacing: 20,
- alignment: WrapAlignment.center,
- runAlignment: WrapAlignment.center,
- crossAxisAlignment: WrapCrossAlignment.center,
- children: [
- if (albumsQuery.pages.isEmpty)
- ...List.generate(
- 10,
- (index) => AlbumCard(FakeData.album),
- )
- else if (albums.isEmpty)
- const Row(
- mainAxisAlignment: MainAxisAlignment.center,
- children: [NotFound()],
- ),
- for (final album in albums)
- AlbumCard(
- TypeConversionUtils.simpleAlbum_X_Album(album),
- ),
- if (albums.isNotEmpty && albumsQuery.hasNextPage)
- Waypoint(
- controller: controller,
- isGrid: true,
- onTouchEdge: albumsQuery.fetchNext,
- child: AlbumCard(FakeData.album),
- )
- ],
- ),
- ),
- ),
- ),
- ),
- ),
- ),
- ),
- );
- }
-}
diff --git a/lib/components/library/user_artists.dart b/lib/components/library/user_artists.dart
deleted file mode 100644
index 36b8528e..00000000
--- a/lib/components/library/user_artists.dart
+++ /dev/null
@@ -1,125 +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: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/shared/fallbacks/anonymous_fallback.dart';
-import 'package:spotube/components/artist/artist_card.dart';
-import 'package:spotube/components/shared/fallbacks/not_found.dart';
-import 'package:spotube/components/shared/inter_scrollbar/inter_scrollbar.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/provider/authentication_provider.dart';
-import 'package:spotube/services/queries/queries.dart';
-
-class UserArtists extends HookConsumerWidget {
- const UserArtists({Key? key}) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final theme = Theme.of(context);
- final auth = ref.watch(AuthenticationNotifier.provider);
-
- final artistQuery = useQueries.artist.followedByMeAll(ref);
-
- final searchText = useState('');
-
- final filteredArtists = useMemoized(() {
- final artists = artistQuery.data ?? [];
-
- 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.data, searchText.value]);
-
- final controller = useScrollController();
-
- if (auth == null) {
- return const AnonymousFallback();
- }
-
- return Scaffold(
- appBar: PreferredSize(
- preferredSize: const Size.fromHeight(50),
- child: Padding(
- padding: const EdgeInsets.symmetric(horizontal: 8.0),
- child: ColoredBox(
- color: theme.scaffoldBackgroundColor,
- child: SearchBar(
- onChanged: (value) => searchText.value = value,
- leading: const Icon(SpotubeIcons.filter),
- hintText: context.l10n.filter_artist,
- ),
- ),
- ),
- ),
- backgroundColor: theme.scaffoldBackgroundColor,
- body: artistQuery.data?.isEmpty == true
- ? Padding(
- padding: const EdgeInsets.all(20),
- child: Row(
- mainAxisAlignment: MainAxisAlignment.center,
- children: [
- const CircularProgressIndicator(),
- const SizedBox(width: 10),
- Text(context.l10n.loading),
- ],
- ),
- )
- : RefreshIndicator(
- onRefresh: () async {
- await artistQuery.refresh();
- },
- child: InterScrollbar(
- controller: controller,
- child: SingleChildScrollView(
- controller: controller,
- child: SizedBox(
- width: double.infinity,
- child: SafeArea(
- child: Center(
- child: Skeletonizer(
- enabled: artistQuery.isLoading,
- child: Wrap(
- spacing: 15,
- runSpacing: 5,
- children: artistQuery.isLoading
- ? List.generate(
- 10, (index) => ArtistCard(FakeData.artist))
- : filteredArtists.isEmpty
- ? [
- const Row(
- mainAxisAlignment:
- MainAxisAlignment.center,
- children: [
- NotFound(),
- ],
- )
- ]
- : filteredArtists
- .mapIndexed((index, artist) =>
- ArtistCard(artist))
- .toList(),
- ),
- ),
- ),
- ),
- ),
- ),
- ),
- ),
- );
- }
-}
diff --git a/lib/components/library/user_local_tracks.dart b/lib/components/library/user_local_tracks.dart
deleted file mode 100644
index 095e6e97..00000000
--- a/lib/components/library/user_local_tracks.dart
+++ /dev/null
@@ -1,324 +0,0 @@
-import 'dart:io';
-
-import 'package:catcher_2/catcher_2.dart';
-import 'package:flutter/foundation.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:collection/collection.dart';
-import 'package:fuzzywuzzy/fuzzywuzzy.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:metadata_god/metadata_god.dart';
-import 'package:mime/mime.dart';
-import 'package:path/path.dart';
-import 'package:path_provider/path_provider.dart';
-import 'package:skeletonizer/skeletonizer.dart';
-
-import 'package:spotify/spotify.dart';
-import 'package:spotube/collections/fake.dart';
-import 'package:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/shared/expandable_search/expandable_search.dart';
-import 'package:spotube/components/shared/fallbacks/not_found.dart';
-import 'package:spotube/components/shared/inter_scrollbar/inter_scrollbar.dart';
-import 'package:spotube/components/shared/sort_tracks_dropdown.dart';
-import 'package:spotube/components/shared/track_tile/track_tile.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/models/local_track.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
-import 'package:spotube/utils/service_utils.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
-import 'package:flutter_rust_bridge/flutter_rust_bridge.dart' show FfiException;
-
-const supportedAudioTypes = [
- "audio/webm",
- "audio/ogg",
- "audio/mpeg",
- "audio/mp4",
- "audio/opus",
- "audio/wav",
- "audio/aac",
-];
-
-const imgMimeToExt = {
- "image/png": ".png",
- "image/jpeg": ".jpg",
- "image/webp": ".webp",
- "image/gif": ".gif",
-};
-
-enum SortBy {
- none,
- ascending,
- descending,
- newest,
- oldest,
- duration,
- artist,
- album,
-}
-
-final localTracksProvider = FutureProvider>((ref) async {
- try {
- if (kIsWeb) return [];
- final downloadLocation = ref.watch(
- userPreferencesProvider.select((s) => s.downloadLocation),
- );
- if (downloadLocation.isEmpty) return [];
- final downloadDir = Directory(downloadLocation);
- if (!await downloadDir.exists()) {
- await downloadDir.create(recursive: true);
- return [];
- }
- final entities = downloadDir.listSync(recursive: true);
-
- final filesWithMetadata = (await Future.wait(
- entities.map((e) => File(e.path)).where((file) {
- final mimetype = lookupMimeType(file.path);
- return mimetype != null && supportedAudioTypes.contains(mimetype);
- }).map(
- (file) async {
- try {
- final metadata = await MetadataGod.readMetadata(file: file.path);
-
- final imageFile = File(join(
- (await getTemporaryDirectory()).path,
- "spotube",
- basenameWithoutExtension(file.path) +
- imgMimeToExt[metadata.picture?.mimeType ?? "image/jpeg"]!,
- ));
- if (!await imageFile.exists() && metadata.picture != null) {
- await imageFile.create(recursive: true);
- await imageFile.writeAsBytes(
- metadata.picture?.data ?? [],
- mode: FileMode.writeOnly,
- );
- }
-
- return {"metadata": metadata, "file": file, "art": imageFile.path};
- } catch (e, stack) {
- if (e is FfiException) {
- return {"file": file};
- }
- Catcher2.reportCheckedError(e, stack);
- return {};
- }
- },
- ),
- ))
- .where((e) => e.isNotEmpty)
- .toList();
-
- final tracks = filesWithMetadata
- .map(
- (fileWithMetadata) => LocalTrack.fromTrack(
- track: TypeConversionUtils.localTrack_X_Track(
- fileWithMetadata["file"],
- metadata: fileWithMetadata["metadata"],
- art: fileWithMetadata["art"],
- ),
- path: fileWithMetadata["file"].path,
- ),
- )
- .toList();
-
- return tracks;
- } catch (e, stack) {
- Catcher2.reportCheckedError(e, stack);
- return [];
- }
-});
-
-class UserLocalTracks extends HookConsumerWidget {
- const UserLocalTracks({Key? key}) : super(key: key);
-
- Future playLocalTracks(
- WidgetRef ref,
- List tracks, {
- LocalTrack? currentTrack,
- }) async {
- final playlist = ref.read(ProxyPlaylistNotifier.provider);
- final playback = ref.read(ProxyPlaylistNotifier.notifier);
- currentTrack ??= tracks.first;
- final isPlaylistPlaying = playlist.containsTracks(tracks);
- if (!isPlaylistPlaying) {
- await playback.load(
- tracks,
- initialIndex: tracks.indexWhere((s) => s.id == currentTrack?.id),
- autoPlay: true,
- );
- } else if (isPlaylistPlaying &&
- currentTrack.id != null &&
- currentTrack.id != playlist.activeTrack?.id) {
- await playback.jumpToTrack(currentTrack);
- }
- }
-
- @override
- Widget build(BuildContext context, ref) {
- final sortBy = useState(SortBy.none);
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final trackSnapshot = ref.watch(localTracksProvider);
- final isPlaylistPlaying =
- playlist.containsTracks(trackSnapshot.value ?? []);
-
- final searchController = useTextEditingController();
- useValueListenable(searchController);
- final searchFocus = useFocusNode();
- final isFiltering = useState(false);
-
- final controller = useScrollController();
-
- return Column(
- children: [
- Padding(
- padding: const EdgeInsets.all(8.0),
- child: Row(
- children: [
- const SizedBox(width: 10),
- FilledButton(
- onPressed: trackSnapshot.value != null
- ? () async {
- if (trackSnapshot.value?.isNotEmpty == true) {
- if (!isPlaylistPlaying) {
- await playLocalTracks(
- ref,
- trackSnapshot.value!,
- );
- } else {
- // TODO: Remove stop capability
- // playlistNotifier.stop();
- }
- }
- }
- : 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: 10),
- FilledButton(
- child: const Icon(SpotubeIcons.refresh),
- onPressed: () {
- ref.refresh(localTracksProvider);
- },
- )
- ],
- ),
- ),
- ExpandableSearchField(
- searchController: searchController,
- searchFocus: searchFocus,
- isFiltering: isFiltering.value,
- onChangeFiltering: (value) => isFiltering.value = value,
- ),
- trackSnapshot.when(
- data: (tracks) {
- final sortedTracks = useMemoized(() {
- return ServiceUtils.sortTracks(tracks, sortBy.value);
- }, [sortBy.value, tracks]);
-
- final filteredTracks = useMemoized(() {
- if (searchController.text.isEmpty) {
- return sortedTracks;
- }
- return sortedTracks
- .map((e) => (
- weightedRatio(
- "${e.name} - ${TypeConversionUtils.artists_X_String(e.artists ?? [])}",
- 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.refresh(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(track: FakeData.track, index: index);
- }
-
- final track = filteredTracks[index];
- return TrackTile(
- index: index,
- 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),
- ),
- ),
- ),
- error: (error, stackTrace) =>
- Text(error.toString() + stackTrace.toString()),
- )
- ],
- );
- }
-}
diff --git a/lib/components/shared/links/anchor_button.dart b/lib/components/links/anchor_button.dart
similarity index 92%
rename from lib/components/shared/links/anchor_button.dart
rename to lib/components/links/anchor_button.dart
index b1b1cfea..c6f0b889 100644
--- a/lib/components/shared/links/anchor_button.dart
+++ b/lib/components/links/anchor_button.dart
@@ -11,13 +11,13 @@ class AnchorButton extends HookWidget {
const AnchorButton(
this.text, {
- Key? key,
+ super.key,
this.onTap,
this.textAlign,
this.overflow,
this.maxLines,
this.style = const TextStyle(),
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
@@ -29,7 +29,7 @@ class AnchorButton extends HookWidget {
onTapUp: (event) => tap.value = false,
onTap: onTap,
child: MouseRegion(
- cursor: MaterialStateMouseCursor.clickable,
+ cursor: WidgetStateMouseCursor.clickable,
child: Text(
text,
style: style.copyWith(
diff --git a/lib/components/links/artist_link.dart b/lib/components/links/artist_link.dart
new file mode 100644
index 00000000..9f06f1b3
--- /dev/null
+++ b/lib/components/links/artist_link.dart
@@ -0,0 +1,81 @@
+import 'package:flutter/material.dart';
+import 'package:spotify/spotify.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';
+
+class ArtistLink extends StatelessWidget {
+ final List artists;
+ final WrapCrossAlignment crossAxisAlignment;
+ final WrapAlignment mainAxisAlignment;
+ final TextStyle textStyle;
+ final bool hideOverflowArtist;
+ final void Function(String route)? onRouteChange;
+ final VoidCallback? onOverflowArtistClick;
+
+ const ArtistLink({
+ super.key,
+ required this.artists,
+ this.crossAxisAlignment = WrapCrossAlignment.center,
+ this.mainAxisAlignment = WrapAlignment.center,
+ this.textStyle = const TextStyle(),
+ this.onRouteChange,
+ this.hideOverflowArtist = true,
+ this.onOverflowArtistClick,
+ }) : assert(hideOverflowArtist ? onOverflowArtistClick != null : true);
+
+ @override
+ Widget build(BuildContext context) {
+ final ThemeData(:colorScheme) = Theme.of(context);
+
+ return Wrap(
+ crossAxisAlignment: crossAxisAlignment,
+ alignment: mainAxisAlignment,
+ children: [
+ ...(hideOverflowArtist ? artists.take(3).toList() : artists)
+ .asMap()
+ .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!,
+ onTap: () {
+ if (onRouteChange != null) {
+ onRouteChange?.call("/artist/${artist.value.id}");
+ } else {
+ ServiceUtils.pushNamed(
+ context,
+ ArtistPage.name,
+ pathParameters: {
+ "id": artist.value.id!,
+ },
+ );
+ }
+ },
+ overflow: TextOverflow.ellipsis,
+ style: textStyle,
+ );
+ }),
+ ),
+ if (hideOverflowArtist && artists.length > 3)
+ AnchorButton(
+ context.l10n.and_n_more(artists.length - 3),
+ onTap: () {
+ onOverflowArtistClick?.call();
+ },
+ overflow: TextOverflow.ellipsis,
+ style: textStyle.copyWith(
+ color: colorScheme.secondary,
+ decoration: TextDecoration.underline,
+ ),
+ ),
+ ],
+ );
+ }
+}
diff --git a/lib/components/shared/links/hyper_link.dart b/lib/components/links/hyper_link.dart
similarity index 88%
rename from lib/components/shared/links/hyper_link.dart
rename to lib/components/links/hyper_link.dart
index fd31298e..32d715e0 100644
--- a/lib/components/shared/links/hyper_link.dart
+++ b/lib/components/links/hyper_link.dart
@@ -1,5 +1,5 @@
import 'package:flutter/material.dart';
-import 'package:spotube/components/shared/links/anchor_button.dart';
+import 'package:spotube/components/links/anchor_button.dart';
import 'package:url_launcher/url_launcher_string.dart';
class Hyperlink extends StatelessWidget {
@@ -13,12 +13,12 @@ class Hyperlink extends StatelessWidget {
const Hyperlink(
this.text,
this.url, {
- Key? key,
+ super.key,
this.textAlign,
this.overflow,
this.style = const TextStyle(),
this.maxLines,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/links/link_text.dart b/lib/components/links/link_text.dart
similarity index 89%
rename from lib/components/shared/links/link_text.dart
rename to lib/components/links/link_text.dart
index d7b00b72..0cab71d0 100644
--- a/lib/components/shared/links/link_text.dart
+++ b/lib/components/links/link_text.dart
@@ -1,5 +1,5 @@
import 'package:flutter/material.dart';
-import 'package:spotube/components/shared/links/anchor_button.dart';
+import 'package:spotube/components/links/anchor_button.dart';
import 'package:spotube/utils/service_utils.dart';
class LinkText extends StatelessWidget {
@@ -15,14 +15,14 @@ class LinkText extends StatelessWidget {
const LinkText(
this.text,
this.route, {
- Key? key,
+ super.key,
this.textAlign,
this.extra,
this.overflow,
this.style = const TextStyle(),
this.maxLines,
this.push = false,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/panels/controller.dart b/lib/components/panels/controller.dart
similarity index 92%
rename from lib/components/shared/panels/controller.dart
rename to lib/components/panels/controller.dart
index a573c06c..4e367701 100644
--- a/lib/components/shared/panels/controller.dart
+++ b/lib/components/panels/controller.dart
@@ -1,4 +1,4 @@
-part of panels;
+part of 'sliding_up_panel.dart';
class PanelController extends ChangeNotifier {
SlidingUpPanelState? _panelState;
@@ -41,29 +41,33 @@ class PanelController extends ChangeNotifier {
bool get isAttached => _panelState != null;
/// Closes the sliding panel to its collapsed state (i.e. to the minHeight)
- Future close() {
+ Future close() async {
assert(isAttached, "PanelController must be attached to a SlidingUpPanel");
- return _panelState!._close();
+ await _panelState!._close();
+ notifyListeners();
}
/// Opens the sliding panel fully
/// (i.e. to the maxHeight)
- Future open() {
+ Future open() async {
assert(isAttached, "PanelController must be attached to a SlidingUpPanel");
- return _panelState!._open();
+ await _panelState!._open();
+ notifyListeners();
}
/// Hides the sliding panel (i.e. is invisible)
- Future hide() {
+ Future hide() async {
assert(isAttached, "PanelController must be attached to a SlidingUpPanel");
- return _panelState!._hide();
+ await _panelState!._hide();
+ notifyListeners();
}
/// Shows the sliding panel in its collapsed state
/// (i.e. "un-hide" the sliding panel)
- Future show() {
+ Future show() async {
assert(isAttached, "PanelController must be attached to a SlidingUpPanel");
- return _panelState!._show();
+ await _panelState!._show();
+ notifyListeners();
}
/// Animates the panel position to the value.
diff --git a/lib/components/shared/panels/helpers.dart b/lib/components/panels/helpers.dart
similarity index 95%
rename from lib/components/shared/panels/helpers.dart
rename to lib/components/panels/helpers.dart
index 2e754bdf..d79fa97c 100644
--- a/lib/components/shared/panels/helpers.dart
+++ b/lib/components/panels/helpers.dart
@@ -1,4 +1,4 @@
-part of panels;
+part of "sliding_up_panel.dart";
/// if you want to prevent the panel from being dragged using the widget,
/// wrap the widget with this
@@ -47,8 +47,7 @@ class ForceDraggableWidgetRenderBox extends RenderPointerListener {
/// To make [ForceDraggableWidget] work in [Scrollable] widgets
class PanelScrollPhysics extends ScrollPhysics {
final PanelController controller;
- const PanelScrollPhysics({required this.controller, ScrollPhysics? parent})
- : super(parent: parent);
+ const PanelScrollPhysics({required this.controller, super.parent});
@override
PanelScrollPhysics applyTo(ScrollPhysics? ancestor) {
return PanelScrollPhysics(
diff --git a/lib/components/shared/panels/sliding_up_panel.dart b/lib/components/panels/sliding_up_panel.dart
similarity index 99%
rename from lib/components/shared/panels/sliding_up_panel.dart
rename to lib/components/panels/sliding_up_panel.dart
index 137d5eb7..e99fe261 100644
--- a/lib/components/shared/panels/sliding_up_panel.dart
+++ b/lib/components/panels/sliding_up_panel.dart
@@ -146,7 +146,7 @@ class SlidingUpPanel extends StatefulWidget {
final BoxDecoration? panelDecoration;
const SlidingUpPanel(
- {Key? key,
+ {super.key,
this.body,
this.collapsed,
this.minHeight = 100.0,
@@ -176,8 +176,7 @@ class SlidingUpPanel extends StatefulWidget {
this.panelBuilder})
: assert(panelBuilder != null),
assert(0 <= backdropOpacity && backdropOpacity <= 1.0),
- assert(snapPoint == null || 0 < snapPoint && snapPoint < 1.0),
- super(key: key);
+ assert(snapPoint == null || 0 < snapPoint && snapPoint < 1.0);
@override
SlidingUpPanelState createState() => SlidingUpPanelState();
diff --git a/lib/components/shared/playbutton_card.dart b/lib/components/playbutton_card.dart
similarity index 91%
rename from lib/components/shared/playbutton_card.dart
rename to lib/components/playbutton_card.dart
index a8a75d30..ae9050d8 100644
--- a/lib/components/shared/playbutton_card.dart
+++ b/lib/components/playbutton_card.dart
@@ -3,23 +3,15 @@ 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/shared/hover_builder.dart';
-import 'package:spotube/components/shared/image/universal_image.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';
-final htmlTagRegexp = RegExp(r"<[^>]*>", caseSensitive: true);
-
-String? useDescription(String? description) {
- return useMemoized(() {
- if (description == null) return null;
- return description.replaceAll(htmlTagRegexp, '');
- }, [description]);
-}
-
class PlaybuttonCard extends HookWidget {
final void Function()? onTap;
final void Function()? onPlaybuttonPressed;
@@ -43,8 +35,8 @@ class PlaybuttonCard extends HookWidget {
this.onAddToQueuePressed,
this.onTap,
this.isOwner = false,
- Key? key,
- }) : super(key: key);
+ super.key,
+ });
@override
Widget build(BuildContext context) {
@@ -66,19 +58,18 @@ class PlaybuttonCard extends HookWidget {
others: 15,
);
- final cleanDescription = useDescription(description);
-
+ final unescapeHtml = description?.unescapeHtml().cleanHtml();
return Container(
constraints: BoxConstraints(maxWidth: size),
margin: margin,
child: Material(
color: Color.lerp(
- theme.colorScheme.surfaceVariant,
+ theme.colorScheme.surfaceContainerHighest,
theme.colorScheme.surface,
useBrightnessValue(.9, .7),
),
borderRadius: radius,
- shadowColor: theme.colorScheme.background,
+ shadowColor: theme.colorScheme.surface,
elevation: 3,
child: InkWell(
mouseCursor: SystemMouseCursors.click,
@@ -137,7 +128,7 @@ class PlaybuttonCard extends HookWidget {
),
if (isHovered)
Text(
- "Owned by you",
+ context.l10n.owned_by_you,
style: theme.textTheme.bodySmall?.copyWith(
color: Colors.white,
),
@@ -158,7 +149,7 @@ class PlaybuttonCard extends HookWidget {
Skeleton.keep(
child: IconButton(
style: IconButton.styleFrom(
- backgroundColor: theme.colorScheme.background,
+ backgroundColor: theme.colorScheme.surface,
foregroundColor: theme.colorScheme.primary,
minimumSize: const Size.square(10),
),
@@ -205,11 +196,11 @@ class PlaybuttonCard extends HookWidget {
overflow: TextOverflow.ellipsis,
),
),
- if (cleanDescription != null)
+ if (description != null)
Padding(
padding: const EdgeInsets.symmetric(horizontal: 12.0),
child: AutoSizeText(
- cleanDescription,
+ unescapeHtml!,
maxLines: 2,
style: theme.textTheme.bodySmall?.copyWith(
color: theme.colorScheme.onSurface.withOpacity(.5),
diff --git a/lib/components/player/player_queue.dart b/lib/components/player/player_queue.dart
deleted file mode 100644
index 2784fb5f..00000000
--- a/lib/components/player/player_queue.dart
+++ /dev/null
@@ -1,283 +0,0 @@
-import 'dart:ui';
-
-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:hooks_riverpod/hooks_riverpod.dart';
-
-import 'package:scroll_to_index/scroll_to_index.dart';
-import 'package:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/shared/fallbacks/not_found.dart';
-import 'package:spotube/components/shared/inter_scrollbar/inter_scrollbar.dart';
-import 'package:spotube/components/shared/track_tile/track_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/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
-
-class PlayerQueue extends HookConsumerWidget {
- final bool floating;
- const PlayerQueue({
- this.floating = true,
- Key? key,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final playlistNotifier = ref.watch(ProxyPlaylistNotifier.notifier);
- final controller = useAutoScrollController();
- final searchText = 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 mediaQuery = MediaQuery.of(context);
- final headlineColor = theme.textTheme.headlineSmall?.color;
-
- final filteredTracks = useMemoized(
- () {
- if (searchText.value.isEmpty) {
- return tracks;
- }
- return tracks
- .map((e) => (
- weightedRatio(
- '${e.name!} - ${TypeConversionUtils.artists_X_String(e.artists!)}',
- searchText.value,
- ),
- e
- ))
- .sorted((a, b) => b.$1.compareTo(a.$1))
- .where((e) => e.$1 > 50)
- .map((e) => e.$2)
- .toList();
- },
- [tracks, searchText.value],
- );
-
- useEffect(() {
- if (playlist.active == null) return null;
-
- if (playlist.active! < 0) return;
- controller.scrollToIndex(
- playlist.active!,
- preferPosition: AutoScrollPosition.middle,
- );
- return null;
- }, []);
-
- if (tracks.isEmpty) {
- return const NotFound(vertical: true);
- }
-
- 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,
- ),
- decoration: BoxDecoration(
- color: theme.colorScheme.surfaceVariant.withOpacity(0.5),
- borderRadius: borderRadius,
- ),
- child: CallbackShortcuts(
- bindings: {
- LogicalKeySet(LogicalKeyboardKey.escape): () {
- if (!isSearching.value) {
- Navigator.of(context).pop();
- }
- isSearching.value = false;
- searchText.value = '';
- }
- },
- child: Column(
- children: [
- if (!floating)
- Container(
- height: 5,
- width: 100,
- margin: const EdgeInsets.only(bottom: 5, top: 2),
- decoration: BoxDecoration(
- color: headlineColor,
- borderRadius: BorderRadius.circular(20),
- ),
- ),
- Row(
- crossAxisAlignment: CrossAxisAlignment.center,
- mainAxisAlignment: MainAxisAlignment.center,
- children: [
- if (mediaQuery.mdAndUp || !isSearching.value) ...[
- const SizedBox(width: 10),
- Text(
- context.l10n.tracks_in_queue(tracks.length),
- style: TextStyle(
- color: headlineColor,
- fontWeight: FontWeight.bold,
- fontSize: 18,
- ),
- ),
- const Spacer(),
- ],
- 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
- 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),
- ],
- ),
- onPressed: () {
- playlistNotifier.stop();
- Navigator.of(context).pop();
- },
- ),
- const SizedBox(width: 10),
- ],
- ],
- ),
- const SizedBox(height: 10),
- if (!isSearching.value && searchText.value.isEmpty)
- Flexible(
- child: ReorderableListView.builder(
- onReorder: (oldIndex, newIndex) {
- playlistNotifier.moveTrack(oldIndex, newIndex);
- },
- scrollController: controller,
- itemCount: tracks.length,
- shrinkWrap: true,
- buildDefaultDragHandles: false,
- onReorderStart: (index) {
- HapticFeedback.selectionClick();
- },
- onReorderEnd: (index) {
- HapticFeedback.selectionClick();
- },
- itemBuilder: (context, i) {
- final track = tracks.elementAt(i);
- return AutoScrollTag(
- key: ValueKey(i),
- controller: controller,
- index: i,
- child: Padding(
- padding:
- const EdgeInsets.symmetric(horizontal: 8.0),
- child: TrackTile(
- index: i,
- track: track,
- onTap: () async {
- if (playlist.activeTrack?.id == track.id) {
- return;
- }
- await playlistNotifier.jumpToTrack(track);
- },
- leadingActions: [
- ReorderableDragStartListener(
- index: i,
- child: const Icon(SpotubeIcons.dragHandle),
- ),
- ],
- ),
- ),
- );
- },
- ),
- )
- else
- Flexible(
- child: InterScrollbar(
- controller: controller,
- child: ListView.builder(
- controller: controller,
- itemCount: filteredTracks.length,
- itemBuilder: (context, i) {
- final track = filteredTracks.elementAt(i);
- return Padding(
- padding:
- const EdgeInsets.symmetric(horizontal: 8.0),
- child: TrackTile(
- index: i,
- track: track,
- onTap: () async {
- if (playlist.activeTrack?.id == track.id) {
- return;
- }
- await playlistNotifier.jumpToTrack(track);
- },
- ),
- );
- },
- ),
- ),
- ),
- ],
- ),
- ),
- ),
- ),
- );
- }
-}
diff --git a/lib/components/playlist/playlist_card.dart b/lib/components/playlist/playlist_card.dart
deleted file mode 100644
index f429a0ab..00000000
--- a/lib/components/playlist/playlist_card.dart
+++ /dev/null
@@ -1,135 +0,0 @@
-import 'package:fl_query/fl_query.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotify/spotify.dart';
-import 'package:spotube/components/shared/playbutton_card.dart';
-import 'package:spotube/extensions/infinite_query.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/provider/spotify_provider.dart';
-import 'package:spotube/services/audio_player/audio_player.dart';
-import 'package:spotube/services/queries/queries.dart';
-import 'package:spotube/utils/service_utils.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
-
-class PlaylistCard extends HookConsumerWidget {
- final PlaylistSimple playlist;
- const PlaylistCard(
- this.playlist, {
- Key? key,
- }) : super(key: key);
- @override
- Widget build(BuildContext context, ref) {
- final playlistQueue = ref.watch(ProxyPlaylistNotifier.provider);
- final playlistNotifier = ref.watch(ProxyPlaylistNotifier.notifier);
- final playing =
- useStream(audioPlayer.playingStream).data ?? audioPlayer.isPlaying;
- final queryClient = QueryClient.of(context);
- final tracks = useState?>(null);
- bool isPlaylistPlaying = useMemoized(
- () => playlistQueue.containsCollection(playlist.id!),
- [playlistQueue, playlist.id],
- );
-
- final updating = useState(false);
- final spotify = ref.watch(spotifyProvider);
- final me = useQueries.user.me(ref);
-
- Future> fetchAllTracks() async {
- if (playlist.id == 'user-liked-tracks') {
- return await queryClient.fetchQuery(
- "user-liked-tracks",
- () => useQueries.playlist.likedTracks(spotify),
- ) ??
- [];
- }
-
- final query = queryClient.createInfiniteQuery, dynamic, int>(
- "playlist-tracks/${playlist.id}",
- (page) => useQueries.playlist.tracksOf(page, spotify, playlist.id!),
- initialPage: 0,
- nextPage: useQueries.playlist.tracksOfQueryNextPage,
- );
-
- return await query.fetchAllTracks(
- getAllTracks: () async {
- final res =
- await spotify.playlists.getTracksByPlaylistId(playlist.id!).all();
- return res.toList();
- },
- );
- }
-
- return PlaybuttonCard(
- margin: const EdgeInsets.symmetric(horizontal: 10),
- title: playlist.name!,
- description: playlist.description,
- imageUrl: TypeConversionUtils.image_X_UrlString(
- playlist.images,
- placeholder: ImagePlaceholder.collection,
- ),
- isPlaying: isPlaylistPlaying,
- isLoading:
- (isPlaylistPlaying && playlistQueue.isFetching) || updating.value,
- isOwner: playlist.owner?.id == me.data?.id && me.data?.id != null,
- onTap: () {
- ServiceUtils.push(
- context,
- "/playlist/${playlist.id}",
- extra: playlist,
- );
- },
- onPlaybuttonPressed: () async {
- try {
- updating.value = true;
- if (isPlaylistPlaying && playing) {
- return audioPlayer.pause();
- } else if (isPlaylistPlaying && !playing) {
- return audioPlayer.resume();
- }
-
- List fetchedTracks = await fetchAllTracks();
-
- if (fetchedTracks.isEmpty) return;
-
- await playlistNotifier.load(fetchedTracks, autoPlay: true);
- playlistNotifier.addCollection(playlist.id!);
- tracks.value = fetchedTracks;
- } finally {
- if (context.mounted) {
- updating.value = false;
- }
- }
- },
- onAddToQueuePressed: () async {
- updating.value = true;
- try {
- if (isPlaylistPlaying) return;
-
- final fetchedTracks = await fetchAllTracks();
-
- if (fetchedTracks.isEmpty) return;
-
- playlistNotifier.addTracks(fetchedTracks);
- playlistNotifier.addCollection(playlist.id!);
- tracks.value = fetchedTracks;
- if (context.mounted) {
- final snackbar = SnackBar(
- content: Text("Added ${tracks.value?.length} tracks to queue"),
- action: SnackBarAction(
- label: "Undo",
- onPressed: () {
- playlistNotifier
- .removeTracks(fetchedTracks.map((e) => e.id!));
- },
- ),
- );
- ScaffoldMessenger.maybeOf(context)?.showSnackBar(snackbar);
- }
- } finally {
- updating.value = false;
- }
- },
- );
- }
-}
diff --git a/lib/components/root/bottom_player.dart b/lib/components/root/bottom_player.dart
deleted file mode 100644
index 617e760b..00000000
--- a/lib/components/root/bottom_player.dart
+++ /dev/null
@@ -1,136 +0,0 @@
-import 'dart:ui';
-
-import 'package:flutter_desktop_tools/flutter_desktop_tools.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/assets.gen.dart';
-import 'package:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/player/player_actions.dart';
-import 'package:spotube/components/player/player_overlay.dart';
-import 'package:spotube/components/player/player_track_details.dart';
-import 'package:spotube/components/player/player_controls.dart';
-import 'package:spotube/components/player/volume_slider.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/logger.dart';
-import 'package:flutter/material.dart';
-import 'package:spotube/provider/authentication_provider.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_state.dart';
-import 'package:spotube/utils/platform.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
-
-class BottomPlayer extends HookConsumerWidget {
- BottomPlayer({Key? key}) : super(key: key);
-
- final logger = getLogger(BottomPlayer);
- @override
- Widget build(BuildContext context, ref) {
- final auth = ref.watch(AuthenticationNotifier.provider);
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final layoutMode =
- ref.watch(userPreferencesProvider.select((s) => s.layoutMode));
-
- final mediaQuery = MediaQuery.of(context);
-
- String albumArt = useMemoized(
- () => playlist.activeTrack?.album?.images?.isNotEmpty == true
- ? TypeConversionUtils.image_X_UrlString(
- playlist.activeTrack?.album?.images,
- 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.surfaceVariant;
-
- final bgColor = useBrightnessValue(
- Color.lerp(bg, Colors.white, 0.7),
- Color.lerp(bg, Colors.black, 0.45)!,
- );
-
- // returning an empty non spacious Container as the overlay will take
- // place in the global overlay stack aka [_entries]
- if (layoutMode == LayoutMode.compact ||
- ((mediaQuery.mdAndDown) && layoutMode == LayoutMode.adaptive)) {
- 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(albumArt: albumArt)),
- // controls
- Flexible(
- flex: 3,
- child: Padding(
- padding: const EdgeInsets.only(top: 5),
- child: PlayerControls(),
- ),
- ),
- // add to saved tracks
- Column(
- children: [
- PlayerActions(
- extraActions: [
- if (auth != null)
- IconButton(
- tooltip: context.l10n.mini_player,
- icon: const Icon(SpotubeIcons.miniPlayer),
- onPressed: () async {
- final prevSize =
- await DesktopTools.window.getSize();
- await DesktopTools.window.setMinimumSize(
- const Size(300, 300),
- );
- await DesktopTools.window.setAlwaysOnTop(true);
- if (!kIsLinux) {
- await DesktopTools.window.setHasShadow(false);
- }
- await DesktopTools.window
- .setAlignment(Alignment.topRight);
- await DesktopTools.window
- .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),
- child: const VolumeSlider(),
- )
- ],
- )
- ],
- ),
- ),
- ),
- ),
- );
- }
-}
diff --git a/lib/components/shared/heart_button.dart b/lib/components/shared/heart_button.dart
deleted file mode 100644
index 81ccffdb..00000000
--- a/lib/components/shared/heart_button.dart
+++ /dev/null
@@ -1,257 +0,0 @@
-import 'package:fl_query/fl_query.dart';
-import 'package:fl_query_hooks/fl_query_hooks.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-
-import 'package:spotify/spotify.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/provider/authentication_provider.dart';
-import 'package:spotube/provider/scrobbler_provider.dart';
-import 'package:spotube/services/mutations/mutations.dart';
-import 'package:spotube/services/queries/queries.dart';
-
-class HeartButton extends HookConsumerWidget {
- final bool isLiked;
- final void Function()? onPressed;
- final IconData? icon;
- final Color? color;
- final String? tooltip;
- const HeartButton({
- required this.isLiked,
- required this.onPressed,
- this.color,
- this.tooltip,
- this.icon,
- Key? key,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final auth = ref.watch(AuthenticationNotifier.provider);
-
- if (auth == null) 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),
- ),
- ),
- onPressed: onPressed,
- );
- }
-}
-
-typedef UseTrackToggleLike = ({
- bool isLiked,
- Mutation toggleTrackLike,
- Query me,
-});
-
-UseTrackToggleLike useTrackToggleLike(Track track, WidgetRef ref) {
- final me = useQueries.user.me(ref);
-
- final savedTracks = useQueries.playlist.likedTracksQuery(ref);
-
- final isLiked = useMemoized(
- () => savedTracks.data?.any((element) => element.id == track.id) ?? false,
- [savedTracks.data, track.id],
- );
-
- final mounted = useIsMounted();
-
- final scrobblerNotifier = ref.read(scrobblerProvider.notifier);
-
- final toggleTrackLike = useMutations.track.toggleFavorite(
- ref,
- track.id!,
- onMutate: (isLiked) {
- if (isLiked) {
- savedTracks.setData(
- savedTracks.data
- ?.where((element) => element.id != track.id)
- .toList() ??
- [],
- );
- } else {
- savedTracks.setData(
- [
- ...?savedTracks.data,
- track,
- ],
- );
- }
- return isLiked;
- },
- onData: (isLiked, recoveryData) async {
- await savedTracks.refresh();
- if (isLiked) {
- await scrobblerNotifier.love(track);
- } else {
- await scrobblerNotifier.unlove(track);
- }
- },
- onError: (payload, isLiked) {
- if (!mounted()) return;
-
- if (isLiked != true) {
- savedTracks.setData(
- savedTracks.data
- ?.where((element) => element.id != track.id)
- .toList() ??
- [],
- );
- } else {
- savedTracks.setData(
- [
- ...?savedTracks.data,
- track,
- ],
- );
- }
- },
- );
-
- return (isLiked: isLiked, toggleTrackLike: toggleTrackLike, me: me);
-}
-
-class TrackHeartButton extends HookConsumerWidget {
- final Track track;
- const TrackHeartButton({
- Key? key,
- required this.track,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final savedTracks = useQueries.playlist.likedTracksQuery(ref);
- final (:me, :isLiked, :toggleTrackLike) = useTrackToggleLike(track, ref);
-
- if (me.isLoading || !me.hasData) {
- return const CircularProgressIndicator();
- }
-
- return HeartButton(
- tooltip: isLiked
- ? context.l10n.remove_from_favorites
- : context.l10n.save_as_favorite,
- isLiked: isLiked,
- onPressed: savedTracks.hasData
- ? () {
- toggleTrackLike.mutate(isLiked);
- }
- : null,
- );
- }
-}
-
-class PlaylistHeartButton extends HookConsumerWidget {
- final PlaylistSimple playlist;
- final IconData? icon;
- final ValueChanged? onData;
-
- const PlaylistHeartButton({
- required this.playlist,
- Key? key,
- this.icon,
- this.onData,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final me = useQueries.user.me(ref);
-
- final isLikedQuery = useQueries.playlist.doesUserFollow(
- ref,
- playlist.id!,
- me.data?.id ?? '',
- );
-
- final togglePlaylistLike = useMutations.playlist.toggleFavorite(
- ref,
- playlist.id!,
- refreshQueries: [
- isLikedQuery.key,
- ],
- onData: onData,
- );
-
- if (me.isLoading || !me.hasData) {
- return const CircularProgressIndicator();
- }
-
- return HeartButton(
- isLiked: isLikedQuery.data ?? false,
- tooltip: isLikedQuery.data ?? false
- ? context.l10n.remove_from_favorites
- : context.l10n.save_as_favorite,
- color: Colors.white,
- icon: icon,
- onPressed: isLikedQuery.hasData
- ? () {
- togglePlaylistLike.mutate(isLikedQuery.data!);
- }
- : null,
- );
- }
-}
-
-class AlbumHeartButton extends HookConsumerWidget {
- final AlbumSimple album;
-
- const AlbumHeartButton({
- required this.album,
- Key? key,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final client = useQueryClient();
- final me = useQueries.user.me(ref);
-
- final albumIsSaved = useQueries.album.isSavedForMe(ref, album.id!);
- final isLiked = albumIsSaved.data ?? false;
-
- final toggleAlbumLike = useMutations.album.toggleFavorite(
- ref,
- album.id!,
- refreshQueries: [albumIsSaved.key],
- onData: (_, __) async {
- await client.refreshInfiniteQueryAllPages("current-user-albums");
- },
- );
-
- if (me.isLoading || !me.hasData) {
- return const CircularProgressIndicator();
- }
-
- return HeartButton(
- isLiked: isLiked,
- tooltip: isLiked
- ? context.l10n.remove_from_favorites
- : context.l10n.save_as_favorite,
- color: Colors.white,
- onPressed: albumIsSaved.hasData
- ? () {
- toggleAlbumLike.mutate(isLiked);
- }
- : null,
- );
- }
-}
diff --git a/lib/components/shared/page_window_title_bar.dart b/lib/components/shared/page_window_title_bar.dart
deleted file mode 100644
index 9aa2d4a8..00000000
--- a/lib/components/shared/page_window_title_bar.dart
+++ /dev/null
@@ -1,600 +0,0 @@
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
-import 'package:spotube/utils/platform.dart';
-import 'package:titlebar_buttons/titlebar_buttons.dart';
-import 'dart:math';
-import 'package:flutter/foundation.dart' show kIsWeb;
-import 'dart:io' show Platform;
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
-
-class PageWindowTitleBar extends StatefulHookConsumerWidget
- implements PreferredSizeWidget {
- final Widget? leading;
- final bool automaticallyImplyLeading;
- final List? actions;
- 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;
-
- const PageWindowTitleBar({
- Key? key,
- this.actions,
- this.title,
- this.toolbarOpacity = 1,
- this.backgroundColor,
- this.actionsIconTheme,
- this.automaticallyImplyLeading = false,
- this.centerTitle,
- this.foregroundColor,
- this.leading,
- this.leadingWidth,
- this.titleSpacing,
- this.titleTextStyle,
- this.titleWidth,
- this.toolbarTextStyle,
- }) : super(key: key);
-
- @override
- Size get preferredSize => const Size.fromHeight(kToolbarHeight);
-
- @override
- ConsumerState createState() => _PageWindowTitleBarState();
-}
-
-class _PageWindowTitleBarState extends ConsumerState {
- void onDrag(details) {
- final systemTitleBar =
- ref.read(userPreferencesProvider.select((s) => s.systemTitleBar));
- if (kIsDesktop && !systemTitleBar) {
- DesktopTools.window.startDragging();
- }
- }
-
- @override
- Widget build(BuildContext context) {
- final mediaQuery = MediaQuery.of(context);
-
- 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: DesktopTools.platform.isMacOS &&
- 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: widget.title,
- ),
- ),
- );
- });
- }
-}
-
-class WindowTitleBarButtons extends HookConsumerWidget {
- final Color? foregroundColor;
- const WindowTitleBarButtons({
- Key? key,
- this.foregroundColor,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final preferences = ref.watch(userPreferencesProvider);
- final isMaximized = useState(null);
- const type = ThemeType.auto;
-
- Future onClose() async {
- await DesktopTools.window.close();
- }
-
- useEffect(() {
- if (kIsDesktop) {
- DesktopTools.window.isMaximized().then((value) {
- isMaximized.value = value;
- });
- }
- return null;
- }, []);
-
- if (!kIsDesktop || kIsMacOS || preferences.systemTitleBar) {
- return const SizedBox.shrink();
- }
-
- if (kIsWindows) {
- final theme = Theme.of(context);
- final colors = WindowButtonColors(
- normal: Colors.transparent,
- iconNormal: foregroundColor ?? theme.colorScheme.onBackground,
- mouseOver: theme.colorScheme.onBackground.withOpacity(0.1),
- mouseDown: theme.colorScheme.onBackground.withOpacity(0.2),
- iconMouseOver: theme.colorScheme.onBackground,
- iconMouseDown: theme.colorScheme.onBackground,
- );
-
- final closeColors = WindowButtonColors(
- normal: Colors.transparent,
- iconNormal: foregroundColor ?? theme.colorScheme.onBackground,
- 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: DesktopTools.window.minimize,
- colors: colors,
- ),
- if (isMaximized.value != true)
- MaximizeWindowButton(
- colors: colors,
- onPressed: () {
- DesktopTools.window.maximize();
- isMaximized.value = true;
- },
- )
- else
- RestoreWindowButton(
- colors: colors,
- onPressed: () {
- DesktopTools.window.unmaximize();
- isMaximized.value = false;
- },
- ),
- CloseWindowButton(
- colors: closeColors,
- onPressed: onClose,
- ),
- ],
- ),
- );
- }
-
- return Padding(
- padding: const EdgeInsets.only(bottom: 20, left: 10),
- child: Row(
- crossAxisAlignment: CrossAxisAlignment.start,
- children: [
- DecoratedMinimizeButton(
- type: type,
- onPressed: DesktopTools.window.minimize,
- ),
- DecoratedMaximizeButton(
- type: type,
- onPressed: () async {
- if (await DesktopTools.window.isMaximized()) {
- await DesktopTools.window.unmaximize();
- isMaximized.value = false;
- } else {
- await DesktopTools.window.maximize();
- isMaximized.value = true;
- }
- },
- ),
- DecoratedCloseButton(
- type: type,
- onPressed: onClose,
- ),
- ],
- ),
- );
- }
-}
-
-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(
- {Key? key,
- WindowButtonColors? colors,
- this.builder,
- @required this.iconBuilder,
- this.padding,
- this.onPressed,
- this.animate = false})
- : super(key: key) {
- 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!();
- },
- );
- }
-}
-
-class MinimizeWindowButton extends WindowButton {
- MinimizeWindowButton(
- {Key? key,
- WindowButtonColors? colors,
- VoidCallback? onPressed,
- bool? animate})
- : super(
- key: key,
- colors: colors,
- animate: animate ?? false,
- iconBuilder: (buttonContext) =>
- MinimizeIcon(color: buttonContext.iconColor),
- onPressed: onPressed,
- );
-}
-
-class MaximizeWindowButton extends WindowButton {
- MaximizeWindowButton(
- {Key? key,
- WindowButtonColors? colors,
- VoidCallback? onPressed,
- bool? animate})
- : super(
- key: key,
- colors: colors,
- animate: animate ?? false,
- iconBuilder: (buttonContext) =>
- MaximizeIcon(color: buttonContext.iconColor),
- onPressed: onPressed,
- );
-}
-
-class RestoreWindowButton extends WindowButton {
- RestoreWindowButton(
- {Key? key,
- WindowButtonColors? colors,
- VoidCallback? onPressed,
- bool? animate})
- : super(
- key: key,
- colors: colors,
- animate: animate ?? false,
- iconBuilder: (buttonContext) =>
- RestoreIcon(color: buttonContext.iconColor),
- onPressed: onPressed,
- );
-}
-
-final _defaultCloseButtonColors = WindowButtonColors(
- mouseOver: const Color(0xFFD32F2F),
- mouseDown: const Color(0xFFB71C1C),
- iconNormal: const Color(0xFF805306),
- iconMouseOver: const Color(0xFFFFFFFF));
-
-class CloseWindowButton extends WindowButton {
- CloseWindowButton(
- {Key? key,
- WindowButtonColors? colors,
- VoidCallback? onPressed,
- bool? animate})
- : super(
- key: key,
- colors: colors ?? _defaultCloseButtonColors,
- animate: animate ?? false,
- iconBuilder: (buttonContext) =>
- CloseIcon(color: buttonContext.iconColor),
- onPressed: onPressed,
- );
-}
-
-// Switched to CustomPaint icons by https://github.com/esDotDev
-
-/// Close
-class CloseIcon extends StatelessWidget {
- final Color color;
- const CloseIcon({Key? key, required this.color}) : super(key: key);
- @override
- Widget build(BuildContext context) => Align(
- alignment: Alignment.topLeft,
- child: Stack(children: [
- // Use rotated containers instead of a painter because it renders slightly crisper than a painter for some reason.
- Transform.rotate(
- angle: pi * .25,
- child:
- Center(child: Container(width: 14, height: 1, color: color))),
- Transform.rotate(
- angle: pi * -.25,
- child:
- Center(child: Container(width: 14, height: 1, color: color))),
- ]),
- );
-}
-
-/// Maximize
-class MaximizeIcon extends StatelessWidget {
- final Color color;
- const MaximizeIcon({Key? key, required this.color}) : super(key: key);
- @override
- Widget build(BuildContext context) => _AlignedPaint(_MaximizePainter(color));
-}
-
-class _MaximizePainter extends _IconPainter {
- _MaximizePainter(Color color) : super(color);
- @override
- void paint(Canvas canvas, Size size) {
- Paint p = getPaint(color);
- canvas.drawRect(Rect.fromLTRB(0, 0, size.width - 1, size.height - 1), p);
- }
-}
-
-/// Restore
-class RestoreIcon extends StatelessWidget {
- final Color color;
- const RestoreIcon({
- Key? key,
- required this.color,
- }) : super(key: key);
- @override
- Widget build(BuildContext context) => _AlignedPaint(_RestorePainter(color));
-}
-
-class _RestorePainter extends _IconPainter {
- _RestorePainter(Color color) : super(color);
- @override
- void paint(Canvas canvas, Size size) {
- Paint p = getPaint(color);
- canvas.drawRect(Rect.fromLTRB(0, 2, size.width - 2, size.height), p);
- canvas.drawLine(const Offset(2, 2), const Offset(2, 0), p);
- canvas.drawLine(const Offset(2, 0), Offset(size.width, 0), p);
- canvas.drawLine(
- Offset(size.width, 0), Offset(size.width, size.height - 2), p);
- canvas.drawLine(Offset(size.width, size.height - 2),
- Offset(size.width - 2, size.height - 2), p);
- }
-}
-
-/// Minimize
-class MinimizeIcon extends StatelessWidget {
- final Color color;
- const MinimizeIcon({Key? key, required this.color}) : super(key: key);
- @override
- Widget build(BuildContext context) => _AlignedPaint(_MinimizePainter(color));
-}
-
-class _MinimizePainter extends _IconPainter {
- _MinimizePainter(Color color) : super(color);
- @override
- void paint(Canvas canvas, Size size) {
- Paint p = getPaint(color);
- canvas.drawLine(
- Offset(0, size.height / 2), Offset(size.width, size.height / 2), p);
- }
-}
-
-/// Helpers
-abstract class _IconPainter extends CustomPainter {
- _IconPainter(this.color);
- final Color color;
-
- @override
- bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
-}
-
-class _AlignedPaint extends StatelessWidget {
- const _AlignedPaint(this.painter, {Key? key}) : super(key: key);
- final CustomPainter painter;
-
- @override
- Widget build(BuildContext context) {
- return Align(
- alignment: Alignment.center,
- child: CustomPaint(size: const Size(10, 10), painter: painter));
- }
-}
-
-Paint getPaint(Color color, [bool isAntiAlias = false]) => Paint()
- ..color = color
- ..style = PaintingStyle.stroke
- ..isAntiAlias = isAntiAlias
- ..strokeWidth = 1;
-
-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({Key? key, required this.builder, this.onPressed})
- : super(key: key);
- @override
- _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/shared/track_tile/track_tile.dart b/lib/components/shared/track_tile/track_tile.dart
deleted file mode 100644
index d268c783..00000000
--- a/lib/components/shared/track_tile/track_tile.dart
+++ /dev/null
@@ -1,270 +0,0 @@
-import 'dart:async';
-
-import 'package:flutter/gestures.dart';
-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:spotify/spotify.dart';
-import 'package:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/shared/hover_builder.dart';
-import 'package:spotube/components/shared/image/universal_image.dart';
-import 'package:spotube/components/shared/links/link_text.dart';
-import 'package:spotube/components/shared/track_tile/track_options.dart';
-import 'package:spotube/extensions/constrains.dart';
-import 'package:spotube/extensions/duration.dart';
-import 'package:spotube/models/local_track.dart';
-import 'package:spotube/provider/blacklist_provider.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
-
-class TrackTile extends HookConsumerWidget {
- /// [index] will not be shown if null
- final int? index;
- final Track track;
- final bool selected;
- final ValueChanged? onChanged;
- final Future Function()? onTap;
- final VoidCallback? onLongPress;
- final bool userPlaylist;
- final String? playlistId;
-
- final List? leadingActions;
-
- const TrackTile({
- Key? key,
- this.index,
- required this.track,
- this.selected = false,
- this.onTap,
- this.onLongPress,
- this.onChanged,
- this.userPlaylist = false,
- this.playlistId,
- this.leadingActions,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final theme = Theme.of(context);
-
- final blacklist = ref.watch(BlackListNotifier.provider);
-
- final isBlackListed = useMemoized(
- () => blacklist.contains(
- BlacklistedElement.track(
- track.id!,
- track.name!,
- ),
- ),
- [blacklist, track],
- );
-
- final showOptionCbRef = useRef?>(null);
-
- final isPlaying = track.id == playlist.activeTrack?.id;
-
- final isLoading = useState(false);
-
- final isSelected = isPlaying || isLoading.value;
-
- 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,
- ),
- );
- },
- child: HoverBuilder(
- permanentState: isSelected || constrains.smAndDown ? true : null,
- builder: (context, isHovering) {
- return ListTile(
- selected: isSelected,
- onTap: () async {
- try {
- isLoading.value = true;
- await onTap?.call();
- } finally {
- if (context.mounted) {
- isLoading.value = false;
- }
- }
- },
- onLongPress: onLongPress,
- enabled: !isBlackListed,
- contentPadding: EdgeInsets.zero,
- tileColor:
- isBlackListed ? theme.colorScheme.errorContainer : null,
- horizontalTitleGap: 12,
- leadingAndTrailingTextStyle: theme.textTheme.bodyMedium,
- 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,
- ),
- Stack(
- children: [
- ClipRRect(
- borderRadius: BorderRadius.circular(4),
- child: AspectRatio(
- aspectRatio: 1,
- child: UniversalImage(
- path: TypeConversionUtils.image_X_UrlString(
- track.album?.images,
- placeholder: ImagePlaceholder.albumArt,
- ),
- fit: BoxFit.cover,
- ),
- ),
- ),
- Positioned.fill(
- child: AnimatedContainer(
- duration: const Duration(milliseconds: 300),
- decoration: BoxDecoration(
- borderRadius: BorderRadius.circular(4),
- color: isHovering
- ? Colors.black.withOpacity(0.4)
- : Colors.transparent,
- ),
- ),
- ),
- Positioned.fill(
- child: Center(
- child: IconTheme(
- data: theme.iconTheme
- .copyWith(size: 26, color: Colors.white),
- child: Skeleton.ignore(
- child: AnimatedSwitcher(
- duration: const Duration(milliseconds: 300),
- child: (isPlaying && playlist.isFetching) ||
- 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),
- ),
- ),
- ),
- ),
- ),
- ],
- ),
- ],
- ),
- title: Row(
- children: [
- Expanded(
- flex: 6,
- child: LinkText(
- track.name!,
- "/track/${track.id}",
- push: true,
- maxLines: 1,
- overflow: TextOverflow.ellipsis,
- ),
- ),
- if (constrains.mdAndUp) ...[
- const SizedBox(width: 8),
- Expanded(
- flex: 4,
- child: switch (track.runtimeType) {
- LocalTrack => 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,
- push: true,
- overflow: TextOverflow.ellipsis,
- ),
- )
- },
- ),
- ],
- ],
- ),
- subtitle: Align(
- alignment: Alignment.centerLeft,
- child: track is LocalTrack
- ? Text(
- TypeConversionUtils.artists_X_String(
- track.artists ?? [],
- ),
- )
- : ClipRect(
- child: ConstrainedBox(
- constraints: const BoxConstraints(maxHeight: 40),
- child: TypeConversionUtils.artists_X_ClickableArtists(
- track.artists ?? [],
- ),
- ),
- ),
- ),
- trailing: Row(
- mainAxisSize: MainAxisSize.min,
- children: [
- const SizedBox(width: 8),
- Text(
- Duration(milliseconds: track.durationMs ?? 0)
- .toHumanReadableString(padZero: false),
- maxLines: 1,
- overflow: TextOverflow.ellipsis,
- ),
- TrackOptions(
- track: track,
- playlistId: playlistId,
- userPlaylist: userPlaylist,
- showMenuCbRef: showOptionCbRef,
- ),
- ],
- ),
- );
- },
- ),
- );
- });
- }
-}
diff --git a/lib/components/shared/tracks_view/sections/body/use_is_user_playlist.dart b/lib/components/shared/tracks_view/sections/body/use_is_user_playlist.dart
deleted file mode 100644
index ca3c6706..00000000
--- a/lib/components/shared/tracks_view/sections/body/use_is_user_playlist.dart
+++ /dev/null
@@ -1,18 +0,0 @@
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotube/services/queries/queries.dart';
-
-bool useIsUserPlaylist(WidgetRef ref, String playlistId) {
- final userPlaylistsQuery = useQueries.playlist.ofMineAll(ref);
- final me = useQueries.user.me(ref);
-
- return useMemoized(
- () =>
- userPlaylistsQuery.data?.any((e) =>
- e.id == playlistId &&
- me.data != null &&
- e.owner?.id == me.data?.id) ??
- false,
- [userPlaylistsQuery.data, playlistId, me.data],
- );
-}
diff --git a/lib/components/shared/tracks_view/sections/header/header_buttons.dart b/lib/components/shared/tracks_view/sections/header/header_buttons.dart
deleted file mode 100644
index bae47f12..00000000
--- a/lib/components/shared/tracks_view/sections/header/header_buttons.dart
+++ /dev/null
@@ -1,137 +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:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.dart';
-import 'package:spotube/extensions/context.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/services/audio_player/audio_player.dart';
-
-class TrackViewHeaderButtons extends HookConsumerWidget {
- final PaletteColor color;
- final bool compact;
- const TrackViewHeaderButtons({
- Key? key,
- required this.color,
- this.compact = false,
- }) : super(key: key);
-
- @override
- Widget build(BuildContext context, ref) {
- final props = InheritedTrackView.of(context);
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final playlistNotifier = ref.watch(ProxyPlaylistNotifier.notifier);
-
- 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 allTracks = await props.pagination.onFetchAll();
-
- await playlistNotifier.load(
- allTracks,
- autoPlay: true,
- initialIndex: Random().nextInt(allTracks.length),
- );
- await audioPlayer.setShuffle(true);
- playlistNotifier.addCollection(props.collectionId);
- } finally {
- isLoading.value = false;
- }
- }
-
- void onPlay() async {
- try {
- isLoading.value = true;
-
- final allTracks = await props.pagination.onFetchAll();
-
- await playlistNotifier.load(allTracks, autoPlay: true);
- playlistNotifier.addCollection(props.collectionId);
- } finally {
- 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/shared/shimmers/shimmer_lyrics.dart b/lib/components/shimmers/shimmer_lyrics.dart
similarity index 94%
rename from lib/components/shared/shimmers/shimmer_lyrics.dart
rename to lib/components/shimmers/shimmer_lyrics.dart
index b225c008..03816202 100644
--- a/lib/components/shared/shimmers/shimmer_lyrics.dart
+++ b/lib/components/shimmers/shimmer_lyrics.dart
@@ -5,7 +5,7 @@ import 'package:gap/gap.dart';
import 'package:skeletonizer/skeletonizer.dart';
class ShimmerLyrics extends HookWidget {
- const ShimmerLyrics({Key? key}) : super(key: key);
+ const ShimmerLyrics({super.key});
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/sort_tracks_dropdown.dart b/lib/components/sort_tracks_dropdown.dart
similarity index 93%
rename from lib/components/shared/sort_tracks_dropdown.dart
rename to lib/components/sort_tracks_dropdown.dart
index ab35b2e3..16727013 100644
--- a/lib/components/shared/sort_tracks_dropdown.dart
+++ b/lib/components/sort_tracks_dropdown.dart
@@ -1,8 +1,8 @@
import 'package:flutter/material.dart';
import 'package:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/library/user_local_tracks.dart';
-import 'package:spotube/components/shared/adaptive/adaptive_pop_sheet_list.dart';
+import 'package:spotube/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 {
@@ -11,8 +11,8 @@ class SortTracksDropdown extends StatelessWidget {
const SortTracksDropdown({
this.onChanged,
this.value,
- Key? key,
- }) : super(key: key);
+ super.key,
+ });
@override
Widget build(BuildContext context) {
diff --git a/lib/components/shared/spotube_page_route.dart b/lib/components/spotube_page_route.dart
similarity index 100%
rename from lib/components/shared/spotube_page_route.dart
rename to lib/components/spotube_page_route.dart
diff --git a/lib/components/shared/themed_button_tab_bar.dart b/lib/components/themed_button_tab_bar.dart
similarity index 87%
rename from lib/components/shared/themed_button_tab_bar.dart
rename to lib/components/themed_button_tab_bar.dart
index d5798189..c245e5f4 100644
--- a/lib/components/shared/themed_button_tab_bar.dart
+++ b/lib/components/themed_button_tab_bar.dart
@@ -5,7 +5,8 @@ import 'package:spotube/hooks/utils/use_brightness_value.dart';
class ThemedButtonsTabBar extends HookWidget implements PreferredSizeWidget {
final List tabs;
- const ThemedButtonsTabBar({Key? key, required this.tabs}) : super(key: key);
+ final TabController? controller;
+ const ThemedButtonsTabBar({super.key, required this.tabs, this.controller});
@override
Widget build(BuildContext context) {
@@ -21,6 +22,7 @@ class ThemedButtonsTabBar extends HookWidget implements PreferredSizeWidget {
bottom: 8,
),
child: ButtonsTabBar(
+ controller: controller,
radius: 100,
decoration: BoxDecoration(
color: bgColor,
@@ -32,7 +34,7 @@ class ThemedButtonsTabBar extends HookWidget implements PreferredSizeWidget {
),
borderWidth: 0,
unselectedDecoration: BoxDecoration(
- color: theme.colorScheme.background,
+ color: theme.colorScheme.surface,
borderRadius: BorderRadius.circular(15),
),
unselectedLabelStyle: theme.textTheme.labelLarge?.copyWith(
diff --git a/lib/components/titlebar/mouse_state.dart b/lib/components/titlebar/mouse_state.dart
new file mode 100644
index 00000000..9af2a8b0
--- /dev/null
+++ b/lib/components/titlebar/mouse_state.dart
@@ -0,0 +1,73 @@
+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
new file mode 100644
index 00000000..76a5ec8a
--- /dev/null
+++ b/lib/components/titlebar/titlebar.dart
@@ -0,0 +1,179 @@
+import 'package:flutter/material.dart';
+import 'package:hooks_riverpod/hooks_riverpod.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 bool automaticallyImplyLeading;
+ final List? actions;
+ 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 bool _sliver;
+
+ const PageWindowTitleBar({
+ super.key,
+ this.actions,
+ this.title,
+ this.toolbarOpacity = 1,
+ 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;
+
+ 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) {
+ final systemTitleBar =
+ ref.read(userPreferencesProvider.select((s) => s.systemTitleBar));
+ if (kIsDesktop && !systemTitleBar) {
+ windowManager.startDragging();
+ }
+ }
+
+ @override
+ Widget build(BuildContext context) {
+ final mediaQuery = MediaQuery.of(context);
+
+ if (widget._sliver) {
+ return SliverLayoutBuilder(
+ builder: (context, constraints) {
+ final hasFullscreen =
+ mediaQuery.size.width == constraints.crossAxisExtent;
+ final hasLeadingOrCanPop =
+ widget.leading != null || Navigator.canPop(context);
+
+ 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),
+ ],
+ 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,
+ ),
+ );
+ },
+ );
+ }
+
+ 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,
+ ),
+ ),
+ );
+ });
+ }
+}
diff --git a/lib/components/titlebar/titlebar_buttons.dart b/lib/components/titlebar/titlebar_buttons.dart
new file mode 100644
index 00000000..35cdf08e
--- /dev/null
+++ b/lib/components/titlebar/titlebar_buttons.dart
@@ -0,0 +1,124 @@
+import 'package:flutter/material.dart';
+import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:spotube/components/titlebar/titlebar_icon_buttons.dart';
+import 'package:spotube/components/titlebar/window_button.dart';
+import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
+import 'package:spotube/utils/platform.dart';
+import 'package:titlebar_buttons/titlebar_buttons.dart';
+import 'package:window_manager/window_manager.dart';
+
+class WindowTitleBarButtons extends HookConsumerWidget {
+ final Color? foregroundColor;
+ const WindowTitleBarButtons({
+ super.key,
+ this.foregroundColor,
+ });
+
+ @override
+ Widget build(BuildContext context, ref) {
+ final preferences = ref.watch(userPreferencesProvider);
+ final isMaximized = useState(null);
+ const type = ThemeType.auto;
+
+ Future onClose() async {
+ await windowManager.close();
+ }
+
+ useEffect(() {
+ if (kIsDesktop) {
+ windowManager.isMaximized().then((value) {
+ isMaximized.value = value;
+ });
+ }
+ return null;
+ }, []);
+
+ if (!kIsDesktop || kIsMacOS || 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,
+ ),
+ if (isMaximized.value != true)
+ MaximizeWindowButton(
+ colors: colors,
+ onPressed: () {
+ windowManager.maximize();
+ isMaximized.value = true;
+ },
+ )
+ else
+ RestoreWindowButton(
+ colors: colors,
+ onPressed: () {
+ windowManager.unmaximize();
+ isMaximized.value = false;
+ },
+ ),
+ CloseWindowButton(
+ colors: closeColors,
+ onPressed: onClose,
+ ),
+ ],
+ ),
+ );
+ }
+
+ 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,
+ ),
+ ],
+ ),
+ );
+ }
+}
diff --git a/lib/components/titlebar/titlebar_icon_buttons.dart b/lib/components/titlebar/titlebar_icon_buttons.dart
new file mode 100644
index 00000000..70170262
--- /dev/null
+++ b/lib/components/titlebar/titlebar_icon_buttons.dart
@@ -0,0 +1,161 @@
+import 'dart:math';
+
+import 'package:flutter/material.dart';
+import 'package:spotube/components/titlebar/window_button.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 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;
+ const CloseIcon({super.key, required this.color});
+ @override
+ Widget build(BuildContext context) => Align(
+ alignment: Alignment.topLeft,
+ child: Stack(children: [
+ // Use rotated containers instead of a painter because it renders slightly crisper than a painter for some reason.
+ Transform.rotate(
+ angle: pi * .25,
+ child:
+ Center(child: Container(width: 14, height: 1, color: color))),
+ Transform.rotate(
+ angle: pi * -.25,
+ child:
+ Center(child: Container(width: 14, height: 1, color: color))),
+ ]),
+ );
+}
+
+/// Maximize
+class MaximizeIcon extends StatelessWidget {
+ final Color color;
+ const MaximizeIcon({super.key, required this.color});
+ @override
+ Widget build(BuildContext context) => _AlignedPaint(_MaximizePainter(color));
+}
+
+class _MaximizePainter extends _IconPainter {
+ _MaximizePainter(super.color);
+ @override
+ void paint(Canvas canvas, Size size) {
+ Paint p = getPaint(color);
+ canvas.drawRect(Rect.fromLTRB(0, 0, size.width - 1, size.height - 1), p);
+ }
+}
+
+/// Restore
+class RestoreIcon extends StatelessWidget {
+ final Color color;
+ const RestoreIcon({
+ super.key,
+ required this.color,
+ });
+ @override
+ Widget build(BuildContext context) => _AlignedPaint(_RestorePainter(color));
+}
+
+class _RestorePainter extends _IconPainter {
+ _RestorePainter(super.color);
+ @override
+ void paint(Canvas canvas, Size size) {
+ Paint p = getPaint(color);
+ canvas.drawRect(Rect.fromLTRB(0, 2, size.width - 2, size.height), p);
+ canvas.drawLine(const Offset(2, 2), const Offset(2, 0), p);
+ canvas.drawLine(const Offset(2, 0), Offset(size.width, 0), p);
+ canvas.drawLine(
+ Offset(size.width, 0), Offset(size.width, size.height - 2), p);
+ canvas.drawLine(Offset(size.width, size.height - 2),
+ Offset(size.width - 2, size.height - 2), p);
+ }
+}
+
+/// Minimize
+class MinimizeIcon extends StatelessWidget {
+ final Color color;
+ const MinimizeIcon({super.key, required this.color});
+ @override
+ Widget build(BuildContext context) => _AlignedPaint(_MinimizePainter(color));
+}
+
+class _MinimizePainter extends _IconPainter {
+ _MinimizePainter(super.color);
+ @override
+ void paint(Canvas canvas, Size size) {
+ Paint p = getPaint(color);
+ canvas.drawLine(
+ Offset(0, size.height / 2), Offset(size.width, size.height / 2), p);
+ }
+}
+
+/// Helpers
+abstract class _IconPainter extends CustomPainter {
+ _IconPainter(this.color);
+ final Color color;
+
+ @override
+ bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
+}
+
+class _AlignedPaint extends StatelessWidget {
+ const _AlignedPaint(this.painter);
+ final CustomPainter painter;
+
+ @override
+ Widget build(BuildContext context) {
+ return Align(
+ alignment: Alignment.center,
+ child: CustomPaint(size: const Size(10, 10), painter: painter));
+ }
+}
+
+Paint getPaint(Color color, [bool isAntiAlias = false]) => Paint()
+ ..color = color
+ ..style = PaintingStyle.stroke
+ ..isAntiAlias = isAntiAlias
+ ..strokeWidth = 1;
diff --git a/lib/components/titlebar/window_button.dart b/lib/components/titlebar/window_button.dart
new file mode 100644
index 00000000..3201d191
--- /dev/null
+++ b/lib/components/titlebar/window_button.dart
@@ -0,0 +1,133 @@
+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/shared/track_tile/track_options.dart b/lib/components/track_tile/track_options.dart
similarity index 54%
rename from lib/components/shared/track_tile/track_options.dart
rename to lib/components/track_tile/track_options.dart
index a094259d..d2cb92cf 100644
--- a/lib/components/shared/track_tile/track_options.dart
+++ b/lib/components/track_tile/track_options.dart
@@ -1,6 +1,5 @@
import 'dart:io';
-import 'package:fl_query/fl_query.dart';
import 'package:flutter/material.dart' hide Page;
import 'package:flutter/services.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
@@ -9,24 +8,28 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotify/spotify.dart';
import 'package:spotube/collections/assets.gen.dart';
import 'package:spotube/collections/spotube_icons.dart';
-import 'package:spotube/components/library/user_local_tracks.dart';
-import 'package:spotube/components/shared/adaptive/adaptive_pop_sheet_list.dart';
-import 'package:spotube/components/shared/dialogs/playlist_add_track_dialog.dart';
-import 'package:spotube/components/shared/dialogs/prompt_dialog.dart';
-import 'package:spotube/components/shared/dialogs/track_details_dialog.dart';
-import 'package:spotube/components/shared/heart_button.dart';
-import 'package:spotube/components/shared/image/universal_image.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/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/provider/authentication_provider.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/proxy_playlist/proxy_playlist_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/services/mutations/mutations.dart';
-import 'package:spotube/services/queries/search.dart';
-import 'package:spotube/utils/type_conversion_utils.dart';
+import 'package:spotube/utils/service_utils.dart';
+
import 'package:url_launcher/url_launcher_string.dart';
enum TrackOptionValue {
@@ -53,13 +56,13 @@ class TrackOptions extends HookConsumerWidget {
final ObjectRef?>? showMenuCbRef;
final Widget? icon;
const TrackOptions({
- Key? key,
+ super.key,
required this.track,
this.showMenuCbRef,
this.userPlaylist = false,
this.playlistId,
this.icon,
- }) : super(key: key);
+ });
void actionShare(BuildContext context, Track track) {
final data = "https://open.spotify.com/track/${track.id}";
@@ -95,25 +98,16 @@ class TrackOptions extends HookConsumerWidget {
WidgetRef ref,
Track track,
) async {
- final playback = ref.read(ProxyPlaylistNotifier.notifier);
- final playlist = ref.read(ProxyPlaylistNotifier.provider);
+ final playback = ref.read(audioPlayerProvider.notifier);
+ final playlist = ref.read(audioPlayerProvider);
final spotify = ref.read(spotifyProvider);
final query = "${track.name} Radio";
- final pages = await QueryClient.of(context)
- .fetchInfiniteQueryJob, dynamic, int, SearchParams>(
- job: SearchQueries.queryJob(query),
- args: (
- spotify: spotify,
- searchType: SearchType.playlist,
- query: query,
- ),
- ) ??
- [];
+ final pages =
+ await spotify.search.get(query, types: [SearchType.playlist]).first();
final radios = pages
- .expand((e) => e.items?.toList() ?? [])
- .toList()
- .cast();
+ .expand((e) => e.items?.cast().toList() ?? [])
+ .toList();
final artists = track.artists!.map((e) => e.name);
@@ -170,30 +164,26 @@ class TrackOptions extends HookConsumerWidget {
final router = GoRouter.of(context);
final ThemeData(:colorScheme) = Theme.of(context);
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final playback = ref.watch(ProxyPlaylistNotifier.notifier);
- final auth = ref.watch(AuthenticationNotifier.provider);
+ 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(BlackListNotifier.provider);
+ final blacklist = ref.watch(blacklistProvider);
+ final me = ref.watch(meProvider);
final favorites = useTrackToggleLike(track, ref);
final isBlackListed = useMemoized(
- () => blacklist.contains(
- BlacklistedElement.track(
- track.id!,
- track.name!,
- ),
+ () => blacklist.asData?.value.any(
+ (element) => element.elementId == track.id,
),
[blacklist, track],
);
final removingTrack = useState(null);
- final removeTrack = useMutations.playlist.removeTrackOf(
- ref,
- playlistId ?? "",
- );
+ final favoritePlaylistsNotifier =
+ ref.watch(favoritePlaylistsProvider.notifier);
final isInQueue = useMemoized(() {
if (playlist.activeTrack == null) return false;
@@ -209,6 +199,8 @@ class TrackOptions extends HookConsumerWidget {
return downloadManager.getProgressNotifier(spotubeTrack);
});
+ final isLocalTrack = track is LocalTrack;
+
final adaptivePopSheetList = AdaptivePopSheetList(
onSelected: (value) async {
switch (value) {
@@ -220,7 +212,7 @@ class TrackOptions extends HookConsumerWidget {
break;
case TrackOptionValue.delete:
await File((track as LocalTrack).path).delete();
- ref.refresh(localTracksProvider);
+ ref.invalidate(localTracksProvider);
break;
case TrackOptionValue.addToQueue:
await playback.addTrack(track);
@@ -257,23 +249,27 @@ class TrackOptions extends HookConsumerWidget {
);
break;
case TrackOptionValue.favorite:
- favorites.toggleTrackLike.mutate(favorites.isLiked);
+ favorites.toggleTrackLike(track);
break;
case TrackOptionValue.addToPlaylist:
actionAddToPlaylist(context, track);
break;
case TrackOptionValue.removeFromPlaylist:
removingTrack.value = track.uri;
- removeTrack.mutate(track.uri!);
+ favoritePlaylistsNotifier
+ .removeTracks(playlistId ?? "", [track.id!]);
break;
case TrackOptionValue.blacklist:
- if (isBlackListed) {
- ref.read(BlackListNotifier.provider.notifier).remove(
- BlacklistedElement.track(track.id!, track.name!),
- );
+ if (isBlackListed == null) break;
+ if (isBlackListed == true) {
+ await ref.read(blacklistProvider.notifier).remove(track.id!);
} else {
- ref.read(BlackListNotifier.provider.notifier).add(
- BlacklistedElement.track(track.id!, track.name!),
+ await ref.read(blacklistProvider.notifier).add(
+ BlacklistTableCompanion.insert(
+ name: track.name!,
+ elementId: track.id!,
+ elementType: BlacklistedType.track,
+ ),
);
}
break;
@@ -307,8 +303,8 @@ class TrackOptions extends HookConsumerWidget {
child: ClipRRect(
borderRadius: BorderRadius.circular(10),
child: UniversalImage(
- path: TypeConversionUtils.image_X_UrlString(track.album!.images,
- placeholder: ImagePlaceholder.albumArt),
+ path: track.album!.images
+ .asUrlString(placeholder: ImagePlaceholder.albumArt),
fit: BoxFit.cover,
),
),
@@ -321,127 +317,133 @@ class TrackOptions extends HookConsumerWidget {
),
subtitle: Align(
alignment: Alignment.centerLeft,
- child: TypeConversionUtils.artists_X_ClickableArtists(
- track.artists!,
+ child: ArtistLink(
+ artists: track.artists!,
+ onOverflowArtistClick: () => ServiceUtils.pushNamed(
+ context,
+ TrackPage.name,
+ pathParameters: {
+ "id": track.id!,
+ },
+ ),
),
),
),
],
- children: switch (track.runtimeType) {
- LocalTrack => [
- PopSheetEntry(
- value: TrackOptionValue.delete,
- leading: const Icon(SpotubeIcons.trash),
- title: Text(context.l10n.delete),
- )
- ],
- _ => [
- if (mediaQuery.smAndDown)
- PopSheetEntry(
- value: TrackOptionValue.album,
- leading: const Icon(SpotubeIcons.album),
- title: Text(context.l10n.go_to_album),
- subtitle: Text(track.album!.name!),
- ),
- if (!playlist.containsTrack(track)) ...[
- PopSheetEntry(
- value: TrackOptionValue.addToQueue,
- leading: const Icon(SpotubeIcons.queueAdd),
- title: Text(context.l10n.add_to_queue),
- ),
- PopSheetEntry(
- value: TrackOptionValue.playNext,
- leading: const Icon(SpotubeIcons.lightning),
- title: Text(context.l10n.play_next),
- ),
- ] else
- PopSheetEntry(
- value: TrackOptionValue.removeFromQueue,
- enabled: playlist.activeTrack?.id != track.id,
- leading: const Icon(SpotubeIcons.queueRemove),
- title: Text(context.l10n.remove_from_queue),
- ),
- if (favorites.me.hasData)
- PopSheetEntry(
- value: TrackOptionValue.favorite,
- leading: favorites.isLiked
- ? const Icon(
- SpotubeIcons.heartFilled,
- color: Colors.pink,
- )
- : const Icon(SpotubeIcons.heart),
- title: Text(
- favorites.isLiked
- ? context.l10n.remove_from_favorites
- : context.l10n.save_as_favorite,
- ),
- ),
- if (auth != null) ...[
- PopSheetEntry(
- value: TrackOptionValue.startRadio,
- leading: const Icon(SpotubeIcons.radio),
- title: Text(context.l10n.start_a_radio),
- ),
- PopSheetEntry(
- value: TrackOptionValue.addToPlaylist,
- leading: const Icon(SpotubeIcons.playlistAdd),
- title: Text(context.l10n.add_to_playlist),
- ),
- ],
- if (userPlaylist && auth != null)
- PopSheetEntry(
- value: TrackOptionValue.removeFromPlaylist,
- leading: (removeTrack.isMutating || !removeTrack.hasData) &&
- removingTrack.value == track.uri
- ? const CircularProgressIndicator()
- : const Icon(SpotubeIcons.removeFilled),
- title: Text(context.l10n.remove_from_playlist),
- ),
- PopSheetEntry(
- value: TrackOptionValue.download,
- enabled: !isInQueue,
- leading: isInQueue
- ? HookBuilder(builder: (context) {
- final progress = useListenable(progressNotifier!);
- return CircularProgressIndicator(
- value: progress.value,
- );
- })
- : const Icon(SpotubeIcons.download),
- title: Text(context.l10n.download_track),
+ children: [
+ if (isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.delete,
+ leading: const Icon(SpotubeIcons.trash),
+ title: Text(context.l10n.delete),
+ ),
+ if (mediaQuery.smAndDown && !isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.album,
+ leading: const Icon(SpotubeIcons.album),
+ title: Text(context.l10n.go_to_album),
+ subtitle: Text(track.album!.name!),
+ ),
+ if (!playlist.containsTrack(track)) ...[
+ PopSheetEntry(
+ value: TrackOptionValue.addToQueue,
+ leading: const Icon(SpotubeIcons.queueAdd),
+ title: Text(context.l10n.add_to_queue),
+ ),
+ PopSheetEntry(
+ value: TrackOptionValue.playNext,
+ leading: const Icon(SpotubeIcons.lightning),
+ title: Text(context.l10n.play_next),
+ ),
+ ] else
+ PopSheetEntry(
+ value: TrackOptionValue.removeFromQueue,
+ enabled: playlist.activeTrack?.id != track.id,
+ leading: const Icon(SpotubeIcons.queueRemove),
+ title: Text(context.l10n.remove_from_queue),
+ ),
+ if (me.asData?.value != null && !isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.favorite,
+ leading: favorites.isLiked
+ ? const Icon(
+ SpotubeIcons.heartFilled,
+ color: Colors.pink,
+ )
+ : const Icon(SpotubeIcons.heart),
+ title: Text(
+ favorites.isLiked
+ ? context.l10n.remove_from_favorites
+ : context.l10n.save_as_favorite,
),
- PopSheetEntry(
- value: TrackOptionValue.blacklist,
- leading: const Icon(SpotubeIcons.playlistRemove),
- iconColor: !isBlackListed ? Colors.red[400] : null,
- textColor: !isBlackListed ? Colors.red[400] : null,
- title: Text(
- isBlackListed
- ? context.l10n.remove_from_blacklist
- : context.l10n.add_to_blacklist,
- ),
+ ),
+ if (auth.asData?.value != null && !isLocalTrack) ...[
+ PopSheetEntry(
+ value: TrackOptionValue.startRadio,
+ leading: const Icon(SpotubeIcons.radio),
+ title: Text(context.l10n.start_a_radio),
+ ),
+ PopSheetEntry(
+ value: TrackOptionValue.addToPlaylist,
+ leading: const Icon(SpotubeIcons.playlistAdd),
+ title: Text(context.l10n.add_to_playlist),
+ ),
+ ],
+ if (userPlaylist && auth.asData?.value != null && !isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.removeFromPlaylist,
+ leading: const Icon(SpotubeIcons.removeFilled),
+ title: Text(context.l10n.remove_from_playlist),
+ ),
+ if (!isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.download,
+ enabled: !isInQueue,
+ leading: isInQueue
+ ? HookBuilder(builder: (context) {
+ final progress = useListenable(progressNotifier!);
+ return CircularProgressIndicator(
+ value: progress.value,
+ );
+ })
+ : const Icon(SpotubeIcons.download),
+ title: Text(context.l10n.download_track),
+ ),
+ if (!isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.blacklist,
+ leading: const Icon(SpotubeIcons.playlistRemove),
+ iconColor: isBlackListed != true ? Colors.red[400] : null,
+ textColor: isBlackListed != true ? Colors.red[400] : null,
+ title: Text(
+ isBlackListed == true
+ ? context.l10n.remove_from_blacklist
+ : context.l10n.add_to_blacklist,
),
- PopSheetEntry(
- value: TrackOptionValue.share,
- leading: const Icon(SpotubeIcons.share),
- title: Text(context.l10n.share),
+ ),
+ if (!isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.share,
+ leading: const Icon(SpotubeIcons.share),
+ title: Text(context.l10n.share),
+ ),
+ if (!isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.songlink,
+ leading: Assets.logos.songlinkTransparent.image(
+ width: 22,
+ height: 22,
+ color: colorScheme.onSurface.withOpacity(0.5),
),
- PopSheetEntry(
- value: TrackOptionValue.songlink,
- leading: Assets.logos.songlinkTransparent.image(
- width: 22,
- height: 22,
- color: colorScheme.onSurface.withOpacity(0.5),
- ),
- title: Text(context.l10n.song_link),
- ),
- PopSheetEntry(
- value: TrackOptionValue.details,
- leading: const Icon(SpotubeIcons.info),
- title: Text(context.l10n.details),
- ),
- ]
- },
+ title: Text(context.l10n.song_link),
+ ),
+ if (!isLocalTrack)
+ PopSheetEntry(
+ value: TrackOptionValue.details,
+ leading: const Icon(SpotubeIcons.info),
+ title: Text(context.l10n.details),
+ ),
+ ],
);
//! This is the most ANTI pattern I've ever done, but it works
diff --git a/lib/components/track_tile/track_tile.dart b/lib/components/track_tile/track_tile.dart
new file mode 100644
index 00000000..8ab889f8
--- /dev/null
+++ b/lib/components/track_tile/track_tile.dart
@@ -0,0 +1,289 @@
+import 'dart:async';
+
+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:skeletonizer/skeletonizer.dart';
+import 'package:spotify/spotify.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/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/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';
+
+class TrackTile extends HookConsumerWidget {
+ /// [index] will not be shown if null
+ final int? index;
+ final Track track;
+ final bool selected;
+ final ValueChanged? onChanged;
+ final Future Function()? onTap;
+ final VoidCallback? onLongPress;
+ final bool userPlaylist;
+ final String? playlistId;
+ final AudioPlayerState playlist;
+
+ final List? leadingActions;
+
+ const TrackTile({
+ super.key,
+ this.index,
+ required this.track,
+ this.selected = false,
+ required this.playlist,
+ this.onTap,
+ this.onLongPress,
+ this.onChanged,
+ this.userPlaylist = false,
+ this.playlistId,
+ this.leadingActions,
+ });
+
+ @override
+ 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 isLoading = useState(false);
+
+ final isPlaying = playlist.activeTrack?.id == track.id;
+
+ final isSelected = isPlaying || isLoading.value;
+
+ 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,
+ ),
+ );
+ },
+ child: HoverBuilder(
+ permanentState: isSelected || constrains.smAndDown ? true : null,
+ builder: (context, isHovering) => ListTile(
+ selected: isSelected,
+ onTap: () async {
+ try {
+ isLoading.value = true;
+ await onTap?.call();
+ } finally {
+ if (context.mounted) {
+ isLoading.value = false;
+ }
+ }
+ },
+ onLongPress: onLongPress,
+ enabled: !isBlackListed,
+ contentPadding: EdgeInsets.zero,
+ tileColor: isBlackListed ? theme.colorScheme.errorContainer : null,
+ horizontalTitleGap: 12,
+ leadingAndTrailingTextStyle: theme.textTheme.bodyMedium,
+ 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,
+ ),
+ Stack(
+ children: [
+ ClipRRect(
+ borderRadius: BorderRadius.circular(4),
+ child: AspectRatio(
+ aspectRatio: 1,
+ child: UniversalImage(
+ path: (track.album?.images).asUrlString(
+ placeholder: ImagePlaceholder.albumArt,
+ ),
+ fit: BoxFit.cover,
+ ),
+ ),
+ ),
+ Positioned.fill(
+ child: AnimatedContainer(
+ duration: const Duration(milliseconds: 300),
+ decoration: BoxDecoration(
+ borderRadius: BorderRadius.circular(4),
+ color: isHovering
+ ? Colors.black.withOpacity(0.4)
+ : 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),
+ );
+ },
+ ),
+ ),
+ ),
+ ),
+ ),
+ ],
+ ),
+ ],
+ ),
+ title: Row(
+ children: [
+ Expanded(
+ flex: 6,
+ child: switch (track) {
+ LocalTrack() => Text(
+ track.name!,
+ maxLines: 1,
+ overflow: TextOverflow.ellipsis,
+ ),
+ _ => LinkText(
+ track.name!,
+ "/track/${track.id}",
+ push: true,
+ maxLines: 1,
+ overflow: TextOverflow.ellipsis,
+ ),
+ },
+ ),
+ if (constrains.mdAndUp) ...[
+ const SizedBox(width: 8),
+ Expanded(
+ flex: 4,
+ child: switch (track) {
+ LocalTrack() => 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,
+ push: true,
+ overflow: TextOverflow.ellipsis,
+ ),
+ )
+ },
+ ),
+ ],
+ ],
+ ),
+ subtitle: Align(
+ alignment: Alignment.centerLeft,
+ child: track is LocalTrack
+ ? Text(
+ 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!,
+ },
+ ),
+ ),
+ ),
+ ),
+ ),
+ trailing: Row(
+ mainAxisSize: MainAxisSize.min,
+ children: [
+ const SizedBox(width: 8),
+ Text(
+ Duration(milliseconds: track.durationMs ?? 0)
+ .toHumanReadableString(padZero: false),
+ maxLines: 1,
+ overflow: TextOverflow.ellipsis,
+ ),
+ TrackOptions(
+ track: track,
+ playlistId: playlistId,
+ userPlaylist: userPlaylist,
+ showMenuCbRef: showOptionCbRef,
+ ),
+ if (kIsDesktop) const Gap(10),
+ ],
+ ),
+ ),
+ ),
+ );
+ });
+ }
+}
diff --git a/lib/components/shared/tracks_view/sections/body/track_view_body.dart b/lib/components/tracks_view/sections/body/track_view_body.dart
similarity index 54%
rename from lib/components/shared/tracks_view/sections/body/track_view_body.dart
rename to lib/components/tracks_view/sections/body/track_view_body.dart
index 33c8fa82..0f161b0c 100644
--- a/lib/components/shared/tracks_view/sections/body/track_view_body.dart
+++ b/lib/components/tracks_view/sections/body/track_view_body.dart
@@ -8,23 +8,29 @@ 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/shared/expandable_search/expandable_search.dart';
-import 'package:spotube/components/shared/track_tile/track_tile.dart';
-import 'package:spotube/components/shared/tracks_view/sections/body/track_view_body_headers.dart';
-import 'package:spotube/components/shared/tracks_view/sections/body/use_is_user_playlist.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_provider.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.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({Key? key}) : super(key: key);
+ const TrackViewBodySection({super.key});
@override
Widget build(BuildContext context, ref) {
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final playlistNotifier = ref.watch(ProxyPlaylistNotifier.notifier);
+ 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));
@@ -60,6 +66,56 @@ class TrackViewBodySection extends HookConsumerWidget {
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(
@@ -89,6 +145,7 @@ class TrackViewBodySection extends HookConsumerWidget {
loadingBuilder: (context) => Skeletonizer(
enabled: true,
child: TrackTile(
+ playlist: playlist,
track: FakeData.track,
index: 0,
),
@@ -98,13 +155,18 @@ class TrackViewBodySection extends HookConsumerWidget {
child: Column(
children: List.generate(
10,
- (index) => TrackTile(track: FakeData.track, index: index),
+ (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!),
@@ -119,24 +181,7 @@ class TrackViewBodySection extends HookConsumerWidget {
trackViewState.selectTrack(track.id!);
HapticFeedback.selectionClick();
},
- onTap: () async {
- if (trackViewState.isSelecting) {
- trackViewState.toggleTrackSelection(track.id!);
- return;
- }
-
- if (isActive || playlist.tracks.contains(track)) {
- await playlistNotifier.jumpToTrack(track);
- } else {
- final tracks = await props.pagination.onFetchAll();
- await playlistNotifier.load(
- tracks,
- initialIndex: index,
- autoPlay: true,
- );
- playlistNotifier.addCollection(props.collectionId);
- }
- },
+ onTap: () => onTapTrackTile(track, index),
);
},
),
diff --git a/lib/components/shared/tracks_view/sections/body/track_view_body_headers.dart b/lib/components/tracks_view/sections/body/track_view_body_headers.dart
similarity index 85%
rename from lib/components/shared/tracks_view/sections/body/track_view_body_headers.dart
rename to lib/components/tracks_view/sections/body/track_view_body_headers.dart
index 7e4522a0..82cc7706 100644
--- a/lib/components/shared/tracks_view/sections/body/track_view_body_headers.dart
+++ b/lib/components/tracks_view/sections/body/track_view_body_headers.dart
@@ -1,22 +1,24 @@
import 'package:flutter/material.dart';
+import 'package:gap/gap.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotube/components/shared/expandable_search/expandable_search.dart';
-import 'package:spotube/components/shared/sort_tracks_dropdown.dart';
-import 'package:spotube/components/shared/tracks_view/sections/body/track_view_options.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_provider.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({
- Key? key,
+ super.key,
required this.isFiltering,
required this.searchFocus,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context, ref) {
@@ -94,6 +96,7 @@ class TrackViewBodyHeaders extends HookConsumerWidget {
},
),
const TrackViewBodyOptions(),
+ if (kIsDesktop) const Gap(10),
],
);
},
diff --git a/lib/components/shared/tracks_view/sections/body/track_view_options.dart b/lib/components/tracks_view/sections/body/track_view_options.dart
similarity index 74%
rename from lib/components/shared/tracks_view/sections/body/track_view_options.dart
rename to lib/components/tracks_view/sections/body/track_view_options.dart
index 583c9107..23198aec 100644
--- a/lib/components/shared/tracks_view/sections/body/track_view_options.dart
+++ b/lib/components/tracks_view/sections/body/track_view_options.dart
@@ -1,19 +1,21 @@
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/shared/adaptive/adaptive_pop_sheet_list.dart';
-import 'package:spotube/components/shared/dialogs/confirm_download_dialog.dart';
-import 'package:spotube/components/shared/dialogs/playlist_add_track_dialog.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_provider.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/proxy_playlist/proxy_playlist_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';
-import 'package:spotube/provider/user_preferences/user_preferences_state.dart';
class TrackViewBodyOptions extends HookConsumerWidget {
- const TrackViewBodyOptions({Key? key}) : super(key: key);
+ const TrackViewBodyOptions({super.key});
@override
Widget build(BuildContext context, ref) {
@@ -22,7 +24,8 @@ class TrackViewBodyOptions extends HookConsumerWidget {
ref.watch(downloadManagerProvider);
final downloader = ref.watch(downloadManagerProvider.notifier);
- final playlistNotifier = ref.watch(ProxyPlaylistNotifier.notifier);
+ final playlistNotifier = ref.watch(audioPlayerProvider.notifier);
+ final historyNotifier = ref.watch(playbackHistoryActionsProvider);
final audioSource =
ref.watch(userPreferencesProvider.select((s) => s.audioSource));
@@ -72,6 +75,12 @@ class TrackViewBodyOptions extends HookConsumerWidget {
{
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;
}
@@ -79,6 +88,12 @@ class TrackViewBodyOptions extends HookConsumerWidget {
{
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;
}
diff --git a/lib/components/tracks_view/sections/body/use_is_user_playlist.dart b/lib/components/tracks_view/sections/body/use_is_user_playlist.dart
new file mode 100644
index 00000000..2f87ccc8
--- /dev/null
+++ b/lib/components/tracks_view/sections/body/use_is_user_playlist.dart
@@ -0,0 +1,18 @@
+import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:spotube/provider/spotify/spotify.dart';
+
+bool useIsUserPlaylist(WidgetRef ref, String playlistId) {
+ final userPlaylistsQuery = ref.watch(favoritePlaylistsProvider);
+ final me = ref.watch(meProvider);
+
+ return useMemoized(
+ () =>
+ userPlaylistsQuery.asData?.value.items.any((e) =>
+ e.id == playlistId &&
+ me.asData?.value != null &&
+ e.owner?.id == me.asData?.value.id) ??
+ false,
+ [userPlaylistsQuery.asData?.value, playlistId, me.asData?.value],
+ );
+}
diff --git a/lib/components/shared/tracks_view/sections/header/flexible_header.dart b/lib/components/tracks_view/sections/header/flexible_header.dart
similarity index 88%
rename from lib/components/shared/tracks_view/sections/header/flexible_header.dart
rename to lib/components/tracks_view/sections/header/flexible_header.dart
index 19241dc6..508d289c 100644
--- a/lib/components/shared/tracks_view/sections/header/flexible_header.dart
+++ b/lib/components/tracks_view/sections/header/flexible_header.dart
@@ -1,20 +1,21 @@
import 'dart:ui';
import 'package:flutter/material.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
+
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotube/collections/assets.gen.dart';
-import 'package:spotube/components/shared/image/universal_image.dart';
-import 'package:spotube/components/shared/playbutton_card.dart';
-import 'package:spotube/components/shared/tracks_view/sections/header/header_actions.dart';
-import 'package:spotube/components/shared/tracks_view/sections/header/header_buttons.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.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({Key? key}) : super(key: key);
+ const TrackViewFlexHeader({super.key});
@override
Widget build(BuildContext context, ref) {
@@ -23,8 +24,6 @@ class TrackViewFlexHeader extends HookConsumerWidget {
final defaultTextStyle = DefaultTextStyle.of(context);
final mediaQuery = MediaQuery.of(context);
- final description = useDescription(props.description);
-
final palette = usePaletteColor(props.image, ref);
return IconTheme(
@@ -53,7 +52,7 @@ class TrackViewFlexHeader extends HookConsumerWidget {
floating: false,
pinned: true,
expandedHeight: 450,
- automaticallyImplyLeading: DesktopTools.platform.isMobile,
+ automaticallyImplyLeading: kIsMobile,
backgroundColor: palette.color,
title: isExpanded ? null : Text(props.title, style: headingStyle),
flexibleSpace: FlexibleSpaceBar(
@@ -126,10 +125,12 @@ class TrackViewFlexHeader extends HookConsumerWidget {
overflow: TextOverflow.ellipsis,
),
const SizedBox(height: 10),
- if (description != null &&
- description.isNotEmpty)
+ if (props.description != null &&
+ props.description!.isNotEmpty)
Text(
- description,
+ props.description!
+ .unescapeHtml()
+ .cleanHtml(),
style:
defaultTextStyle.style.copyWith(
color: palette.bodyTextColor,
diff --git a/lib/components/shared/tracks_view/sections/header/header_actions.dart b/lib/components/tracks_view/sections/header/header_actions.dart
similarity index 67%
rename from lib/components/shared/tracks_view/sections/header/header_actions.dart
rename to lib/components/tracks_view/sections/header/header_actions.dart
index 75aa3f61..8e378f97 100644
--- a/lib/components/shared/tracks_view/sections/header/header_actions.dart
+++ b/lib/components/tracks_view/sections/header/header_actions.dart
@@ -2,24 +2,27 @@ 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/components/playlist/playlist_create_dialog.dart';
-import 'package:spotube/components/shared/heart_button.dart';
-import 'package:spotube/components/shared/tracks_view/sections/body/use_is_user_playlist.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.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_provider.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.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({Key? key}) : super(key: key);
+ const TrackViewHeaderActions({super.key});
@override
Widget build(BuildContext context, ref) {
final props = InheritedTrackView.of(context);
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
- final playlistNotifier = ref.watch(ProxyPlaylistNotifier.notifier);
+ final playlist = ref.watch(audioPlayerProvider);
+ final playlistNotifier = ref.watch(audioPlayerProvider.notifier);
+ final historyNotifier = ref.watch(playbackHistoryActionsProvider);
final isActive = playlist.collections.contains(props.collectionId);
@@ -27,7 +30,10 @@ class TrackViewHeaderActions extends HookConsumerWidget {
final scaffoldMessenger = ScaffoldMessenger.of(context);
- final auth = ref.watch(AuthenticationNotifier.provider);
+ final auth = ref.watch(authenticationProvider);
+
+ final copiedText =
+ context.l10n.copied_shareurl_to_clipboard(props.shareUrl);
return Row(
mainAxisSize: MainAxisSize.min,
@@ -45,7 +51,7 @@ class TrackViewHeaderActions extends HookConsumerWidget {
width: 300,
behavior: SnackBarBehavior.floating,
content: Text(
- "Copied ${props.shareUrl} to clipboard",
+ copiedText,
textAlign: TextAlign.center,
),
),
@@ -61,9 +67,16 @@ class TrackViewHeaderActions extends HookConsumerWidget {
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 != null)
+ if (props.onHeart != null && auth.asData?.value != null)
HeartButton(
isLiked: props.isLiked,
icon: isUserPlaylist ? SpotubeIcons.trash : null,
diff --git a/lib/components/tracks_view/sections/header/header_buttons.dart b/lib/components/tracks_view/sections/header/header_buttons.dart
new file mode 100644
index 00000000..54e0f0cf
--- /dev/null
+++ b/lib/components/tracks_view/sections/header/header_buttons.dart
@@ -0,0 +1,206 @@
+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/shared/tracks_view/track_view.dart b/lib/components/tracks_view/track_view.dart
similarity index 69%
rename from lib/components/shared/tracks_view/track_view.dart
rename to lib/components/tracks_view/track_view.dart
index 4103573c..2a3f5237 100644
--- a/lib/components/shared/tracks_view/track_view.dart
+++ b/lib/components/tracks_view/track_view.dart
@@ -1,16 +1,17 @@
import 'package:flutter/material.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.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/shared/inter_scrollbar/inter_scrollbar.dart';
-import 'package:spotube/components/shared/page_window_title_bar.dart';
-import 'package:spotube/components/shared/tracks_view/sections/header/flexible_header.dart';
-import 'package:spotube/components/shared/tracks_view/sections/body/track_view_body.dart';
-import 'package:spotube/components/shared/tracks_view/track_view_props.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({Key? key}) : super(key: key);
+ const TrackView({super.key});
@override
Widget build(BuildContext context, ref) {
@@ -18,7 +19,7 @@ class TrackView extends HookConsumerWidget {
final controller = useScrollController();
return Scaffold(
- appBar: DesktopTools.platform.isDesktop
+ appBar: kIsDesktop
? const PageWindowTitleBar(
backgroundColor: Colors.transparent,
foregroundColor: Colors.white,
diff --git a/lib/components/shared/tracks_view/track_view_props.dart b/lib/components/tracks_view/track_view_props.dart
similarity index 82%
rename from lib/components/shared/tracks_view/track_view_props.dart
rename to lib/components/tracks_view/track_view_props.dart
index 21bbaec7..b0a00ae2 100644
--- a/lib/components/shared/tracks_view/track_view_props.dart
+++ b/lib/components/tracks_view/track_view_props.dart
@@ -1,6 +1,5 @@
import 'dart:async';
-import 'package:fl_query/fl_query.dart';
import 'package:flutter/material.dart' hide Page;
import 'package:spotify/spotify.dart';
@@ -19,19 +18,6 @@ class PaginationProps {
required this.onRefresh,
});
- factory PaginationProps.fromQuery(
- InfiniteQuery, dynamic, int> query, {
- required Future> Function() onFetchAll,
- }) {
- return PaginationProps(
- hasNextPage: query.hasNextPage,
- isLoading: query.isLoadingNextPage,
- onFetchMore: query.fetchNext,
- onFetchAll: onFetchAll,
- onRefresh: query.refreshAll,
- );
- }
-
@override
operator ==(Object other) {
return other is PaginationProps &&
@@ -53,7 +39,7 @@ class PaginationProps {
}
class InheritedTrackView extends InheritedWidget {
- final String collectionId;
+ final Object collection;
final String title;
final String? description;
final String image;
@@ -69,7 +55,7 @@ class InheritedTrackView extends InheritedWidget {
const InheritedTrackView({
super.key,
required super.child,
- required this.collectionId,
+ required this.collection,
required this.title,
this.description,
required this.image,
@@ -79,7 +65,11 @@ class InheritedTrackView extends InheritedWidget {
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) {
@@ -92,7 +82,7 @@ class InheritedTrackView extends InheritedWidget {
oldWidget.onHeart != onHeart ||
oldWidget.shareUrl != shareUrl ||
oldWidget.routePath != routePath ||
- oldWidget.collectionId != collectionId ||
+ oldWidget.collection != collection ||
oldWidget.child != child;
}
diff --git a/lib/components/shared/tracks_view/track_view_provider.dart b/lib/components/tracks_view/track_view_provider.dart
similarity index 95%
rename from lib/components/shared/tracks_view/track_view_provider.dart
rename to lib/components/tracks_view/track_view_provider.dart
index 14dc1136..16aa6d9c 100644
--- a/lib/components/shared/tracks_view/track_view_provider.dart
+++ b/lib/components/tracks_view/track_view_provider.dart
@@ -1,7 +1,7 @@
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotify/spotify.dart';
-import 'package:spotube/components/library/user_local_tracks.dart';
+import 'package:spotube/modules/library/user_local_tracks.dart';
class TrackViewNotifier extends ChangeNotifier {
List tracks;
diff --git a/lib/components/shared/waypoint.dart b/lib/components/waypoint.dart
similarity index 85%
rename from lib/components/shared/waypoint.dart
rename to lib/components/waypoint.dart
index abd9f98d..cf00e29b 100644
--- a/lib/components/shared/waypoint.dart
+++ b/lib/components/waypoint.dart
@@ -11,17 +11,15 @@ class Waypoint extends HookWidget {
final bool isGrid;
const Waypoint({
- Key? key,
+ super.key,
required this.controller,
this.isGrid = false,
this.onTouchEdge,
this.child,
- }) : super(key: key);
+ });
@override
Widget build(BuildContext context) {
- final isMounted = useIsMounted();
-
useEffect(() {
if (isGrid) {
return null;
@@ -32,19 +30,19 @@ class Waypoint extends HookWidget {
// scrollController fetches the next paginated data when the current
// position of the user on the screen has surpassed
- if (controller.position.pixels >= nextPageTrigger && isMounted()) {
+ if (controller.position.pixels >= nextPageTrigger && context.mounted) {
await onTouchEdge?.call();
}
}
WidgetsBinding.instance.addPostFrameCallback((_) {
- if (controller.hasClients && isMounted()) {
+ if (controller.hasClients && context.mounted) {
listener();
controller.addListener(listener);
}
});
return () => controller.removeListener(listener);
- }, [controller, onTouchEdge, isMounted]);
+ }, [controller, onTouchEdge]);
if (isGrid) {
return VisibilityDetector(
diff --git a/lib/extensions/album_simple.dart b/lib/extensions/album_simple.dart
index 00db4dca..5678390c 100644
--- a/lib/extensions/album_simple.dart
+++ b/lib/extensions/album_simple.dart
@@ -1,18 +1,21 @@
import 'package:spotify/spotify.dart';
-extension AlbumJson on AlbumSimple {
- Map toJson() {
- return {
- "albumType": albumType?.name,
- "id": id,
- "name": name,
- "images": images
- ?.map((image) => {
- "height": image.height,
- "url": image.url,
- "width": image.width,
- })
- .toList(),
- };
+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
index caf2e510..7997355d 100644
--- a/lib/extensions/artist_simple.dart
+++ b/lib/extensions/artist_simple.dart
@@ -1,13 +1,7 @@
import 'package:spotify/spotify.dart';
-extension ArtistJson on ArtistSimple {
- Map toJson() {
- return {
- "href": href,
- "id": id,
- "name": name,
- "type": type,
- "uri": uri,
- };
+extension ArtistExtension on List {
+ String asString() {
+ return map((e) => e.name?.replaceAll(",", " ")).join(", ");
}
}
diff --git a/lib/extensions/constrains.dart b/lib/extensions/constrains.dart
index 1177f5ac..dc1027e2 100644
--- a/lib/extensions/constrains.dart
+++ b/lib/extensions/constrains.dart
@@ -1,6 +1,20 @@
import 'package:flutter/rendering.dart';
import 'package:flutter/widgets.dart';
+enum Breakpoint {
+ xs,
+ sm,
+ md,
+ lg,
+ xl,
+ xxl;
+
+ bool operator <=(Breakpoint other) => index <= other.index;
+ bool operator <(Breakpoint other) => index < other.index;
+ bool operator >(Breakpoint other) => index > other.index;
+ bool operator >=(Breakpoint other) => index >= other.index;
+}
+
// ignore: constant_identifier_names
const Breakpoints = (
xs: 480.0,
@@ -22,6 +36,15 @@ extension SliverBreakpoints on SliverConstraints {
crossAxisExtent > Breakpoints.lg && crossAxisExtent <= Breakpoints.xl;
bool get is2Xl => crossAxisExtent > Breakpoints.xl;
+ Breakpoint get breakpoint {
+ if (isXs) return Breakpoint.xs;
+ if (isSm) return Breakpoint.sm;
+ if (isMd) return Breakpoint.md;
+ if (isLg) return Breakpoint.lg;
+ if (isXl) return Breakpoint.xl;
+ return Breakpoint.xxl;
+ }
+
bool get smAndUp => isSm || isMd || isLg || isXl || is2Xl;
bool get mdAndUp => isMd || isLg || isXl || is2Xl;
bool get lgAndUp => isLg || isXl || is2Xl;
@@ -45,6 +68,15 @@ extension ContainerBreakpoints on BoxConstraints {
biggest.width > Breakpoints.lg && biggest.width <= Breakpoints.xl;
bool get is2Xl => biggest.width > Breakpoints.xl;
+ Breakpoint get breakpoint {
+ if (isXs) return Breakpoint.xs;
+ if (isSm) return Breakpoint.sm;
+ if (isMd) return Breakpoint.md;
+ if (isLg) return Breakpoint.lg;
+ if (isXl) return Breakpoint.xl;
+ return Breakpoint.xxl;
+ }
+
bool get smAndUp => isSm || isMd || isLg || isXl || is2Xl;
bool get mdAndUp => isMd || isLg || isXl || is2Xl;
bool get lgAndUp => isLg || isXl || is2Xl;
diff --git a/lib/extensions/image.dart b/lib/extensions/image.dart
new file mode 100644
index 00000000..ee78653a
--- /dev/null
+++ b/lib/extensions/image.dart
@@ -0,0 +1,34 @@
+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/infinite_query.dart b/lib/extensions/infinite_query.dart
deleted file mode 100644
index 2181ab3c..00000000
--- a/lib/extensions/infinite_query.dart
+++ /dev/null
@@ -1,34 +0,0 @@
-import 'package:fl_query/fl_query.dart';
-import 'package:spotify/spotify.dart';
-
-extension FetchAllTracks on InfiniteQuery, dynamic, int> {
- Future> fetchAllTracks({
- required Future> Function() getAllTracks,
- }) async {
- if (pages.isNotEmpty && !hasNextPage) {
- return pages.expand((page) => page).toList();
- }
- final tracks = await getAllTracks();
-
- final numOfPages = (tracks.length / 20).round();
-
- final Map> pagedTracks = {};
-
- for (var i = 0; i < numOfPages; i++) {
- if (i == numOfPages - 1) {
- final pageTracks = tracks.sublist(i * 20);
- pagedTracks[i] = pageTracks;
- break;
- }
-
- final pageTracks = tracks.sublist(i * 20, (i + 1) * 20);
- pagedTracks[i] = pageTracks;
- }
-
- for (final group in pagedTracks.entries) {
- setPageData(group.key, group.value);
- }
-
- return tracks.toList();
- }
-}
diff --git a/lib/extensions/list.dart b/lib/extensions/list.dart
index 6ecf6cf6..ddd36e4d 100644
--- a/lib/extensions/list.dart
+++ b/lib/extensions/list.dart
@@ -1,99 +1,19 @@
-import 'package:collection/collection.dart';
-import 'package:spotube/models/logger.dart';
+extension UniqueItemExtension on List {
+ List unique(bool Function(T a, T b) equals) {
+ final copy = [];
-final logger = getLogger("List");
-
-extension MultiSortListMap on List {
- /// [preference] - List of properties in which you want to sort the list
- /// i.e.
- /// ```
- /// List preference = ['property1','property2'];
- /// ```
- /// This will first sort the list by property1 then by property2
- ///
- /// [criteria] - List of booleans that specifies the criteria of sort
- /// i.e., For ascending order `true` and for descending order `false`.
- /// ```
- /// List criteria = [true. false];
- /// ```
- List sortByProperties(List criteria, List preference) {
- if (preference.isEmpty || criteria.isEmpty || isEmpty) {
- return this;
- }
- if (preference.length != criteria.length) {
- logger.d('Criteria length is not equal to preference');
- return this;
+ for (final item in this) {
+ if (copy.any((element) => equals(element, item))) continue;
+ copy.add(item);
}
- int compare(int i, Map a, Map b) {
- if (a[preference[i]] == b[preference[i]]) {
- return 0;
- } else if (a[preference[i]] > b[preference[i]]) {
- return criteria[i] ? 1 : -1;
- } else {
- return criteria[i] ? -1 : 1;
- }
- }
+ return copy;
+ }
- int sortAll(Map a, Map b) {
- int i = 0;
- int result = 0;
- while (i < preference.length) {
- result = compare(i, a, b);
- if (result != 0) break;
- i++;
- }
- return result;
+ bool containsBy(T item, dynamic Function(T a) fn) {
+ for (final el in this) {
+ if (fn(el) == fn(item)) return true;
}
-
- return sorted((a, b) => sortAll(a, b));
- }
-}
-
-extension MultiSortListTupleMap on List<(Map, V)> {
- /// [preference] - List of properties in which you want to sort the list
- /// i.e.
- /// ```
- /// List preference = ['property1','property2'];
- /// ```
- /// This will first sort the list by property1 then by property2
- ///
- /// [criteria] - List of booleans that specifies the criteria of sort
- /// i.e., For ascending order `true` and for descending order `false`.
- /// ```
- /// List criteria = [true. false];
- /// ```
- List<(Map, V)> sortByProperties(
- List criteria, List preference) {
- if (preference.isEmpty || criteria.isEmpty || isEmpty) {
- return this;
- }
- if (preference.length != criteria.length) {
- logger.d('Criteria length is not equal to preference');
- return this;
- }
-
- int compare(int i, (Map, V) a, (Map, V) b) {
- if (a.$1[preference[i]] == b.$1[preference[i]]) {
- return 0;
- } else if (a.$1[preference[i]] > b.$1[preference[i]]) {
- return criteria[i] ? 1 : -1;
- } else {
- return criteria[i] ? -1 : 1;
- }
- }
-
- int sortAll((Map, V) a, (Map, V) b) {
- int i = 0;
- int result = 0;
- while (i < preference.length) {
- result = compare(i, a, b);
- if (result != 0) break;
- i++;
- }
- return result;
- }
-
- return sorted((a, b) => sortAll(a, b));
+ return false;
}
}
diff --git a/lib/extensions/string.dart b/lib/extensions/string.dart
index b7ab7514..94123fe3 100644
--- a/lib/extensions/string.dart
+++ b/lib/extensions/string.dart
@@ -1,11 +1,20 @@
import 'package:html_unescape/html_unescape.dart';
+import 'package:html/parser.dart';
final htmlEscape = HtmlUnescape();
extension UnescapeHtml on String {
+ String cleanHtml() => parse("$this
").documentElement!.text;
String unescapeHtml() => htmlEscape.convert(this);
}
extension NullableUnescapeHtml on String? {
- String? unescapeHtml() => this == null ? null : htmlEscape.convert(this!);
+ String? cleanHtml() => this?.cleanHtml();
+ String? unescapeHtml() => this?.unescapeHtml();
+}
+
+extension StringExtension on String {
+ String capitalize() {
+ return "${this[0].toUpperCase()}${substring(1)}";
+ }
}
diff --git a/lib/extensions/track.dart b/lib/extensions/track.dart
index 51498b33..02c0c492 100644
--- a/lib/extensions/track.dart
+++ b/lib/extensions/track.dart
@@ -1,32 +1,70 @@
+import 'dart:io';
+
+import 'package:metadata_god/metadata_god.dart';
+import 'package:path/path.dart';
import 'package:spotify/spotify.dart';
-import 'package:spotube/extensions/album_simple.dart';
-import 'package:spotube/extensions/artist_simple.dart';
+import 'package:spotube/services/audio_player/audio_player.dart';
-extension TrackJson on Track {
- Map toJson() {
- return TrackJson.trackToJson(this);
- }
+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"
+ ];
- static Map trackToJson(Track track) {
- return {
- "album": track.album?.toJson(),
- "artists": track.artists?.map((artist) => artist.toJson()).toList(),
- "available_markets": track.availableMarkets?.map((e) => e.name).toList(),
- "disc_number": track.discNumber,
- "duration_ms": track.durationMs,
- "explicit": track.explicit,
- // "external_ids"track.: externalIds,
- // "external_urls"track.: externalUrls,
- "href": track.href,
- "id": track.id,
- "is_playable": track.isPlayable,
- // "linked_from"track.: linkedFrom,
- "name": track.name,
- "popularity": track.popularity,
- "preview_rrl": track.previewUrl,
- "track_number": track.trackNumber,
- "type": track.type,
- "uri": track.uri,
- };
+ id = metadata?.title ?? basenameWithoutExtension(file.path);
+ name = metadata?.title ?? basenameWithoutExtension(file.path);
+ type = "track";
+ uri = file.path;
+ durationMs = (metadata?.durationMs?.toInt() ?? 0);
+
+ return this;
+ }
+}
+
+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_close_behavior.dart b/lib/hooks/configurators/use_close_behavior.dart
index 05c03fff..2bdc65ef 100644
--- a/lib/hooks/configurators/use_close_behavior.dart
+++ b/lib/hooks/configurators/use_close_behavior.dart
@@ -1,28 +1,32 @@
import 'dart:io';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotube/hooks/configurators/use_window_listener.dart';
+import 'package:spotube/models/database/database.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_state.dart';
-import 'package:local_notifier/local_notifier.dart';
-final closeNotification = DesktopTools.createNotification(
- title: 'Spotube',
- message: 'Running in background. Minimized to System Tray',
- actions: [
- LocalNotificationAction(text: 'Close The App'),
- ],
-)?..onClickAction = (value) {
- exit(0);
- };
+import 'package:local_notifier/local_notifier.dart';
+import 'package:spotube/utils/platform.dart';
+import 'package:window_manager/window_manager.dart';
+
+final closeNotification = !kIsDesktop
+ ? null
+ : (LocalNotification(
+ title: 'Spotube',
+ body: 'Running in background. Minimized to System Tray',
+ actions: [
+ LocalNotificationAction(text: 'Close The App'),
+ ],
+ )..onClickAction = (value) {
+ exit(0);
+ });
void useCloseBehavior(WidgetRef ref) {
useWindowListener(
onWindowClose: () async {
final preferences = ref.read(userPreferencesProvider);
if (preferences.closeBehavior == CloseBehavior.minimizeToTray) {
- await DesktopTools.window.hide();
+ await windowManager.hide();
closeNotification?.show();
} else {
exit(0);
diff --git a/lib/hooks/configurators/use_deep_linking.dart b/lib/hooks/configurators/use_deep_linking.dart
index f11a1cff..ec6d8516 100644
--- a/lib/hooks/configurators/use_deep_linking.dart
+++ b/lib/hooks/configurators/use_deep_linking.dart
@@ -1,24 +1,21 @@
import 'dart:async';
import 'package:app_links/app_links.dart';
-import 'package:fl_query_hooks/fl_query_hooks.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotify/spotify.dart';
import 'package:spotube/collections/routes.dart';
import 'package:spotube/provider/spotify_provider.dart';
import 'package:flutter_sharing_intent/flutter_sharing_intent.dart';
import 'package:flutter_sharing_intent/model/sharing_file.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
+import 'package:spotube/services/logger/logger.dart';
+import 'package:spotube/utils/platform.dart';
final appLinks = AppLinks();
-final linkStream = appLinks.allStringLinkStream.asBroadcastStream();
+final linkStream = appLinks.stringLinkStream.asBroadcastStream();
void useDeepLinking(WidgetRef ref) {
// single instance no worries
final spotify = ref.watch(spotifyProvider);
- final queryClient = useQueryClient();
-
final router = ref.watch(routerProvider);
useEffect(() {
@@ -32,10 +29,7 @@ void useDeepLinking(WidgetRef ref) {
case "album":
router.push(
"/album/${url.pathSegments.last}",
- extra: await queryClient.fetchQuery(
- "album/${url.pathSegments.last}",
- () => spotify.albums.get(url.pathSegments.last),
- ),
+ extra: await spotify.albums.get(url.pathSegments.last),
);
break;
case "artist":
@@ -44,10 +38,7 @@ void useDeepLinking(WidgetRef ref) {
case "playlist":
router.push(
"/playlist/${url.pathSegments.last}",
- extra: await queryClient.fetchQuery(
- "playlist/${url.pathSegments.last}",
- () => spotify.playlists.get(url.pathSegments.last),
- ),
+ extra: await spotify.playlists.get(url.pathSegments.last),
);
break;
case "track":
@@ -63,7 +54,7 @@ void useDeepLinking(WidgetRef ref) {
StreamSubscription? mediaStream;
- if (DesktopTools.platform.isMobile) {
+ if (kIsMobile) {
FlutterSharingIntent.instance.getInitialSharing().then(uriListener);
mediaStream =
@@ -71,36 +62,34 @@ void useDeepLinking(WidgetRef ref) {
}
final subscription = linkStream.listen((uri) async {
- final startSegment = uri.split(":").take(2).join(":");
- final endSegment = uri.split(":").last;
+ 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 queryClient.fetchQuery(
- "album/$endSegment",
- () => 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 queryClient.fetchQuery(
- "playlist/$endSegment",
- () => spotify.playlists.get(endSegment),
- ),
- );
- break;
- default:
- break;
+ 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);
}
});
@@ -108,5 +97,5 @@ void useDeepLinking(WidgetRef ref) {
mediaStream?.cancel();
subscription.cancel();
};
- }, [spotify, queryClient]);
+ }, [spotify]);
}
diff --git a/lib/hooks/configurators/use_disable_battery_optimizations.dart b/lib/hooks/configurators/use_disable_battery_optimizations.dart
index c1155d19..4aa51b74 100644
--- a/lib/hooks/configurators/use_disable_battery_optimizations.dart
+++ b/lib/hooks/configurators/use_disable_battery_optimizations.dart
@@ -1,47 +1,21 @@
import 'package:disable_battery_optimization/disable_battery_optimization.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
-import 'package:shared_preferences/shared_preferences.dart';
-import 'package:spotube/hooks/utils/use_async_effect.dart';
-bool _asked = false;
+import 'package:spotube/hooks/utils/use_async_effect.dart';
+import 'package:spotube/services/kv_store/kv_store.dart';
+import 'package:spotube/utils/platform.dart';
+
void useDisableBatteryOptimizations() {
useAsyncEffect(() async {
- if (!DesktopTools.platform.isAndroid || _asked) return;
- final localStorage = await SharedPreferences.getInstance();
+ if (!kIsAndroid || KVStoreService.askedForBatteryOptimization) return;
- final rawIsBatteryOptimizationDisabled =
- localStorage.getBool("isBatteryOptimizationDisabled");
- final isBatteryOptimizationDisabled =
- await DisableBatteryOptimization.isBatteryOptimizationDisabled;
- if (rawIsBatteryOptimizationDisabled != false &&
- isBatteryOptimizationDisabled == false) {
- final hasDisabled = await DisableBatteryOptimization
- .showDisableBatteryOptimizationSettings();
+ await DisableBatteryOptimization.showDisableBatteryOptimizationSettings();
- localStorage.setBool(
- "isBatteryOptimizationDisabled",
- hasDisabled == true,
- );
- }
+ await DisableBatteryOptimization
+ .showDisableManufacturerBatteryOptimizationSettings(
+ "Your device has additional battery optimization",
+ "Follow the steps and disable the optimizations to allow smooth functioning of this app",
+ );
- final rawIsManBatteryOptimizationDisabled =
- localStorage.getBool("isManufacturerBatteryOptimizationDisabled");
- final isManBatteryOptimizationDisabled = await DisableBatteryOptimization
- .isManufacturerBatteryOptimizationDisabled;
-
- if (rawIsManBatteryOptimizationDisabled != false &&
- isManBatteryOptimizationDisabled == false) {
- final hasDisabled = await DisableBatteryOptimization
- .showDisableManufacturerBatteryOptimizationSettings(
- "Your device has additional battery optimization",
- "Follow the steps and disable the optimizations to allow smooth functioning of this app",
- );
-
- localStorage.setBool(
- "isManufacturerBatteryOptimizationDisabled",
- hasDisabled == true,
- );
- }
- _asked = true;
+ await KVStoreService.setAskedForBatteryOptimization(true);
}, null, []);
}
diff --git a/lib/hooks/configurators/use_endless_playback.dart b/lib/hooks/configurators/use_endless_playback.dart
index f5d11829..e2fb1e6e 100644
--- a/lib/hooks/configurators/use_endless_playback.dart
+++ b/lib/hooks/configurators/use_endless_playback.dart
@@ -1,47 +1,35 @@
-import 'package:catcher_2/catcher_2.dart';
-import 'package:fl_query_hooks/fl_query_hooks.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_provider.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.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';
-import 'package:spotube/services/queries/search.dart';
void useEndlessPlayback(WidgetRef ref) {
- final auth = ref.watch(AuthenticationNotifier.provider);
- final playback = ref.watch(ProxyPlaylistNotifier.notifier);
- final playlist = ref.watch(ProxyPlaylistNotifier.provider);
+ 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 endlessPlayback =
ref.watch(userPreferencesProvider.select((s) => s.endlessPlayback));
- final queryClient = useQueryClient();
-
useEffect(
() {
- if (!endlessPlayback || auth == null) return null;
+ if (!endlessPlayback || auth.asData?.value == null) return null;
void listener(int index) async {
try {
- final playlist = ref.read(ProxyPlaylistNotifier.provider);
+ final playlist = ref.read(audioPlayerProvider);
if (index != playlist.tracks.length - 1) return;
final track = playlist.tracks.last;
final query = "${track.name} Radio";
- final pages = await queryClient.fetchInfiniteQueryJob,
- dynamic, int, SearchParams>(
- job: SearchQueries.queryJob(query),
- args: (
- spotify: spotify,
- searchType: SearchType.playlist,
- query: query
- ),
- ) ??
- [];
+ final pages = await spotify.search
+ .get(query, types: [SearchType.playlist]).first();
final radios = pages
.expand((e) => e.items?.toList() ?? [])
@@ -68,22 +56,22 @@ void useEndlessPlayback(WidgetRef ref) {
await playback.addTracks(
tracks.toList()
..removeWhere((e) {
- final playlist = ref.read(ProxyPlaylistNotifier.provider);
+ final playlist = ref.read(audioPlayerProvider);
final isDuplicate = playlist.tracks.any((t) => t.id == e.id);
return e.id == track.id || isDuplicate;
}),
);
} catch (e, stack) {
- Catcher2.reportCheckedError(e, stack);
+ AppLogger.reportError(e, stack);
}
}
// 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.active == playlist.tracks.length - 1 &&
+ if (playlist.index == playlist.medias.length - 1 &&
audioPlayer.isPlaying) {
- listener(playlist.active!);
+ listener(playlist.index);
}
final subscription =
@@ -94,8 +82,7 @@ void useEndlessPlayback(WidgetRef ref) {
[
spotify,
playback,
- queryClient,
- playlist.tracks,
+ playlist.medias,
endlessPlayback,
auth,
],
diff --git a/lib/hooks/configurators/use_fix_window_stretching.dart b/lib/hooks/configurators/use_fix_window_stretching.dart
new file mode 100644
index 00000000..a6603d59
--- /dev/null
+++ b/lib/hooks/configurators/use_fix_window_stretching.dart
@@ -0,0 +1,21 @@
+import 'package:flutter/material.dart';
+import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:spotube/utils/platform.dart';
+import 'package:window_manager/window_manager.dart';
+
+void useFixWindowStretching() {
+ useEffect(() {
+ if (!kIsWindows) return;
+ WidgetsBinding.instance.addPostFrameCallback((Duration timeStamp) async {
+ await Future.delayed(const Duration(milliseconds: 100), () {
+ windowManager.getSize().then((Size value) {
+ windowManager.setSize(
+ Size(value.width + 1, value.height + 1),
+ );
+ });
+ });
+ });
+
+ return null;
+ }, []);
+}
diff --git a/lib/hooks/configurators/use_get_storage_perms.dart b/lib/hooks/configurators/use_get_storage_perms.dart
index 3fcb369b..f860aaa7 100644
--- a/lib/hooks/configurators/use_get_storage_perms.dart
+++ b/lib/hooks/configurators/use_get_storage_perms.dart
@@ -1,35 +1,46 @@
import 'package:device_info_plus/device_info_plus.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
+
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:permission_handler/permission_handler.dart';
-import 'package:spotube/components/library/user_local_tracks.dart';
import 'package:spotube/hooks/utils/use_async_effect.dart';
+import 'package:spotube/provider/local_tracks/local_tracks_provider.dart';
+import 'package:spotube/utils/platform.dart';
void useGetStoragePermissions(WidgetRef ref) {
- final isMounted = useIsMounted();
+ final context = useContext();
useAsyncEffect(
() async {
- if (!DesktopTools.platform.isMobile) return;
+ if (kIsAndroid) {
+ final androidInfo = await DeviceInfoPlugin().androidInfo;
- final androidInfo = await DeviceInfoPlugin().androidInfo;
+ final hasNoStoragePerm = androidInfo.version.sdkInt < 33 &&
+ !await Permission.storage.isGranted &&
+ !await Permission.storage.isLimited;
- final hasNoStoragePerm = androidInfo.version.sdkInt < 33 &&
- !await Permission.storage.isGranted &&
- !await Permission.storage.isLimited;
+ final hasNoAudioPerm = androidInfo.version.sdkInt >= 33 &&
+ !await Permission.audio.isGranted &&
+ !await Permission.audio.isLimited;
- final hasNoAudioPerm = androidInfo.version.sdkInt >= 33 &&
- !await Permission.audio.isGranted &&
- !await Permission.audio.isLimited;
-
- if (hasNoStoragePerm) {
- await Permission.storage.request();
- if (isMounted()) ref.refresh(localTracksProvider);
+ if (hasNoStoragePerm) {
+ await Permission.storage.request();
+ if (context.mounted) ref.invalidate(localTracksProvider);
+ }
+ if (hasNoAudioPerm) {
+ await Permission.audio.request();
+ if (context.mounted) ref.invalidate(localTracksProvider);
+ }
}
- if (hasNoAudioPerm) {
- await Permission.audio.request();
- if (isMounted()) ref.refresh(localTracksProvider);
+
+ if (kIsIOS) {
+ final hasStoragePerm = await Permission.storage.isGranted ||
+ await Permission.storage.isLimited;
+
+ if (!hasStoragePerm) {
+ await Permission.storage.request();
+ if (context.mounted) ref.invalidate(localTracksProvider);
+ }
}
},
null,
diff --git a/lib/hooks/configurators/use_has_touch.dart b/lib/hooks/configurators/use_has_touch.dart
new file mode 100644
index 00000000..75353f27
--- /dev/null
+++ b/lib/hooks/configurators/use_has_touch.dart
@@ -0,0 +1,27 @@
+import 'package:flutter/gestures.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:spotube/utils/platform.dart';
+
+bool useHasTouch() {
+ final hasTouch = useState(kIsMobile);
+
+ useEffect(() {
+ void globalRoute(PointerEvent event) {
+ if (hasTouch.value) return;
+ hasTouch.value = event.kind == PointerDeviceKind.touch ||
+ event.kind == PointerDeviceKind.stylus ||
+ event.kind == PointerDeviceKind.invertedStylus;
+ }
+
+ WidgetsBinding.instance.addPostFrameCallback((_) {
+ GestureBinding.instance.pointerRouter.addGlobalRoute(globalRoute);
+ });
+
+ return () {
+ GestureBinding.instance.pointerRouter.removeGlobalRoute(globalRoute);
+ };
+ }, []);
+
+ return hasTouch.value;
+}
diff --git a/lib/hooks/configurators/use_init_sys_tray.dart b/lib/hooks/configurators/use_init_sys_tray.dart
deleted file mode 100644
index 8080bea6..00000000
--- a/lib/hooks/configurators/use_init_sys_tray.dart
+++ /dev/null
@@ -1,128 +0,0 @@
-import 'dart:io';
-
-import 'package:flutter/material.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotube/collections/intents.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist.dart';
-import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
-
-void useInitSysTray(WidgetRef ref) {
- final context = useContext();
- final systemTray = useRef(null);
-
- final initializeMenu = useCallback(() async {
- systemTray.value?.destroy();
- final playlist = ref.read(ProxyPlaylistNotifier.provider);
- final playlistQueue = ref.read(ProxyPlaylistNotifier.notifier);
- final preferences = ref.read(userPreferencesProvider);
- if (!preferences.showSystemTrayIcon) {
- await systemTray.value?.destroy();
- systemTray.value = null;
- return;
- }
- final enabled = !playlist.isFetching;
- systemTray.value = await DesktopTools.createSystemTrayMenu(
- title: DesktopTools.platform.isWindows ? "Spotube" : "",
- iconPath: "assets/spotube-logo.png",
- windowsIconPath: "assets/spotube-logo.ico",
- items: [
- MenuItemLabel(
- label: "Show/Hide",
- name: "show-hide",
- onClicked: (item) async {
- if (await DesktopTools.window.isVisible()) {
- await DesktopTools.window.hide();
- } else {
- await DesktopTools.window.show();
- }
- },
- ),
- MenuSeparator(),
- MenuItemLabel(
- label: "Play/Pause",
- name: "play-pause",
- enabled: enabled,
- onClicked: (_) async {
- Actions.maybeInvoke(
- context, PlayPauseIntent(ref)) ??
- PlayPauseAction().invoke(PlayPauseIntent(ref));
- },
- ),
- MenuItemLabel(
- label: "Next",
- name: "next",
- enabled: enabled && (playlist.tracks.length) > 1,
- onClicked: (p0) async {
- await playlistQueue.next();
- },
- ),
- MenuItemLabel(
- label: "Previous",
- name: "previous",
- enabled: enabled && (playlist.tracks.length) > 1,
- onClicked: (p0) async {
- await playlistQueue.previous();
- },
- ),
- MenuSeparator(),
- MenuItemLabel(
- label: "Quit",
- name: "quit",
- onClicked: (item) async {
- exit(0);
- },
- ),
- ],
- onEvent: (event, tray) async {
- if (DesktopTools.platform.isWindows) {
- switch (event) {
- case SystemTrayEvent.click:
- await DesktopTools.window.show();
- break;
- case SystemTrayEvent.rightClick:
- await tray.popUpContextMenu();
- break;
- default:
- }
- } else {
- switch (event) {
- case SystemTrayEvent.rightClick:
- await DesktopTools.window.show();
- break;
- case SystemTrayEvent.click:
- await tray.popUpContextMenu();
- break;
- default:
- }
- }
- },
- );
- }, [ref]);
-
- useReassemble(initializeMenu);
-
- ref.listen(
- ProxyPlaylistNotifier.provider,
- (previous, next) {
- initializeMenu();
- },
- );
- ref.listen(
- userPreferencesProvider.select((s) => s.showSystemTrayIcon),
- (previous, next) {
- initializeMenu();
- },
- );
-
- useEffect(() {
- WidgetsBinding.instance.addPostFrameCallback((_) {
- initializeMenu();
- });
- return () async {
- await systemTray.value?.destroy();
- };
- }, [initializeMenu]);
-}
diff --git a/lib/hooks/configurators/use_update_checker.dart b/lib/hooks/configurators/use_update_checker.dart
deleted file mode 100644
index 1a6a5be5..00000000
--- a/lib/hooks/configurators/use_update_checker.dart
+++ /dev/null
@@ -1,100 +0,0 @@
-import 'dart:async';
-import 'dart:convert';
-
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:flutter_riverpod/flutter_riverpod.dart';
-import 'package:http/http.dart' as http;
-import 'package:spotube/collections/env.dart';
-
-import 'package:spotube/components/shared/links/anchor_button.dart';
-import 'package:spotube/hooks/controllers/use_package_info.dart';
-import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
-import 'package:url_launcher/url_launcher_string.dart';
-import 'package:version/version.dart';
-
-void useUpdateChecker(WidgetRef ref) {
- final isCheckUpdateEnabled =
- ref.watch(userPreferencesProvider.select((s) => s.checkUpdate));
- final packageInfo = usePackageInfo(
- appName: 'Spotube',
- packageName: 'spotube',
- );
- final Future> Function() checkUpdate = useCallback(
- () async {
- final value = await http.get(
- Uri.parse(
- "https://api.github.com/repos/KRTirtho/spotube/releases/latest"),
- );
- final tagName =
- (jsonDecode(value.body)["tag_name"] as String).replaceAll("v", "");
- final currentVersion = packageInfo.version == "Unknown"
- ? null
- : Version.parse(packageInfo.version);
- final latestVersion =
- tagName == "nightly" ? null : Version.parse(tagName);
- return [currentVersion, latestVersion];
- },
- [packageInfo.version],
- );
-
- final context = useContext();
-
- download(String url) => launchUrlString(
- url,
- mode: LaunchMode.externalApplication,
- );
-
- useEffect(() {
- if (!Env.enableUpdateChecker) return;
- if (!isCheckUpdateEnabled) return null;
- checkUpdate().then((value) {
- final currentVersion = value.first;
- final latestVersion = value.last;
- if (currentVersion == null ||
- latestVersion == null ||
- (latestVersion.isPreRelease && !currentVersion.isPreRelease) ||
- (!latestVersion.isPreRelease && currentVersion.isPreRelease)) return;
- if (latestVersion <= currentVersion) return;
- showDialog(
- context: context,
- barrierDismissible: true,
- barrierColor: Colors.black26,
- builder: (context) {
- const url =
- "https://spotube.krtirtho.dev/other-downloads/stable-downloads";
- return AlertDialog(
- title: const Text("Spotube has an update"),
- actions: [
- FilledButton(
- child: const Text("Download Now"),
- onPressed: () => download(url),
- ),
- ],
- content: Column(
- mainAxisSize: MainAxisSize.min,
- children: [
- Text("Spotube v${value.last} has been released"),
- Row(
- mainAxisAlignment: MainAxisAlignment.center,
- children: [
- const Text("Read the latest "),
- AnchorButton(
- "release notes",
- style: const TextStyle(color: Colors.blue),
- onTap: () => launchUrlString(
- url,
- mode: LaunchMode.externalApplication,
- ),
- ),
- ],
- ),
- ],
- ),
- );
- },
- );
- });
- return null;
- }, [packageInfo, isCheckUpdateEnabled]);
-}
diff --git a/lib/hooks/configurators/use_window_listener.dart b/lib/hooks/configurators/use_window_listener.dart
index b91ad413..5977ea8e 100644
--- a/lib/hooks/configurators/use_window_listener.dart
+++ b/lib/hooks/configurators/use_window_listener.dart
@@ -1,6 +1,8 @@
import 'package:flutter/widgets.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.dart';
+
import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:spotube/utils/platform.dart';
+import 'package:window_manager/window_manager.dart';
class CallbackWindowListener implements WindowListener {
final VoidCallback? _onWindowClose;
@@ -154,6 +156,8 @@ void useWindowListener({
VoidCallback? onWindowEvent,
}) {
useEffect(() {
+ if (!kIsDesktop) return null;
+
final listener = CallbackWindowListener(
onWindowClose: onWindowClose,
onWindowFocus: onWindowFocus,
@@ -172,9 +176,9 @@ void useWindowListener({
onWindowUndocked: onWindowUndocked,
onWindowEvent: onWindowEvent,
);
- DesktopTools.window.addListener(listener);
+ windowManager.addListener(listener);
return () {
- DesktopTools.window.removeListener(listener);
+ windowManager.removeListener(listener);
};
}, [
onWindowClose,
diff --git a/lib/hooks/controllers/use_auto_scroll_controller.dart b/lib/hooks/controllers/use_auto_scroll_controller.dart
index 8edfb041..0c7119e4 100644
--- a/lib/hooks/controllers/use_auto_scroll_controller.dart
+++ b/lib/hooks/controllers/use_auto_scroll_controller.dart
@@ -39,8 +39,8 @@ class _AutoScrollControllerHook extends Hook {
this.copyTagsFrom,
this.suggestedRowHeight,
this.debugLabel,
- List? keys,
- }) : super(keys: keys);
+ super.keys,
+ });
final double initialScrollOffset;
final bool keepScrollOffset;
diff --git a/lib/hooks/controllers/use_package_info.dart b/lib/hooks/controllers/use_package_info.dart
index 9b142ced..b3c05665 100644
--- a/lib/hooks/controllers/use_package_info.dart
+++ b/lib/hooks/controllers/use_package_info.dart
@@ -44,8 +44,8 @@ class _PackageInfoHook extends Hook {
required this.version,
required this.buildNumber,
this.buildSignature = '',
- List? keys,
- }) : super(keys: keys);
+ super.keys,
+ });
@override
HookState> createState() =>
diff --git a/lib/hooks/controllers/use_sidebarx_controller.dart b/lib/hooks/controllers/use_sidebarx_controller.dart
index 5af921b7..a14c3305 100644
--- a/lib/hooks/controllers/use_sidebarx_controller.dart
+++ b/lib/hooks/controllers/use_sidebarx_controller.dart
@@ -24,8 +24,8 @@ class _SidebarXControllerHook extends Hook {
const _SidebarXControllerHook({
required this.selectedIndex,
this.extended,
- List? keys,
- }) : super(keys: keys);
+ super.keys,
+ });
final int selectedIndex;
final bool? extended;
diff --git a/lib/hooks/spotify/use_spotify_infinite_query.dart b/lib/hooks/spotify/use_spotify_infinite_query.dart
deleted file mode 100644
index 2063b083..00000000
--- a/lib/hooks/spotify/use_spotify_infinite_query.dart
+++ /dev/null
@@ -1,53 +0,0 @@
-import 'dart:async';
-
-import 'package:fl_query/fl_query.dart';
-import 'package:fl_query_hooks/fl_query_hooks.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotify/spotify.dart';
-import 'package:spotube/provider/spotify_provider.dart';
-
-InfiniteQuery
- useSpotifyInfiniteQuery(
- String queryKey,
- FutureOr Function(PageType page, SpotifyApi spotify) queryFn, {
- required WidgetRef ref,
- required InfiniteQueryNextPage nextPage,
- required PageType initialPage,
- RetryConfig? retryConfig,
- RefreshConfig? refreshConfig,
- JsonConfig? jsonConfig,
- ValueChanged>? onData,
- ValueChanged>? onError,
- bool enabled = true,
- List? keys,
-}) {
- final spotify = ref.watch(spotifyProvider);
- final query = useInfiniteQuery(
- queryKey,
- (page) => queryFn(page, spotify),
- nextPage: nextPage,
- initialPage: initialPage,
- retryConfig: retryConfig,
- refreshConfig: refreshConfig,
- jsonConfig: jsonConfig,
- onData: onData,
- onError: onError,
- enabled: enabled,
- keys: keys,
- );
-
- useEffect(() {
- return ref.listenManual(
- spotifyProvider,
- (previous, next) {
- if (previous != next) {
- query.refreshAll();
- }
- },
- ).close;
- }, [query]);
-
- return query;
-}
diff --git a/lib/hooks/spotify/use_spotify_mutation.dart b/lib/hooks/spotify/use_spotify_mutation.dart
deleted file mode 100644
index 637f778f..00000000
--- a/lib/hooks/spotify/use_spotify_mutation.dart
+++ /dev/null
@@ -1,36 +0,0 @@
-import 'package:fl_query/fl_query.dart';
-import 'package:fl_query_hooks/fl_query_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotify/spotify.dart';
-import 'package:spotube/provider/spotify_provider.dart';
-
-Mutation
- useSpotifyMutation(
- String mutationKey,
- Future Function(VariablesType variables, SpotifyApi spotify)
- mutationFn, {
- required WidgetRef ref,
- RetryConfig? retryConfig,
- MutationOnDataFn? onData,
- MutationOnErrorFn? onError,
- MutationOnMutationFn? onMutate,
- List? refreshQueries,
- List? refreshInfiniteQueries,
- List? keys,
-}) {
- final spotify = ref.watch(spotifyProvider);
- final mutation =
- useMutation(
- mutationKey,
- (variables) => mutationFn(variables, spotify),
- retryConfig: retryConfig,
- onData: onData,
- onError: onError,
- onMutate: onMutate,
- refreshQueries: refreshQueries,
- refreshInfiniteQueries: refreshInfiniteQueries,
- keys: keys,
- );
-
- return mutation;
-}
diff --git a/lib/hooks/spotify/use_spotify_query.dart b/lib/hooks/spotify/use_spotify_query.dart
deleted file mode 100644
index 0c79de91..00000000
--- a/lib/hooks/spotify/use_spotify_query.dart
+++ /dev/null
@@ -1,52 +0,0 @@
-import 'dart:async';
-
-import 'package:fl_query/fl_query.dart';
-import 'package:fl_query_hooks/fl_query_hooks.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:spotify/spotify.dart';
-import 'package:spotube/provider/spotify_provider.dart';
-
-typedef SpotifyQueryFn = FutureOr Function(
- SpotifyApi spotify);
-
-Query useSpotifyQuery(
- final String queryKey,
- final SpotifyQueryFn queryFn, {
- required WidgetRef ref,
- final DataType? initial,
- final RetryConfig? retryConfig,
- final RefreshConfig? refreshConfig,
- final JsonConfig? jsonConfig,
- final ValueChanged? onData,
- final ValueChanged? onError,
- final bool enabled = true,
-}) {
- final spotify = ref.watch(spotifyProvider);
-
- final query = useQuery(
- queryKey,
- () => queryFn(spotify),
- initial: initial,
- retryConfig: retryConfig,
- refreshConfig: refreshConfig,
- jsonConfig: jsonConfig,
- onData: onData,
- onError: onError,
- enabled: enabled,
- );
-
- useEffect(() {
- return ref.listenManual(
- spotifyProvider,
- (previous, next) {
- if (previous != next) {
- query.refresh();
- }
- },
- ).close;
- }, [query]);
-
- return query;
-}
diff --git a/lib/hooks/utils/use_custom_status_bar_color.dart b/lib/hooks/utils/use_custom_status_bar_color.dart
index d1266fe2..7c5c7b27 100644
--- a/lib/hooks/utils/use_custom_status_bar_color.dart
+++ b/lib/hooks/utils/use_custom_status_bar_color.dart
@@ -19,11 +19,13 @@ void useCustomStatusBarColor(
),
);
+ // ignore: invalid_use_of_visible_for_testing_member
final statusBarColor = SystemChrome.latestStyle?.statusBarColor;
useEffect(() {
WidgetsBinding.instance.addPostFrameCallback((_) {
if (automaticSystemUiAdjustment != null) {
+ // ignore: deprecated_member_use
WidgetsBinding.instance.renderView.automaticSystemUiAdjustment =
automaticSystemUiAdjustment;
}
@@ -43,6 +45,7 @@ void useCustomStatusBarColor(
});
return () {
if (automaticSystemUiAdjustment != null) {
+ // ignore: deprecated_member_use
WidgetsBinding.instance.renderView.automaticSystemUiAdjustment = false;
}
};
diff --git a/lib/hooks/utils/use_force_update.dart b/lib/hooks/utils/use_force_update.dart
index 74151a65..268f0f04 100644
--- a/lib/hooks/utils/use_force_update.dart
+++ b/lib/hooks/utils/use_force_update.dart
@@ -2,5 +2,6 @@ import 'package:flutter_hooks/flutter_hooks.dart';
void Function() useForceUpdate() {
final state = useState(null);
+ // ignore: invalid_use_of_protected_member, invalid_use_of_visible_for_testing_member
return () => state.notifyListeners();
}
diff --git a/lib/hooks/utils/use_palette_color.dart b/lib/hooks/utils/use_palette_color.dart
index 9269edd7..64994d2b 100644
--- a/lib/hooks/utils/use_palette_color.dart
+++ b/lib/hooks/utils/use_palette_color.dart
@@ -2,7 +2,7 @@ import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:palette_generator/palette_generator.dart';
-import 'package:spotube/components/shared/image/universal_image.dart';
+import 'package:spotube/components/image/universal_image.dart';
final _paletteColorState = StateProvider(
(ref) {
@@ -14,7 +14,6 @@ PaletteColor usePaletteColor(String imageUrl, WidgetRef ref) {
final context = useContext();
final theme = Theme.of(context);
final paletteColor = ref.watch(_paletteColorState);
- final mounted = useIsMounted();
useEffect(() {
WidgetsBinding.instance.addPostFrameCallback((timeStamp) async {
@@ -25,7 +24,7 @@ PaletteColor usePaletteColor(String imageUrl, WidgetRef ref) {
width: 50,
),
);
- if (!mounted()) return;
+ if (!context.mounted) return;
final color = theme.brightness == Brightness.light
? palette.lightMutedColor ?? palette.lightVibrantColor
: palette.darkMutedColor ?? palette.darkVibrantColor;
@@ -41,7 +40,7 @@ PaletteColor usePaletteColor(String imageUrl, WidgetRef ref) {
PaletteGenerator usePaletteGenerator(String imageUrl) {
final palette = useState(PaletteGenerator.fromColors([]));
- final mounted = useIsMounted();
+ final context = useContext();
useEffect(() {
WidgetsBinding.instance.addPostFrameCallback((timeStamp) async {
@@ -52,7 +51,7 @@ PaletteGenerator usePaletteGenerator(String imageUrl) {
width: 50,
),
);
- if (!mounted()) return;
+ if (!context.mounted) return;
palette.value = newPalette;
});
diff --git a/lib/l10n/app_ar.arb b/lib/l10n/app_ar.arb
index eebede99..141e10f0 100644
--- a/lib/l10n/app_ar.arb
+++ b/lib/l10n/app_ar.arb
@@ -286,5 +286,106 @@
"step_3_steps": "انسخ قيمة الكوكي \"sp_dc\"",
"step_4_steps": "الصق قيمة \"sp_dc\" المنسوخة",
"friends": "أصدقاء",
- "no_lyrics_available": "عذرًا، تعذر العثور على كلمات الأغنية لهذه العنصر"
+ "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": "تشعر بالحرية؟ نفس يوتيوب ولكن أكثر حرية.",
+ "jiosaavn_source_description": "الأفضل لمنطقة جنوب آسيا.",
+ "highest_quality": "أعلى جودة: {quality}",
+ "select_audio_source": "اختر مصدر الصوت",
+ "endless_playback_description": "إلحاق الأغاني الجديدة تلقائيًا\nإلى نهاية قائمة التشغيل",
+ "choose_your_region": "اختر منطقتك",
+ "choose_your_region_description": "سيساعدك هذا في عرض المحتوى المناسب\nلموقعك.",
+ "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": "يوجد تحديث لسبوتيوب",
+ "download_now": "تحميل الآن",
+ "nightly_version": "تم إصدار سبوتيوب الليلي {nightlyBuildNum}",
+ "release_version": "تم إصدار سبوتيوب 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": "آخر سنتين",
+ "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": "*هذا محسوب بناءً على الدفع لكل بث من سبوتيفاي\nبقيمة 0.003 إلى 0.005 دولار. هذا حساب افتراضي\nلإعطاء المستخدم فكرة عن المبلغ الذي\nكان سيدفعه للفنانين إذا كانوا قد استمعوا\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": "المنصة غير مدعومة"
}
\ No newline at end of file
diff --git a/lib/l10n/app_bn.arb b/lib/l10n/app_bn.arb
index 2711f8d2..ae088b45 100644
--- a/lib/l10n/app_bn.arb
+++ b/lib/l10n/app_bn.arb
@@ -286,5 +286,106 @@
"step_3_steps": "কুকি \"sp_dc\" এর মানটি কপি করুন",
"step_4_steps": "কপি করা \"sp_dc\" মানটি পেস্ট করুন",
"friends": "বন্ধু",
- "no_lyrics_available": "দুঃখিত, এই ট্র্যাকের জন্য কথা খুঁজে পাওয়া গেলনা"
+ "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": "মন খারাপ? ইউটিউবের মতো আবার ফ্রি।",
+ "jiosaavn_source_description": "দক্ষিণ এশিয়ান অঞ্চলের জন্য সেরা।",
+ "highest_quality": "সর্বোচ্চ গুণগতি: {quality}",
+ "select_audio_source": "অডিও উৎস নির্বাচন করুন",
+ "endless_playback_description": "নতুন গান নিজে নিজে প্লেলিস্টের শেষে\nসংযুক্ত করুন",
+ "choose_your_region": "আপনার অঞ্চল নির্বাচন করুন",
+ "choose_your_region_description": "এটি স্পটুবে আপনাকে আপনার অবস্থানের জন্য ঠিক কন্টেন্ট দেখানোর সাহায্য করবে।",
+ "choose_your_language": "আপনার ভাষা নির্বাচন করুন",
+ "help_project_grow": "এই প্রকল্পের বৃদ্ধি করুন",
+ "help_project_grow_description": "স্পটুব একটি ওপেন সোর্স প্রকল্প। আপনি প্রকল্পে অবদান রাখেন, বাগ রিপোর্ট করেন, বা নতুন বৈশিষ্ট্যগুলি সুপারিশ করেন।",
+ "contribute_on_github": "গিটহাবে অবদান রাখুন",
+ "donate_on_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": "স্পটিউবে একটি আপডেট আছে",
+ "download_now": "এখনই ডাউনলোড করুন",
+ "nightly_version": "স্পটিউব নাইটলি {nightlyBuildNum} প্রকাশিত হয়েছে",
+ "release_version": "স্পটিউব 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": "গত ৬ মাস",
+ "this_year": "এই বছর",
+ "last_2_years": "গত ২ বছর",
+ "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": "*এটি স্পোটিফাইয়ের প্রতি স্ট্রিম\n$0.003 থেকে $0.005 পেআউটের ভিত্তিতে গণনা করা হয়েছে। এটি একটি ধারণাগত\nগণনা ব্যবহারকারীদেরকে জানাতে দেয় যে কত টাকা\nতারা শিল্পীদের দিতো যদি তারা স্পোটিফাইতে\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_not_found_description": "আপনার ডিভাইসে কোনো ওয়েবভিউ রানটাইম ইনস্টল করা নেই।\nযদি ইনস্টল থাকে, তা নিশ্চিত করুন যে এটি environment PATH এ রয়েছে\n\nইনস্টল করার পর, অ্যাপটি পুনরায় চালু করুন",
+ "unsupported_platform": "সমর্থিত প্ল্যাটফর্ম নয়"
}
\ No newline at end of file
diff --git a/lib/l10n/app_ca.arb b/lib/l10n/app_ca.arb
index f46cfae4..58805e62 100644
--- a/lib/l10n/app_ca.arb
+++ b/lib/l10n/app_ca.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Copia el valor de la cookie \"sp_dc\"",
"step_4_steps": "Pega el valor copiado de \"sp_dc\"",
"friends": "Amics",
- "no_lyrics_available": "Ho sentim, no es poden trobar les lletres d'aquesta pista"
+ "no_lyrics_available": "Ho sentim, no es poden trobar les lletres d'aquesta pista",
+ "sort_duration": "Ordenar per Durada",
+ "start_a_radio": "Inicia una ràdio",
+ "how_to_start_radio": "Com vols començar la ràdio?",
+ "replace_queue_question": "Voleu substituir la cua actual o afegir-hi?",
+ "endless_playback": "Reproducció infinita",
+ "delete_playlist": "Suprimeix la llista de reproducció",
+ "delete_playlist_confirmation": "Esteu segur que voleu suprimir aquesta llista de reproducció?",
+ "local_tracks": "Pistes locals",
+ "song_link": "Enllaç de la cançó",
+ "skip_this_nonsense": "Omet aquesta tonteria",
+ "freedom_of_music": "“Llibertat de la música”",
+ "freedom_of_music_palm": "“Llibertat de la música a la palma de la mà”",
+ "get_started": "Comencem",
+ "youtube_source_description": "Recomanat i funciona millor.",
+ "piped_source_description": "Et sents lliure? El mateix que YouTube però més lliure.",
+ "jiosaavn_source_description": "El millor per a la regió del sud d'Àsia.",
+ "highest_quality": "Qualitat més alta: {quality}",
+ "select_audio_source": "Seleccioneu la font d'àudio",
+ "endless_playback_description": "Afegiu automàticament noves cançons\nal final de la cua",
+ "choose_your_region": "Trieu la vostra regió",
+ "choose_your_region_description": "Això ajudarà a Spotube a mostrar-vos el contingut adequat\nper a la vostra ubicació.",
+ "choose_your_language": "Trieu el vostre idioma",
+ "help_project_grow": "Ajuda a fer créixer aquest projecte",
+ "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.",
+ "contribute_on_github": "Contribueix a GitHub",
+ "donate_on_open_collective": "Fes una donació a Open Collective",
+ "browse_anonymously": "Navega de manera anònima",
+ "enable_connect": "Habilita la connexió",
+ "enable_connect_description": "Controla Spotube des d'altres dispositius",
+ "devices": "Dispositius",
+ "select": "Selecciona",
+ "connect_client_alert": "Estàs sent controlat per {client}",
+ "this_device": "Aquest dispositiu",
+ "remote": "Remot",
+ "local_library": "Biblioteca local",
+ "add_library_location": "Afegeix a la biblioteca",
+ "remove_library_location": "Elimina de la biblioteca",
+ "local_tab": "Local",
+ "stats": "Estadístiques",
+ "and_n_more": "i {count} més",
+ "recently_played": "Reproduït recentment",
+ "browse_more": "Navega més",
+ "no_title": "Sense títol",
+ "not_playing": "No s'està reproduint",
+ "epic_failure": "Fracàs èpic!",
+ "added_num_tracks_to_queue": "Afegit {tracks_length} pistes a la cua",
+ "spotube_has_an_update": "Spotube té una actualització",
+ "download_now": "Descarregar ara",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} ha estat publicat",
+ "release_version": "Spotube v{version} ha estat publicat",
+ "read_the_latest": "Llegeix el més recent",
+ "release_notes": "notes de la versió",
+ "pick_color_scheme": "Tria l'esquema de colors",
+ "save": "Desar",
+ "choose_the_device": "Tria el dispositiu:",
+ "multiple_device_connected": "Hi ha diversos dispositius connectats.\nTria el dispositiu on vols realitzar aquesta acció",
+ "nothing_found": "No s'ha trobat res",
+ "the_box_is_empty": "La caixa està buida",
+ "top_artists": "Millors artistes",
+ "top_albums": "Millors àlbums",
+ "this_week": "Aquesta setmana",
+ "this_month": "Aquest mes",
+ "last_6_months": "Últims 6 mesos",
+ "this_year": "Aquest any",
+ "last_2_years": "Últims 2 anys",
+ "all_time": "Tots els temps",
+ "powered_by_provider": "Funciona amb {providerName}",
+ "email": "Correu electrònic",
+ "profile_followers": "Seguidors",
+ "birthday": "Aniversari",
+ "subscription": "Subscripció",
+ "not_born": "No ha nascut",
+ "hacker": "Hacker",
+ "profile": "Perfil",
+ "no_name": "Sense nom",
+ "edit": "Editar",
+ "user_profile": "Perfil d'usuari",
+ "count_plays": "{count} reproduccions",
+ "streaming_fees_hypothetical": "Comissions de streaming (hipotètic)",
+ "minutes_listened": "minuts escoltats",
+ "streamed_songs": "cançons reproduïdes",
+ "count_streams": "{count} reproduccions",
+ "owned_by_you": "De la teva propietat",
+ "copied_shareurl_to_clipboard": "S'ha copiat {shareUrl} al porta-retalls",
+ "spotify_hipotetical_calculation": "*Això es calcula basant-se en els\npagaments per reproducció de Spotify de $0.003 a $0.005.\nAquest és un càlcul hipotètic per\ndonar als usuaris una idea de quant\nhaurien pagat als artistes si haguessin escoltat\nla seva cançó a Spotify.",
+ "count_mins": "{minutes} minuts",
+ "summary_minutes": "minuts",
+ "summary_listened_to_music": "has escoltat música",
+ "summary_songs": "cançons",
+ "summary_streamed_overall": "reproduït en general",
+ "summary_owed_to_artists": "degut als artistes\nAquest mes",
+ "summary_artists": "artistes",
+ "summary_music_reached_you": "La música t'ha arribat",
+ "summary_full_albums": "Àlbums complets",
+ "summary_got_your_love": "ha aconseguit el teu amor",
+ "summary_playlists": "llistes de reproducció",
+ "summary_were_on_repeat": "estaven en repetició",
+ "total_money": "total {money}",
+ "webview_not_found": "No s'ha trobat el Webview",
+ "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ó",
+ "unsupported_platform": "Plataforma no compatible"
}
\ No newline at end of file
diff --git a/lib/l10n/app_cs.arb b/lib/l10n/app_cs.arb
new file mode 100644
index 00000000..99ee0962
--- /dev/null
+++ b/lib/l10n/app_cs.arb
@@ -0,0 +1,391 @@
+{
+ "guest": "Host",
+ "browse": "Procházet",
+ "search": "Hledat",
+ "library": "Knihovna",
+ "lyrics": "Texty",
+ "settings": "Nastavení",
+ "genre_categories_filter": "Filtrovat kategorie nebo žánry...",
+ "genre": "Žánr",
+ "personalized": "Personalizované",
+ "featured": "Doporučené",
+ "new_releases": "Nově vydané",
+ "songs": "Skladby",
+ "playing_track": "Hraje {track}",
+ "queue_clear_alert": "Toto vymaže aktuální frontu. {track_length} skladeb bude odstraněno\nChcete pokračovat?",
+ "load_more": "Načíst více",
+ "playlists": "Playlisty",
+ "artists": "Umělci",
+ "albums": "Alba",
+ "tracks": "Skladby",
+ "downloads": "Stahování",
+ "filter_playlists": "Filtrovat playlisty...",
+ "liked_tracks": "Oblíbené skladby",
+ "liked_tracks_description": "Všechny vaše oblíbené skladby",
+ "create_playlist": "Vytvořit playlist",
+ "create_a_playlist": "Vytvořit playlist",
+ "update_playlist": "Aktualizovat playlist",
+ "create": "Vytvořit",
+ "cancel": "Zrušit",
+ "update": "Aktualizovat",
+ "playlist_name": "Název playlistu",
+ "name_of_playlist": "Název playlistu",
+ "description": "Popis",
+ "public": "Veřejné",
+ "collaborative": "Společný",
+ "search_local_tracks": "Hledat místní skladby...",
+ "play": "Přehrát",
+ "delete": "Smazat",
+ "none": "Žádné",
+ "sort_a_z": "Seřadit od A-Z",
+ "sort_z_a": "Seřadit od Z-A",
+ "sort_artist": "Seřadit podle umělce",
+ "sort_album": "Seřadit podle alba",
+ "sort_duration": "Seřadit podle délky",
+ "sort_tracks": "Seřadit skladby",
+ "currently_downloading": "Právě se stahuje ({tracks_length})",
+ "cancel_all": "Zrušit vše",
+ "filter_artist": "Filtrovat umělce...",
+ "followers": "{followers} Sledující",
+ "add_artist_to_blacklist": "Přidat umělce na černou listinu",
+ "top_tracks": "Top skladby",
+ "fans_also_like": "Fanoušci mají také rádi",
+ "loading": "Načítání...",
+ "artist": "Umělec",
+ "blacklisted": "Na černé listině",
+ "following": "Sleduje",
+ "follow": "Sledovat",
+ "artist_url_copied": "URL umělce zkopírována do schránky",
+ "added_to_queue": "Přidáno {tracks} skladeb do fronty",
+ "filter_albums": "Filtrovat alba...",
+ "synced": "Synchronizováno",
+ "plain": "Jednoduché",
+ "shuffle": "Zamíchat",
+ "search_tracks": "Hledat skladby...",
+ "released": "Vydáno",
+ "error": "Chyba {error}",
+ "title": "Název",
+ "time": "Čas",
+ "more_actions": "Více akcí",
+ "download_count": "Stáhnout ({count})",
+ "add_count_to_playlist": "Přidat ({count}) do playlistu",
+ "add_count_to_queue": "Přidat ({count}) do fronty",
+ "play_count_next": "Přehrát ({count}) dalších",
+ "album": "Album",
+ "copied_to_clipboard": "Zkopírováno {data} do schránky",
+ "add_to_following_playlists": "Přidat {track} do následujících playlistů",
+ "add": "Přidat",
+ "added_track_to_queue": "Přidána skladba {track} do fronty",
+ "add_to_queue": "Přidat do fronty",
+ "track_will_play_next": "{track} se přehraje jako další",
+ "play_next": "Přehrát další",
+ "removed_track_from_queue": "Odstraněna skladba {track} z fronty",
+ "remove_from_queue": "Odstranit z fronty",
+ "remove_from_favorites": "Odstranit z oblíbených",
+ "save_as_favorite": "Uložit jako oblíbené",
+ "add_to_playlist": "Přidat do playlistu",
+ "remove_from_playlist": "Odstranit z playlistu",
+ "add_to_blacklist": "Přidat na černou listinu",
+ "remove_from_blacklist": "Odstranit z černé listiny",
+ "share": "Sdílet",
+ "mini_player": "Mini přehrávač",
+ "slide_to_seek": "Táhněte pro posunutí vpřed nebo vzad",
+ "shuffle_playlist": "Zamíchat playlist",
+ "unshuffle_playlist": "Zrušit zamíchání playlistu",
+ "previous_track": "Předchozí skladba",
+ "next_track": "Další skladba",
+ "pause_playback": "Pozastavit přehrávání",
+ "resume_playback": "Pokračovat v přehrávání",
+ "loop_track": "Opakovat skladbu",
+ "repeat_playlist": "Opakovat playlist",
+ "queue": "Fronta",
+ "alternative_track_sources": "Alternativní zdroje skladeb",
+ "download_track": "Stáhnout skladbu",
+ "tracks_in_queue": "{tracks} skladeb ve frontě",
+ "clear_all": "Vymazat vše",
+ "show_hide_ui_on_hover": "Zobrazit/Skrýt UI při najetí",
+ "always_on_top": "Vždy nahoře",
+ "exit_mini_player": "Zavřít mini přehrávač",
+ "download_location": "Umístění stahování",
+ "account": "Účet",
+ "login_with_spotify": "Přihlásit se pomocí Spotify účtu",
+ "connect_with_spotify": "Připojit k Spotify",
+ "logout": "Odhlásit se",
+ "logout_of_this_account": "Odhlásit se z tohoto účtu",
+ "language_region": "Jazyk a region",
+ "language": "Jazyk",
+ "system_default": "Systém",
+ "market_place_region": "Region",
+ "recommendation_country": "Země pro doporučení",
+ "appearance": "Vzhled",
+ "layout_mode": "Režim rozložení",
+ "override_layout_settings": "Přepsat režim rozložení",
+ "adaptive": "Adaptivní",
+ "compact": "Kompaktní",
+ "extended": "Rozšířený",
+ "theme": "Téma",
+ "dark": "Tmavé",
+ "light": "Světlé",
+ "system": "Systém",
+ "accent_color": "Barva akcentu",
+ "sync_album_color": "Synchronizovat barvu alba",
+ "sync_album_color_description": "Používá dominantní barvu obalu alba jako barvu akcentu",
+ "playback": "Přehrávání",
+ "audio_quality": "Kvalita zvuku",
+ "high": "Vysoká",
+ "low": "Nízká",
+ "pre_download_play": "Předstáhnout a přehrát",
+ "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)",
+ "skip_non_music": "Přeskočit nehudební segmenty (SponsorBlock)",
+ "blacklist_description": "Zakázané skladby a umělci",
+ "wait_for_download_to_finish": "Počkejte, až se dokončí stahování",
+ "desktop": "Desktop",
+ "close_behavior": "Chování při zavření",
+ "close": "Zavřít",
+ "minimize_to_tray": "Minimalizovat do lišty",
+ "show_tray_icon": "Zobrazit ikonu v systémové liště",
+ "about": "O aplikaci",
+ "u_love_spotube": "Víme, že milujete Spotube",
+ "check_for_updates": "Zkontrolovat aktualizace",
+ "about_spotube": "O Spotube",
+ "blacklist": "Černá listina",
+ "please_sponsor": "Sponzorovat/darovat",
+ "spotube_description": "Spotube, rychlý, multiplatformní, bezplatný Spotify klient",
+ "version": "Verze",
+ "build_number": "Číslo sestavení",
+ "founder": "Zakladatel",
+ "repository": "Repozitář",
+ "bug_issues": "Chyby+Problémy",
+ "made_with": "Vytvořeno s ❤️ v Bangladéši🇧🇩",
+ "kingkor_roy_tirtho": "Kingkor Roy Tirtho",
+ "copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
+ "license": "Licence",
+ "add_spotify_credentials": "Přidejte své přihlašovací údaje Spotify a začněte",
+ "credentials_will_not_be_shared_disclaimer": "Nebojte, žádné z vašich údajů nebudou shromažďovány ani s nikým sdíleny",
+ "know_how_to_login": "Nevíte, jak na to?",
+ "follow_step_by_step_guide": "Postupujte podle návodu",
+ "spotify_cookie": "Cookie Spotify {name}",
+ "cookie_name_cookie": "Cookie {name}",
+ "fill_in_all_fields": "Vyplňte prosím všechna pole",
+ "submit": "Odeslat",
+ "exit": "Ukončit",
+ "previous": "Předchozí",
+ "next": "Další",
+ "done": "Hotovo",
+ "step_1": "Krok 1",
+ "first_go_to": "Nejprve jděte na",
+ "login_if_not_logged_in": "a přihlašte se nebo se zaregistrujte, pokud nejste přihlášeni",
+ "step_2": "Krok 2",
+ "step_2_steps": "1. Jakmile jste přihlášeni, stiskněte F12 nebo pravé tlačítko myši > Prozkoumat, abyste otevřeli nástroje pro vývojáře prohlížeče.\n2. Poté přejděte na kartu \"Aplikace\" (Chrome, Edge, Brave atd.) nebo kartu \"Úložiště\" (Firefox, Palemoon atd.)\n3. Přejděte do sekce \"Cookies\" a pak do podsekce \"https://accounts.spotify.com\"",
+ "step_3": "Krok 3",
+ "step_3_steps": "Zkopírujte hodnotu cookie \"sp_dc\"",
+ "success_emoji": "Úspěch🥳",
+ "success_message": "Nyní jste úspěšně přihlášeni pomocí svého Spotify účtu. Dobrá práce, kamaráde!",
+ "step_4": "Krok 4",
+ "step_4_steps": "Vložte zkopírovanou hodnotu \"sp_dc\"",
+ "something_went_wrong": "Něco se pokazilo",
+ "piped_instance": "Instance serveru Piped",
+ "piped_description": "Instance serveru Piped, kterou použít pro hledání skladeb",
+ "piped_warning": "Některé z nich nemusí dobře fungovat. Používejte na vlastní riziko",
+ "generate_playlist": "Vygenerovat playlist",
+ "track_exists": "Skladba {track} již existuje",
+ "replace_downloaded_tracks": "Nahradit všechny stažené skladby",
+ "skip_download_tracks": "Přeskočit stahování všech stažených skladeb",
+ "do_you_want_to_replace": "Chcete nahradit existující skladbu??",
+ "replace": "Nahradit",
+ "skip": "Přeskočit",
+ "select_up_to_count_type": "Vyberte až {count} {type}",
+ "select_genres": "Vyberte žánry",
+ "add_genres": "Přidat žánry",
+ "country": "Země",
+ "number_of_tracks_generate": "Počet skladeb k vygenerování",
+ "acousticness": "Akustičnost",
+ "danceability": "Tanečnost",
+ "energy": "Energie",
+ "instrumentalness": "Instrumentálnost",
+ "liveness": "Živost",
+ "loudness": "Hlasitost",
+ "speechiness": "Mluvnost",
+ "valence": "Valence",
+ "popularity": "Popularita",
+ "key": "Klíč",
+ "duration": "Délka (s)",
+ "tempo": "Tempo (BPM)",
+ "mode": "Režim",
+ "time_signature": "Udání taktu",
+ "short": "Krátký",
+ "medium": "Střední",
+ "long": "Dlouhý",
+ "min": "Min",
+ "max": "Max",
+ "target": "Cíl",
+ "moderate": "Mírný",
+ "deselect_all": "Zrušit výběr",
+ "select_all": "Vybrat vše",
+ "are_you_sure": "Jste si jisti?",
+ "generating_playlist": "Generování vašeho vlastního playlistu...",
+ "selected_count_tracks": "Vybráno {count} skladeb",
+ "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ů",
+ "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",
+ "by_clicking_accept_terms": "Kliknutím na 'přijmout' souhlasíte s následujícími podmínkami:",
+ "download_agreement_1": "Vím, že pirátím hudbu. Jsem špatný",
+ "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í",
+ "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",
+ "decline": "Odmítnout",
+ "accept": "Přijmout",
+ "details": "Podrobnosti",
+ "youtube": "YouTube",
+ "channel": "Kanál",
+ "likes": "Líbí se",
+ "dislikes": "Nelíbí se",
+ "views": "Zobrazení",
+ "streamUrl": "URL streamu",
+ "stop": "Zastavit",
+ "sort_newest": "Seřadit od nejnovějších",
+ "sort_oldest": "Seřadit od nejstarších",
+ "sleep_timer": "Časovač spánku",
+ "mins": "{minutes} Minut",
+ "hours": "{hours} Hodin",
+ "hour": "{hours} Hodina",
+ "custom_hours": "Vlastní hodiny",
+ "logs": "Protokoly",
+ "developers": "Vývojáři",
+ "not_logged_in": "Nejste přihlášeni",
+ "search_mode": "Režim hledání",
+ "audio_source": "Zdroj zvuku",
+ "ok": "Ok",
+ "failed_to_encrypt": "Šifrování selhalo",
+ "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.)",
+ "querying_info": "Získávání informací...",
+ "piped_api_down": "Piped API je mimo provoz",
+ "piped_down_error_instructions": "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",
+ "you_are_offline": "Momentálně jste offline",
+ "connection_restored": "Vaše internetové připojení bylo obnoveno",
+ "use_system_title_bar": "Použít systémové záhlaví okna",
+ "crunching_results": "Zpracovávání výsledků...",
+ "search_to_get_results": "Hledejte pro získání výsledků",
+ "use_amoled_mode": "Úplně černé téma",
+ "pitch_dark_theme": "AMOLED režim",
+ "normalize_audio": "Normalizovat audio",
+ "change_cover": "Změnit obal",
+ "add_cover": "Přidat obal",
+ "restore_defaults": "Obnovit výchozí",
+ "download_music_codec": "Kodek pro stahování",
+ "streaming_music_codec": "Kodek pro streamování",
+ "login_with_lastfm": "Přihlásit se pomocí Last.fm",
+ "connect": "Připojit",
+ "disconnect_lastfm": "Odpojit Last.fm",
+ "disconnect": "Odpojit",
+ "username": "Uživatelské jméno",
+ "password": "Heslo",
+ "login": "Přihlásit se",
+ "login_with_your_lastfm": "Přihlásit se pomocí vašeho Last.fm účtu",
+ "scrobble_to_lastfm": "Scrobble na Last.fm",
+ "go_to_album": "Přejít na album",
+ "discord_rich_presence": "Discord Rich Presence",
+ "browse_all": "Procházet vše",
+ "genres": "Žánry",
+ "explore_genres": "Prozkoumat žánry",
+ "friends": "Přátelé",
+ "no_lyrics_available": "Omlouváme se, není možné najít texty pro tuto skladbu",
+ "start_a_radio": "Vytvořit rádio",
+ "how_to_start_radio": "Jak chcete vytvořit rádio?",
+ "replace_queue_question": "Chcete nahradit aktuální frontu nebo k ní přidat?",
+ "endless_playback": "Nekonečné přehrávání",
+ "delete_playlist": "Smazat playlist",
+ "delete_playlist_confirmation": "Jste si jisti, že chcete smazat tento playlist?",
+ "local_tracks": "Místní skladby",
+ "song_link": "Odkaz na skladbu",
+ "skip_this_nonsense": "Přeskočit tenhle nesmysl",
+ "freedom_of_music": "“Svobodná hudba”",
+ "freedom_of_music_palm": "“Svobodná hudba ve vaší dlani”",
+ "get_started": "Začít",
+ "youtube_source_description": "Doporučeno a funguje nejlépe.",
+ "piped_source_description": "Nechcete být sledováni? Stejné jako YouTube, ale respektuje soukromí.",
+ "jiosaavn_source_description": "Nejlepší pro jihoasijský region.",
+ "highest_quality": "Nejvyšší kvalita: {quality}",
+ "select_audio_source": "Vyberte zdroj zvuku",
+ "endless_playback_description": "Automaticky přidávat nové skladby\nna konec fronty",
+ "choose_your_region": "Vyberte svůj region",
+ "choose_your_region_description": "To pomůže Spotube ukázat vám správný obsah\npro vaši lokalitu.",
+ "choose_your_language": "Vyberte svůj jazyk",
+ "help_project_grow": "Pomozte tomuto projektu růst",
+ "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.",
+ "contribute_on_github": "Přispějte na GitHub",
+ "donate_on_open_collective": "Darujte na Open Collective",
+ "browse_anonymously": "Procházet anonymně",
+ "enable_connect": "Povolit ovládání",
+ "enable_connect_description": "Ovládejte Spotube z jiného zařízení",
+ "devices": "Zařízení",
+ "select": "Vybrat",
+ "connect_client_alert": "Zařízení je ovládáno z {client}",
+ "this_device": "Toto zařízení",
+ "remote": "Ovladač",
+ "local_library": "Místní knihovna",
+ "add_library_location": "Přidat do knihovny",
+ "remove_library_location": "Odebrat z knihovny",
+ "local_tab": "Místní",
+ "stats": "Statistiky",
+ "and_n_more": "a dalších {count}",
+ "recently_played": "Nedávno přehráno",
+ "browse_more": "Procházet více",
+ "no_title": "Bez názvu",
+ "not_playing": "Nepřehrává se",
+ "epic_failure": "Epické selhání!",
+ "added_num_tracks_to_queue": "Přidáno {tracks_length} skladeb do fronty",
+ "spotube_has_an_update": "Spotube má aktualizaci",
+ "download_now": "Stáhnout nyní",
+ "nightly_version": "Byla vydána noční verze Spotube {nightlyBuildNum}",
+ "release_version": "Byla vydána verze Spotube v{version}",
+ "read_the_latest": "Přečtěte si nejnovější ",
+ "release_notes": "poznámky k vydání",
+ "pick_color_scheme": "Vyberte barevné schéma",
+ "save": "Uložit",
+ "choose_the_device": "Vyberte zařízení:",
+ "multiple_device_connected": "Je připojeno více zařízení.\nVyberte zařízení, na kterém chcete provést tuto akci",
+ "nothing_found": "Nic nenalezeno",
+ "the_box_is_empty": "Krabice je prázdná",
+ "top_artists": "Nejlepší umělci",
+ "top_albums": "Nejlepší alba",
+ "this_week": "Tento týden",
+ "this_month": "Tento měsíc",
+ "last_6_months": "Posledních 6 měsíců",
+ "this_year": "Tento rok",
+ "last_2_years": "Poslední 2 roky",
+ "all_time": "Všechny časy",
+ "powered_by_provider": "Pohání {providerName}",
+ "email": "Email",
+ "profile_followers": "Sledující",
+ "birthday": "Narozeniny",
+ "subscription": "Předplatné",
+ "not_born": "Nenarozen",
+ "hacker": "Hacker",
+ "profile": "Profil",
+ "no_name": "Bez jména",
+ "edit": "Upravit",
+ "user_profile": "Uživatelský profil",
+ "count_plays": "{count} přehrání",
+ "streaming_fees_hypothetical": "Poplatky za streamování (hypotetické)",
+ "minutes_listened": "Poslouchané minuty",
+ "streamed_songs": "Streamované skladby",
+ "count_streams": "{count} streamů",
+ "owned_by_you": "Vlastněno vámi",
+ "copied_shareurl_to_clipboard": "Zkopírováno {shareUrl} do schránky",
+ "spotify_hipotetical_calculation": "*Toto je vypočítáno na základě výplaty\nza stream Spotify od $0.003 do $0.005.\nToto je hypotetický výpočet,\nabyste měli představu o tom, kolik\nbyste zaplatili umělcům,\npokud byste poslouchali jejich píseň na Spotify.",
+ "count_mins": "{minutes} minut",
+ "summary_minutes": "minuty",
+ "summary_listened_to_music": "Poslouchal(a) hudbu",
+ "summary_songs": "písně",
+ "summary_streamed_overall": "Streamováno celkově",
+ "summary_owed_to_artists": "Dluženo umělcům\nTento měsíc",
+ "summary_artists": "umělců",
+ "summary_music_reached_you": "Hudba vás oslovila",
+ "summary_full_albums": "plná alba",
+ "summary_got_your_love": "Získal vaši lásku",
+ "summary_playlists": "playlisty",
+ "summary_were_on_repeat": "Byly na opakování",
+ "total_money": "Celkem {money}",
+ "webview_not_found": "Webview nebyl nalezen",
+ "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",
+ "unsupported_platform": "Nepodporovaná platforma"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_de.arb b/lib/l10n/app_de.arb
index ebaa0329..36da0b3e 100644
--- a/lib/l10n/app_de.arb
+++ b/lib/l10n/app_de.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Kopiere den Wert des Cookies \"sp_dc\"",
"step_4_steps": "Füge den kopierten Wert von \"sp_dc\" ein",
"friends": "Freunde",
- "no_lyrics_available": "Entschuldigung, Texte für diesen Track konnten nicht gefunden werden"
+ "no_lyrics_available": "Entschuldigung, Texte für diesen Track konnten nicht gefunden werden",
+ "sort_duration": "Nach Dauer sortieren",
+ "start_a_radio": "Radio starten",
+ "how_to_start_radio": "Wie möchten Sie das Radio starten?",
+ "replace_queue_question": "Möchten Sie die aktuelle Wiedergabeliste ersetzen oder hinzufügen?",
+ "endless_playback": "Endlose Wiedergabe",
+ "delete_playlist": "Wiedergabeliste löschen",
+ "delete_playlist_confirmation": "Sind Sie sicher, dass Sie diese Wiedergabeliste löschen möchten?",
+ "local_tracks": "Lokale Titel",
+ "song_link": "Lied-Link",
+ "skip_this_nonsense": "Diesen Unsinn überspringen",
+ "freedom_of_music": "“Freiheit der Musik”",
+ "freedom_of_music_palm": "“Freiheit der Musik in Ihrer Handfläche”",
+ "get_started": "Lass uns anfangen",
+ "youtube_source_description": "Empfohlen und funktioniert am besten.",
+ "piped_source_description": "Fühlen Sie sich frei? Wie YouTube, aber viel freier.",
+ "jiosaavn_source_description": "Am besten für die südasiatische Region.",
+ "highest_quality": "Höchste Qualität: {quality}",
+ "select_audio_source": "Audioquelle auswählen",
+ "endless_playback_description": "Neue Lieder automatisch\nam Ende der Wiedergabeliste hinzufügen",
+ "choose_your_region": "Wählen Sie Ihre Region",
+ "choose_your_region_description": "Dies wird Spotube helfen, Ihnen den richtigen Inhalt\nfür Ihren Standort anzuzeigen.",
+ "choose_your_language": "Wählen Sie Ihre Sprache",
+ "help_project_grow": "Helfen Sie diesem Projekt zu wachsen",
+ "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.",
+ "contribute_on_github": "Auf GitHub beitragen",
+ "donate_on_open_collective": "Auf Open Collective spenden",
+ "browse_anonymously": "Anonym durchsuchen",
+ "enable_connect": "Verbindung aktivieren",
+ "enable_connect_description": "Spotube von anderen Geräten steuern",
+ "devices": "Geräte",
+ "select": "Auswählen",
+ "connect_client_alert": "Du wirst von {client} gesteuert",
+ "this_device": "Dieses Gerät",
+ "remote": "Fernbedienung",
+ "local_library": "Lokale Bibliothek",
+ "add_library_location": "Zur Bibliothek hinzufügen",
+ "remove_library_location": "Aus der Bibliothek entfernen",
+ "local_tab": "Lokal",
+ "stats": "Statistiken",
+ "and_n_more": "und {count} mehr",
+ "recently_played": "Zuletzt gespielt",
+ "browse_more": "Mehr durchsuchen",
+ "no_title": "Kein Titel",
+ "not_playing": "Wird nicht abgespielt",
+ "epic_failure": "Episches Versagen!",
+ "added_num_tracks_to_queue": "{tracks_length} Titel zur Warteschlange hinzugefügt",
+ "spotube_has_an_update": "Spotube hat ein Update",
+ "download_now": "Jetzt herunterladen",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} wurde veröffentlicht",
+ "release_version": "Spotube v{version} wurde veröffentlicht",
+ "read_the_latest": "Lese die neuesten ",
+ "release_notes": "Versionshinweise",
+ "pick_color_scheme": "Farbschema wählen",
+ "save": "Speichern",
+ "choose_the_device": "Wähle das Gerät:",
+ "multiple_device_connected": "Es sind mehrere Geräte verbunden.\nWähle das Gerät, auf dem diese Aktion ausgeführt werden soll",
+ "nothing_found": "Nichts gefunden",
+ "the_box_is_empty": "Die Box ist leer",
+ "top_artists": "Top-Künstler",
+ "top_albums": "Top-Alben",
+ "this_week": "Diese Woche",
+ "this_month": "Diesen Monat",
+ "last_6_months": "Letzte 6 Monate",
+ "this_year": "Dieses Jahr",
+ "last_2_years": "Letzte 2 Jahre",
+ "all_time": "Alle Zeiten",
+ "powered_by_provider": "Bereitgestellt von {providerName}",
+ "email": "Email",
+ "profile_followers": "Follower",
+ "birthday": "Geburtstag",
+ "subscription": "Abonnement",
+ "not_born": "Nicht geboren",
+ "hacker": "Hacker",
+ "profile": "Profil",
+ "no_name": "Kein Name",
+ "edit": "Bearbeiten",
+ "user_profile": "Benutzerprofil",
+ "count_plays": "{count} Wiedergaben",
+ "streaming_fees_hypothetical": "Streaming-Gebühren (hypothetisch)",
+ "minutes_listened": "Gehörte Minuten",
+ "streamed_songs": "Gestreamte Lieder",
+ "count_streams": "{count} Streams",
+ "owned_by_you": "In Ihrem Besitz",
+ "copied_shareurl_to_clipboard": "{shareUrl} in die Zwischenablage kopiert",
+ "spotify_hipotetical_calculation": "*Dies ist basierend auf Spotifys\npro Stream Auszahlung von $0,003 bis $0,005\nberechnet. Dies ist eine hypothetische Berechnung,\num dem Benutzer Einblick zu geben,\nwieviel sie den Künstlern gezahlt hätten,\nwenn sie ihren Song auf Spotify gehört hätten.",
+ "count_mins": "{minutes} Minuten",
+ "summary_minutes": "Minuten",
+ "summary_listened_to_music": "Hat Musik gehört",
+ "summary_songs": "Lieder",
+ "summary_streamed_overall": "Insgesamt gestreamt",
+ "summary_owed_to_artists": "Den Künstlern geschuldet\nDiesen Monat",
+ "summary_artists": "Künstler",
+ "summary_music_reached_you": "Musik hat Sie erreicht",
+ "summary_full_albums": "volle Alben",
+ "summary_got_your_love": "Hat Ihre Liebe gewonnen",
+ "summary_playlists": "Wiedergabelisten",
+ "summary_were_on_repeat": "Wurden wiederholt",
+ "total_money": "Gesamt {money}",
+ "webview_not_found": "Webview nicht gefunden",
+ "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",
+ "unsupported_platform": "Nicht unterstützte Plattform"
}
\ No newline at end of file
diff --git a/lib/l10n/app_en.arb b/lib/l10n/app_en.arb
index 8257eac9..c63f8543 100644
--- a/lib/l10n/app_en.arb
+++ b/lib/l10n/app_en.arb
@@ -107,6 +107,9 @@
"always_on_top": "Always on top",
"exit_mini_player": "Exit Mini player",
"download_location": "Download location",
+ "local_library": "Local library",
+ "add_library_location": "Add to library",
+ "remove_library_location": "Remove from library",
"account": "Account",
"login_with_spotify": "Login with your Spotify account",
"connect_with_spotify": "Connect with Spotify",
@@ -295,6 +298,7 @@
"delete_playlist": "Delete Playlist",
"delete_playlist_confirmation": "Are you sure you want to delete this playlist?",
"local_tracks": "Local Tracks",
+ "local_tab": "Local",
"song_link": "Song Link",
"skip_this_nonsense": "Skip this nonsense",
"freedom_of_music": "“Freedom of Music”",
@@ -313,5 +317,75 @@
"help_project_grow_description": "Spotube is an open-source project. You can help this project grow by contributing to the project, reporting bugs, or suggesting new features.",
"contribute_on_github": "Contribute on GitHub",
"donate_on_open_collective": "Donate on Open Collective",
- "browse_anonymously": "Browse Anonymously"
+ "browse_anonymously": "Browse Anonymously",
+ "enable_connect": "Enable Connect",
+ "enable_connect_description": "Control Spotube from other devices",
+ "devices": "Devices",
+ "select": "Select",
+ "connect_client_alert": "You're being controlled by {client}",
+ "this_device": "This Device",
+ "remote": "Remote",
+ "stats": "Stats",
+ "and_n_more": "and {count} more",
+ "recently_played": "Recently Played",
+ "browse_more": "Browse More",
+ "no_title": "No Title",
+ "not_playing": "Not playing",
+ "epic_failure": "Epic failure!",
+ "added_num_tracks_to_queue": "Added {tracks_length} tracks to queue",
+ "spotube_has_an_update": "Spotube has an update",
+ "download_now": "Download Now",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} has been released",
+ "release_version": "Spotube v{version} has been released",
+ "read_the_latest": "Read the latest ",
+ "release_notes": "release notes",
+ "pick_color_scheme": "Pick color scheme",
+ "save": "Save",
+ "choose_the_device": "Choose the device:",
+ "multiple_device_connected": "There are multiple device connected.\nChoose the device you want this action to take place",
+ "nothing_found": "Nothing found",
+ "the_box_is_empty": "The box is empty",
+ "top_artists": "Top Artists",
+ "top_albums": "Top Albums",
+ "this_week": "This week",
+ "this_month": "This month",
+ "last_6_months": "Last 6 months",
+ "this_year": "This year",
+ "last_2_years": "Last 2 years",
+ "all_time": "All time",
+ "powered_by_provider": "Powered by {providerName}",
+ "email": "Email",
+ "profile_followers": "Followers",
+ "birthday": "Birthday",
+ "subscription": "Subscription",
+ "not_born": "Not born",
+ "hacker": "Hacker",
+ "profile": "Profile",
+ "no_name": "No Name",
+ "edit": "Edit",
+ "user_profile": "User Profile",
+ "count_plays": "{count} plays",
+ "streaming_fees_hypothetical": "Streaming fees (hypothetical)",
+ "minutes_listened": "Minutes listened",
+ "streamed_songs": "Streamed songs",
+ "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.",
+ "count_mins": "{minutes} mins",
+ "summary_minutes": "minutes",
+ "summary_listened_to_music": "Listened to music",
+ "summary_songs": "songs",
+ "summary_streamed_overall": "Streamed overall",
+ "summary_owed_to_artists": "Owed to artists\nthis month",
+ "summary_artists": "artist's",
+ "summary_music_reached_you": "Music reached you",
+ "summary_full_albums": "full albums",
+ "summary_got_your_love": "Got your love",
+ "summary_playlists": "playlists",
+ "summary_were_on_repeat": "Were on repeat",
+ "total_money": "Total {money}",
+ "webview_not_found": "Webview not found",
+ "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",
+ "unsupported_platform": "Unsupported platform"
}
\ No newline at end of file
diff --git a/lib/l10n/app_es.arb b/lib/l10n/app_es.arb
index 476056cb..d3c8b389 100644
--- a/lib/l10n/app_es.arb
+++ b/lib/l10n/app_es.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Copia el valor de la cookie \"sp_dc\"",
"step_4_steps": "Pega el valor copiado de \"sp_dc\"",
"friends": "Amigos",
- "no_lyrics_available": "Lo siento, no se pueden encontrar las letras de esta pista"
+ "no_lyrics_available": "Lo siento, no se pueden encontrar las letras de esta pista",
+ "sort_duration": "Ordenar por Duración",
+ "start_a_radio": "Iniciar una Radio",
+ "how_to_start_radio": "¿Cómo quieres iniciar la radio?",
+ "replace_queue_question": "¿Quieres reemplazar la lista de reproducción actual o añadir a ella?",
+ "endless_playback": "Reproducción Infinita",
+ "delete_playlist": "Eliminar Lista de Reproducción",
+ "delete_playlist_confirmation": "¿Estás seguro de que quieres eliminar esta lista de reproducción?",
+ "local_tracks": "Pistas Locales",
+ "song_link": "Enlace de la Canción",
+ "skip_this_nonsense": "Saltar esta tontería",
+ "freedom_of_music": "“Libertad de la Música”",
+ "freedom_of_music_palm": "“Libertad de la Música en la palma de tu mano”",
+ "get_started": "Empecemos",
+ "youtube_source_description": "Recomendado y funciona mejor.",
+ "piped_source_description": "¿Te sientes libre? Igual que YouTube pero más libre.",
+ "jiosaavn_source_description": "Lo mejor para la región del sur de Asia.",
+ "highest_quality": "Mayor Calidad: {quality}",
+ "select_audio_source": "Seleccionar Fuente de Audio",
+ "endless_playback_description": "Añadir automáticamente nuevas canciones\nal final de la cola de reproducción",
+ "choose_your_region": "Elige tu región",
+ "choose_your_region_description": "Esto ayudará a Spotube a mostrarte el contenido adecuado\npara tu ubicación.",
+ "choose_your_language": "Elige tu idioma",
+ "help_project_grow": "Ayuda a que este proyecto crezca",
+ "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.",
+ "contribute_on_github": "Contribuir en GitHub",
+ "donate_on_open_collective": "Donar en Open Collective",
+ "browse_anonymously": "Navegar Anónimamente",
+ "enable_connect": "Habilitar conexión",
+ "enable_connect_description": "Controla Spotube desde otros dispositivos",
+ "devices": "Dispositivos",
+ "select": "Seleccionar",
+ "connect_client_alert": "Estás siendo controlado por {client}",
+ "this_device": "Este dispositivo",
+ "remote": "Remoto",
+ "local_library": "Biblioteca local",
+ "add_library_location": "Añadir a la biblioteca",
+ "remove_library_location": "Eliminar de la biblioteca",
+ "local_tab": "Local",
+ "stats": "Estadísticas",
+ "and_n_more": "y {count} más",
+ "recently_played": "Recién reproducido",
+ "browse_more": "Explorar más",
+ "no_title": "Sin título",
+ "not_playing": "No reproduciendo",
+ "epic_failure": "¡Fallo épico!",
+ "added_num_tracks_to_queue": "Se añadieron {tracks_length} canciones a la cola",
+ "spotube_has_an_update": "Spotube tiene una actualización",
+ "download_now": "Descargar ahora",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} ha sido lanzado",
+ "release_version": "Spotube v{version} ha sido lanzado",
+ "read_the_latest": "Lee las últimas ",
+ "release_notes": "notas de la versión",
+ "pick_color_scheme": "Elige esquema de color",
+ "save": "Guardar",
+ "choose_the_device": "Elige el dispositivo:",
+ "multiple_device_connected": "Hay múltiples dispositivos conectados.\nElige el dispositivo en el que deseas realizar esta acción",
+ "nothing_found": "Nada encontrado",
+ "the_box_is_empty": "La caja está vacía",
+ "top_artists": "Artistas principales",
+ "top_albums": "Álbumes principales",
+ "this_week": "Esta semana",
+ "this_month": "Este mes",
+ "last_6_months": "Últimos 6 meses",
+ "this_year": "Este año",
+ "last_2_years": "Últimos 2 años",
+ "all_time": "Todos los tiempos",
+ "powered_by_provider": "Impulsado por {providerName}",
+ "email": "Correo electrónico",
+ "profile_followers": "Seguidores",
+ "birthday": "Cumpleaños",
+ "subscription": "Suscripción",
+ "not_born": "No nacido",
+ "hacker": "Hacker",
+ "profile": "Perfil",
+ "no_name": "Sin nombre",
+ "edit": "Editar",
+ "user_profile": "Perfil de usuario",
+ "count_plays": "{count} reproducciones",
+ "streaming_fees_hypothetical": "Tarifas de streaming (hipotéticas)",
+ "minutes_listened": "Minutos escuchados",
+ "streamed_songs": "Canciones reproducidas",
+ "count_streams": "{count} streams",
+ "owned_by_you": "En tu posesión",
+ "copied_shareurl_to_clipboard": "Copiado {shareUrl} al portapapeles",
+ "spotify_hipotetical_calculation": "*Esto se calcula en base al\npago por stream de Spotify de $0.003 a $0.005.\nEs un cálculo hipotético para dar\nuna idea de cuánto habría\npagado a los artistas si hubieras escuchado\nsu canción en Spotify.",
+ "count_mins": "{minutes} minutos",
+ "summary_minutes": "minutos",
+ "summary_listened_to_music": "Escuchó música",
+ "summary_songs": "canciones",
+ "summary_streamed_overall": "Transmitido en general",
+ "summary_owed_to_artists": "Debido a los artistas\nEste mes",
+ "summary_artists": "artistas",
+ "summary_music_reached_you": "La música te alcanzó",
+ "summary_full_albums": "álbumes completos",
+ "summary_got_your_love": "Obtuvo tu amor",
+ "summary_playlists": "listas de reproducción",
+ "summary_were_on_repeat": "Estaban en repetición",
+ "total_money": "Total {money}",
+ "webview_not_found": "No se encontró el Webview",
+ "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",
+ "unsupported_platform": "Plataforma no soportada"
}
\ No newline at end of file
diff --git a/lib/l10n/app_eu.arb b/lib/l10n/app_eu.arb
new file mode 100644
index 00000000..36986804
--- /dev/null
+++ b/lib/l10n/app_eu.arb
@@ -0,0 +1,391 @@
+{
+ "guest": "Gonbidatua",
+ "browse": "Arakatu",
+ "search": "Bilatu",
+ "library": "Liburutegia",
+ "lyrics": "Hitzak",
+ "settings": "Ezarpenak",
+ "genre_categories_filter": "Kategoria edo generoak filtratu...",
+ "genre": "Generoa",
+ "personalized": "Pertsonalizatua",
+ "featured": "Nabarmenduak",
+ "new_releases": "Argitaratze berriak",
+ "songs": "Abestiak",
+ "playing_track": "{track} erreproduzitzen",
+ "queue_clear_alert": "Uneko zerrenda ezabatuko da. {track_length} abesti ezabatuko dira.\nJarraitu nahi duzu?",
+ "load_more": "Gehiago kargatu",
+ "playlists": "Zerrendak",
+ "artists": "Artistak",
+ "albums": "Albumak",
+ "tracks": "Kantak",
+ "downloads": "Deskargak",
+ "filter_playlists": "Zure zerrendak filtratu...",
+ "liked_tracks": "Gustuko Kantak",
+ "liked_tracks_description": "Zure gustuko kanta guztiak",
+ "create_playlist": "Sortu zerrenda",
+ "create_a_playlist": "Sortu zerrenda bat",
+ "update_playlist": "Eguneratu zerrenda",
+ "create": "Sortu",
+ "cancel": "Ezeztatu",
+ "update": "Eguneratu",
+ "playlist_name": "Zerrenda Izena",
+ "name_of_playlist": "Zerrendaren izena",
+ "description": "Deskribapena",
+ "public": "Publikoa",
+ "collaborative": "Kolaboratiboa",
+ "search_local_tracks": "Bilatu kanta lokalak...",
+ "play": "Erreproduzitu",
+ "delete": "Ezabatu",
+ "none": "Batere ez",
+ "sort_a_z": "Ordenatu A-Z",
+ "sort_z_a": "Ordenatu Z-A",
+ "sort_artist": "Ordenatu Artistaren arabera",
+ "sort_album": "Ordenatu Albumaren arabera",
+ "sort_duration": "Ordenar Iraupenaren arabera",
+ "sort_tracks": "Ordenatu Kantak",
+ "currently_downloading": "Oraintxe ({tracks_length}) deskargatzen",
+ "cancel_all": "Ezeztatu dena",
+ "filter_artist": "Filtratu artistak...",
+ "followers": "{followers} Jarraitzaile",
+ "add_artist_to_blacklist": "Gehitu artista zerrenda beltzera",
+ "top_tracks": "Top Kantak",
+ "fans_also_like": "Fan-ek hau ere gustuko dute",
+ "loading": "Kargatzen...",
+ "artist": "Artista",
+ "blacklisted": "Zerrenda beltzean",
+ "following": "Jarraitzen",
+ "follow": "Jarraitu",
+ "artist_url_copied": "Artistaren URL-a arbelera kopiatua",
+ "added_to_queue": "{tracks} kanta zerrendara gehituak",
+ "filter_albums": "Albumak filtratu...",
+ "synced": "Sinkronizatuta",
+ "plain": "Arrunta",
+ "shuffle": "Ausaz",
+ "search_tracks": "Bilatu kantak...",
+ "released": "Argitaratua",
+ "error": "Errorea: {error}",
+ "title": "Izenburua",
+ "time": "Iraupena",
+ "more_actions": "Ekintza gehiago",
+ "download_count": "({count}) deskarga",
+ "add_count_to_playlist": "Gehitu ({count}) zerrendara",
+ "add_count_to_queue": "Gehitu ({count}) ilarara",
+ "play_count_next": "Erreproduzitu hurrengo ({count})-ak",
+ "album": "Albuma",
+ "copied_to_clipboard": "{data} arbelean kopiatua",
+ "add_to_following_playlists": "Gehitu {track} hurrengo erreprodukzio-zerrendetara",
+ "add": "Gehitu",
+ "added_track_to_queue": "{track} zerrendan gehitua",
+ "add_to_queue": "Gehitu zerrendan",
+ "track_will_play_next": "{track} erreproduzituko da ondoren",
+ "play_next": "Hurrengo erreprodukzioa",
+ "removed_track_from_queue": "{track} zerrendatik ezabatua",
+ "remove_from_queue": "Ezabatu ilaratik",
+ "remove_from_favorites": "Ezabatu gogokoetatik",
+ "save_as_favorite": "Gorde gogokoetan",
+ "add_to_playlist": "Gehitu zerrendara",
+ "remove_from_playlist": "Ezabatu zerrendatik",
+ "add_to_blacklist": "Gehitu zerrenda beltzera",
+ "remove_from_blacklist": "Ezabatu zerrenda beltzetik",
+ "share": "Elkarbanatu",
+ "mini_player": "Mini Erreproduzitzailea",
+ "slide_to_seek": "Arrastatu aurrerantz edo atzearantz bilatzeko",
+ "shuffle_playlist": "Erreproduzitu zerrenda ausazko ordenean",
+ "unshuffle_playlist": "Desgaitu ausazko erreprodukzioa",
+ "previous_track": "Aurreko pista",
+ "next_track": "Hurrengo pista",
+ "pause_playback": "Pausatu erreprodukzioa",
+ "resume_playback": "Berrabiarazi erreprodukzioa",
+ "loop_track": "Kanta begiztan",
+ "repeat_playlist": "Errepikatu lista",
+ "queue": "Ilara",
+ "alternative_track_sources": "Kanten iturri alternatiboak",
+ "download_track": "Deskargatu kanta",
+ "tracks_in_queue": "{tracks} kanta zerrendan",
+ "clear_all": "Garbitu dena",
+ "show_hide_ui_on_hover": "Erakutsi/Ezkutatu interfazea kurtsorea pasatzean",
+ "always_on_top": "Beti ikusgai",
+ "exit_mini_player": "Irten mini erreproduzitzailetik",
+ "download_location": "Deskargen kokapena",
+ "local_library": "Liburutegi lokala",
+ "add_library_location": "Gehitu liburutegira",
+ "remove_library_location": "Kendu liburutegitik",
+ "account": "Kontua",
+ "login_with_spotify": "Hasi saioa zure Spotify kontuarekin",
+ "connect_with_spotify": "Spotify-rekin konektatu",
+ "logout": "Itxi saioa",
+ "logout_of_this_account": "Itxi kontu honen saioa",
+ "language_region": "Hizkuntza eta Herrialdea",
+ "language": "Hizkuntza",
+ "system_default": "Sisteman lehenetsia",
+ "market_place_region": "Dendaren herrialdea",
+ "recommendation_country": "Gomendio herrialdea",
+ "appearance": "Itxura",
+ "layout_mode": "Diseinua",
+ "override_layout_settings": "Responsive diseinuaren ezarpenak ezeztatu",
+ "adaptive": "Moldagarria",
+ "compact": "Trinkoa",
+ "extended": "Hedatua",
+ "theme": "Gaia",
+ "dark": "Iluna",
+ "light": "Argia",
+ "system": "Sistema",
+ "accent_color": "Azentu kolorea",
+ "sync_album_color": "Sinkronizatu albumaren kolorea",
+ "sync_album_color_description": "Albumaren artearen kolore nagusia erabili azentu kolore bezala",
+ "playback": "Erreprodukzioa",
+ "audio_quality": "Audioaren kalitatea",
+ "high": "Altua",
+ "low": "Baxua",
+ "pre_download_play": "Aurre-deskargatu eta erreproduzitu",
+ "pre_download_play_description": "Streaming egin beharrean, byte-ak deskargatu eta erreproduzitu (banda-zabalera handia duten erabiltzaileentzat gomendagarria)",
+ "skip_non_music": "Musika ez diren segmentuak baztertu (SponsorBlock)",
+ "blacklist_description": "Zerrenda beltzeko abesti eta artistak",
+ "wait_for_download_to_finish": "Mesedez, itxaron uneko deskarga bukatu arte",
+ "desktop": "Mahaigaina",
+ "close_behavior": "Ixterako Portaera",
+ "close": "Itxi",
+ "minimize_to_tray": "Sistemako erretilura minimizatu",
+ "show_tray_icon": "Erakutsi ikonoa sistemaren erretiluan",
+ "about": "Honi buruz",
+ "u_love_spotube": "Badakigu Spotube maite duzula",
+ "check_for_updates": "Bilatu eguneraketak",
+ "about_spotube": "Spotube-ri buruz",
+ "blacklist": "Zerrenda beltza",
+ "please_sponsor": "Mesedez, babestu/diruz lagundu",
+ "spotube_description": "Spotube, arina, plataforma-anitza eta doakoa den Spotify-ren bezeroa",
+ "version": "Bertsioa",
+ "build_number": "Konpilazio zenbakia",
+ "founder": "Sortzailea",
+ "repository": "Errepositorioa",
+ "bug_issues": "Erroreak eta arazoak",
+ "made_with": "Bangladesh🇧🇩-en ❤️-z egina",
+ "kingkor_roy_tirtho": "Kingkor Roy Tirtho",
+ "copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
+ "license": "Lizentzia",
+ "add_spotify_credentials": "Gehitu zure Spotify kredentzialak hasi ahal izateko",
+ "credentials_will_not_be_shared_disclaimer": "Ez arduratu, zure kredentzialak ez ditugu bilduko edo inorekin elkarbanatuko",
+ "know_how_to_login": "Ez dakizu nola egin?",
+ "follow_step_by_step_guide": "Jarraitu pausoz-pausoko gida",
+ "spotify_cookie": "Spotify-ren {name} cookiea",
+ "cookie_name_cookie": "{name} cookiea",
+ "fill_in_all_fields": "Mesedez, osatu eremu guztiak",
+ "submit": "Bidali",
+ "exit": "Irten",
+ "previous": "Aurrekoa",
+ "next": "Hurrengoa",
+ "done": "Eginda",
+ "step_1": "1. pausua",
+ "first_go_to": "Hasteko, joan hona",
+ "login_if_not_logged_in": "eta hasi saioa/sortu kontua lehendik ez baduzu eginda",
+ "step_2": "2. pausua",
+ "step_2_steps": "1. Saioa hasita duzularik, sakatu F12 edo saguaren eskuineko botoia klikatu > Ikuskatu nabigatzaileko garapen tresnak irekitzeko.\n2. Joan \"Aplikazio\" (Chrome, Edge, Brave, etab.) edo \"Biltegiratzea\" (Firefox, Palemoon, etab.)\n3. Joan \"Cookieak\" atalera eta gero \"https://accounts.spotify.com\" azpiatalera",
+ "step_3": "3. pausua",
+ "step_3_steps": "Kopiatu \"sp_dc\" cookiearen balioa",
+ "success_emoji": "Eginda! 🥳",
+ "success_message": "Ongi hasi duzu zure Spotify kontua. Lan bikaina, lagun!",
+ "step_4": "4. pausua",
+ "step_4_steps": "Itsatsi \"sp_dc\"-tik kopiatutako balioa",
+ "something_went_wrong": "Zerbaitek huts egin du",
+ "piped_instance": "Piped zerbitzariaren instantzia",
+ "piped_description": "Kanten koizidentzietan erabiltzeko Piped zerbitzariaren instantzia",
+ "piped_warning": "Batzuk agian ez dute ongi funtzionatuko, zure ardurapean erabili",
+ "generate_playlist": "Sortu Zerrenda",
+ "track_exists": "{track} kanta dagoeneko badago",
+ "replace_downloaded_tracks": "Ordezkatu deskargatutako kanta guztiak",
+ "skip_download_tracks": "Deskargatutako kanta guztien deskarga baztertu",
+ "do_you_want_to_replace": "Dagoen kanta ordezkatu nahi duzu??",
+ "replace": "Ordezkatu",
+ "skip": "Baztertu",
+ "select_up_to_count_type": "Aukertu {count} {type}",
+ "select_genres": "Aukeratu Generoak",
+ "add_genres": "Gehitu Generoak",
+ "country": "Herrialdea",
+ "number_of_tracks_generate": "Sortzeko kanta kopurua",
+ "acousticness": "Akustikotasuna",
+ "danceability": "Dantzagarritasuna",
+ "energy": "Energia",
+ "instrumentalness": "Instrumentaltasuna",
+ "liveness": "Zuzenean",
+ "loudness": "Ozentasuna",
+ "speechiness": "Hitzaldia",
+ "valence": "Balentzia",
+ "popularity": "Populartasuna",
+ "key": "Tonua",
+ "duration": "Iraupena (s)",
+ "tempo": "Tenpoa (BPM)",
+ "mode": "Modua",
+ "time_signature": "Konpasa",
+ "short": "Motza",
+ "medium": "Ertaina",
+ "long": "Luzea",
+ "min": "Min.",
+ "max": "Max.",
+ "target": "Helburua",
+ "moderate": "Moderatua",
+ "deselect_all": "Desaukeratu dena",
+ "select_all": "Aukeratu dena",
+ "are_you_sure": "Ziur zaude?",
+ "generating_playlist": "Zure pertsonalizatutako zerrenda sortzen...",
+ "selected_count_tracks": "{count} kanta aukeratuta",
+ "download_warning": "Abesti guztiak aldi berean deskargatuz gero, argi dago musika pirateatzen ari zarela eta musikaren gizarte sortzaileari kalte egiten diozula. Honen jakitun izan eta artisten lan gogorra errespetatu eta babestea espero dut",
+ "download_ip_ban_warning": "Bidenabar, baliteke zure IPa YouTuben blokeatzea deskarga eskera gehiegi egiten badituzu. IPa blokeatzeak esan nahi du ezin izango duzula YouTube erabili (nahiz eta saioa hasia izan) gutxienez 2-3 hilabetez IP helbide horretatik. Eta Spotube ez da erantzule izango hori gertatzen bazaizu",
+ "by_clicking_accept_terms": "'Onartu' klikatzean, ondorengo baldintzak onartzen dituzu:",
+ "download_agreement_1": "Badakit musika pirateatzen ari naizela. Gaiztoa naiz",
+ "download_agreement_2": "Ahal dudanean lagunduko diot artistari baina oraingoz ez dut bere artea erosteko dirurik",
+ "download_agreement_3": "Erabat jakitun naiz YouTubek nire IPa blokea dezakeela eta ez diot Spotube-ri edo bere jabe/laguntzaileei erantzukizunik eskatuko nire oraingo jokaerak ekar ditzakeen arazoengatik",
+ "decline": "Baztertu",
+ "accept": "Onartu",
+ "details": "Xehetasunak",
+ "youtube": "YouTube",
+ "channel": "Kanala",
+ "likes": "Gustukoak",
+ "dislikes": "Ez gustukoak",
+ "views": "Ikuspenak",
+ "streamUrl": "Streaming-aren URLa",
+ "stop": "Gelditu",
+ "sort_newest": "Ordenatu gehitu berrienetik",
+ "sort_oldest": "Ordenatu gehitu zaharrenetik",
+ "sleep_timer": "Itzaltzeko tenporizadorea",
+ "mins": "{minutes} minutu",
+ "hours": "{hours} ordu",
+ "hour": "{hours} ordu",
+ "custom_hours": "Ordu pertsonalizatuak",
+ "logs": "Log-ak",
+ "developers": "Garatzaileak",
+ "not_logged_in": "Ez duzu saioa hasi",
+ "search_mode": "Bilaketa modua",
+ "audio_source": "Audio Iturria",
+ "ok": "OK",
+ "failed_to_encrypt": "Errorea zifratzean",
+ "encryption_failed_warning": "Spotube-ek zifratzea darabil datuak modu seguruan biltegiratzeko. Baina huts egin du. Hori dela eta, biltegiratzea ez da segurua izango\nLinux erabiltzen ari bazara, ziurtatu edozein sekretu-zerbitzu (gnome-keyring, kde-wallet, keepassxc etab.) instalatuta duzula",
+ "querying_info": "Informazioa egiaztatzen...",
+ "piped_api_down": "Piped-en APIa ez dago eskuragarri",
+ "piped_down_error_instructions": "Piped-en {pipedInstance} instantzia ez dago martxan une honetan\n\nAldatu instantzia edo aldatu 'API mota' YouTuberen API ofizialera\n\nZiurtatu aplikazioa berrabiarazten duzula aldaketa eta gero",
+ "you_are_offline": "Une honetan konexiorik gabe zaude",
+ "connection_restored": "Internet konexioa berrezarri egin da",
+ "use_system_title_bar": "Erabili sistemako izenburu barra",
+ "crunching_results": "Emaitzak prozesatzen...",
+ "search_to_get_results": "Bilatu emaitzak lortzeko",
+ "use_amoled_mode": "Erabili AMOLED modua",
+ "pitch_dark_theme": "Dart-en gai iluna",
+ "normalize_audio": "Normalizatu audioa",
+ "change_cover": "Aldatu azala",
+ "add_cover": "Gehitu azala",
+ "restore_defaults": "Berrezarri berezko balioak",
+ "download_music_codec": "Deskargatutako musikaren codec-a",
+ "streaming_music_codec": "Streaming musikaren codec-a",
+ "login_with_lastfm": "Hasi saioa Last.fm-n",
+ "connect": "Konektatu",
+ "disconnect_lastfm": "Deskonektatu Last.fm-tik",
+ "disconnect": "Deskonektatu",
+ "username": "Erabiltzaile izena",
+ "password": "Pasahitza",
+ "login": "Hasi saioa",
+ "login_with_your_lastfm": "Hasi saioa Last.fm-ko zure kontuarekin",
+ "scrobble_to_lastfm": "Scrobble Last.fm-ra",
+ "go_to_album": "Albumera joan",
+ "discord_rich_presence": "Discord-en presentzia aberatsa",
+ "browse_all": "Esploratu dena",
+ "genres": "Generoak",
+ "explore_genres": "Esploratu generoak",
+ "friends": "Lagunak",
+ "no_lyrics_available": "Sentitzen dugu, ezin dira kanta honen hitzak aurkitu",
+ "start_a_radio": "Hasi Irrati bat",
+ "how_to_start_radio": "Nola hasi nahi duzu irratia?",
+ "replace_queue_question": "Uneko zerrenda ordezkatu nahi duzu edo bertan gehitu?",
+ "endless_playback": "Amaigabeko erreprodukzioa",
+ "delete_playlist": "Ezabatu zerrenda",
+ "delete_playlist_confirmation": "Ziur zaude zerrenda ezabatu nahi duzula?",
+ "local_tracks": "Kanta lokalak",
+ "local_tab": "Lokalean",
+ "song_link": "Kantaren lotura",
+ "skip_this_nonsense": "Utzi txorakeria hau",
+ "freedom_of_music": "“Musika Askatasuna”",
+ "freedom_of_music_palm": "“Musika Askatasuna zure eskuetan”",
+ "get_started": "Has gaitezen",
+ "youtube_source_description": "Gomendatua eta hobekien dabilena.",
+ "piped_source_description": "Aske zara? YouTube bezala, baino askeago.",
+ "jiosaavn_source_description": "Asia hegoaldeko herrialdeetarako hoberena.",
+ "highest_quality": "Kalitate Onena: {quality}",
+ "select_audio_source": "Aukeratu Audio Iturria",
+ "endless_playback_description": "Gehitu automatikoki kanta berriak\n ilararen bukaeran",
+ "choose_your_region": "Aukeratu zure herrialdea",
+ "choose_your_region_description": "Honekin Spotube-k zure kokalerakuari dagokion edukia\neskeiniko dizu.",
+ "choose_your_language": "Aukeratu zure hizkuntza",
+ "help_project_grow": "Lagundu proiektu honi hazten",
+ "help_project_grow_description": "Spotube kode irekiko proiektu bat da. Proiektu hau hazten lagundu dezakezu, erroreak jakinaraziz edo ezaugarri berriak proposatuz.",
+ "contribute_on_github": "GitHub-en lagundu",
+ "donate_on_open_collective": "Open Collective-en diruz lagundu",
+ "browse_anonymously": "Nabigatu Anonimoki",
+ "enable_connect": "Gaitu konexioa",
+ "enable_connect_description": "Kontrolatu Spotube beste gailu batzuetatik",
+ "devices": "Gailuak",
+ "select": "Aukeratu",
+ "connect_client_alert": "{client} gailuak kontrolatzen zaitu",
+ "this_device": "Gailu hau",
+ "remote": "Urrunekoa",
+ "stats": "Estatistikak",
+ "and_n_more": "eta {count} gehiago",
+ "recently_played": "Berriki entzunak",
+ "browse_more": "Gehiago Bilatu",
+ "no_title": "Titulurik ez",
+ "not_playing": "Erreprodukziorik ez",
+ "epic_failure": "Sekulako errorea!",
+ "added_num_tracks_to_queue": "{tracks_length} kanta gehitu dira zerrendara",
+ "spotube_has_an_update": "Spotube-ren eguneraketa bat dago",
+ "download_now": "Orain deskargatu",
+ "nightly_version": "Spotube {nightlyBuildNum} Nightly-a argitaratu da",
+ "release_version": "Spotube v{version} argitaratu da",
+ "read_the_latest": "Irakurri azken ",
+ "release_notes": "argitatratze oharrak",
+ "pick_color_scheme": "Aukeratu kolore eskema",
+ "save": "Gorde",
+ "choose_the_device": "Aukeratu gailua:",
+ "multiple_device_connected": "Hainbat gailu daude konektatuta.\nAukeratu zein gailutan aplikatu nahi duzun ekintza hau",
+ "nothing_found": "Ezer ez da aurkitu",
+ "the_box_is_empty": "Kaxa hutsik dago",
+ "top_artists": "Top Artistak",
+ "top_albums": "Top Albumak",
+ "this_week": "Aste honetan",
+ "this_month": "Hilabete honetan",
+ "last_6_months": "Azken 6 hilabeteetan",
+ "this_year": "Aurten",
+ "last_2_years": "Azken 2 urtetan",
+ "all_time": "Betidanik",
+ "powered_by_provider": "{providerName}-ren eskutik",
+ "email": "Email",
+ "profile_followers": "Jarraitzaileak",
+ "birthday": "Jaiotze-data",
+ "subscription": "Harpidetzak",
+ "not_born": "Jaio gabe",
+ "hacker": "Hacker",
+ "profile": "Profila",
+ "no_name": "Izenik Ez",
+ "edit": "Editatu",
+ "user_profile": "Erabiltzaile Profila",
+ "count_plays": "{count} erreprodukzio",
+ "streaming_fees_hypothetical": "Streaming ordainketa (hipotetikoa)",
+ "minutes_listened": "Entzundako minutuak",
+ "streamed_songs": "Streaming-ez entzundako kantak",
+ "count_streams": "{count} stream",
+ "owned_by_you": "Zure jabetzakoa",
+ "copied_shareurl_to_clipboard": "{shareUrl} arbelera kopiatua",
+ "spotify_hipotetical_calculation": "*Sportify-k stream bakoitzeko duen $0.003 eta $0.005\nordainsarian oinarritua da. Kalkulu hipotetiko bat,\nkanta hauek Spotify-n entzun bazenitu,\nberaiek artistari zenbat ordaiduko lioketen jakin dezazun.",
+ "count_mins": "{minutes} minutu",
+ "summary_minutes": "minutu",
+ "summary_listened_to_music": "Musika entzuten",
+ "summary_songs": "kanta",
+ "summary_streamed_overall": "Streaming abesti oro har",
+ "summary_owed_to_artists": "Hilabete honetan\nartistei zor zaiena",
+ "summary_artists": "artisten",
+ "summary_music_reached_you": "Musika ailegatu zaizu",
+ "summary_full_albums": "album osok",
+ "summary_got_your_love": "Jaso dute zure maitasuna",
+ "summary_playlists": "zerrenda",
+ "summary_were_on_repeat": "Dituzu errepikatze moduan",
+ "total_money": "Guztira {money}",
+ "webview_not_found": "Ez da Webview aurkitu",
+ "webview_not_found_description": "Ez dago Webview abiarazte denbora-instalaziorik zure gailuan.\nInstalatuta badago, ziurtatu environment PATH-an dagoela\n\nInstalatu ondoren, berrabiarazi aplikazioa",
+ "unsupported_platform": "Plataforma ez onartua"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_fa.arb b/lib/l10n/app_fa.arb
index 3a2bcb4b..47242a04 100644
--- a/lib/l10n/app_fa.arb
+++ b/lib/l10n/app_fa.arb
@@ -286,5 +286,106 @@
"step_3_steps": "مقدار کوکی \"sp_dc\" را کپی کنید",
"step_4_steps": "مقدار کپی شده \"sp_dc\" را الصاق کنید",
"friends": "دوستان",
- "no_lyrics_available": "متاسفیم، قادر به یافتن متن این قطعه نیستیم"
+ "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": "احساس آزادی میکنید؟ مانند یوتیوب اما بیشتر آزاد.",
+ "jiosaavn_source_description": "بهترین برای منطقه جنوب آسیا.",
+ "highest_quality": "بالاترین کیفیت: {quality}",
+ "select_audio_source": "انتخاب منبع صوتی",
+ "endless_playback_description": "خودکار اضافه کردن آهنگهای جدید\nبه انتهای صف",
+ "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 {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": "۶ ماه گذشته",
+ "this_year": "امسال",
+ "last_2_years": "۲ سال گذشته",
+ "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": "*این بر اساس پرداخت هر پخش اسپاتیفای\nبه مبلغ 0.003 تا 0.005 دلار محاسبه شده است.\nاین یک محاسبه فرضی است که به کاربران نشان دهد چقدر ممکن است\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_not_found_description": "هیچ اجرای وبویو روی دستگاه شما نصب نشده است.\nدر صورت نصب، مطمئن شوید که در environment PATH قرار دارد\n\nپس از نصب، برنامه را مجدداً راهاندازی کنید",
+ "unsupported_platform": "پلتفرم پشتیبانی نمیشود"
}
\ No newline at end of file
diff --git a/lib/l10n/app_fi.arb b/lib/l10n/app_fi.arb
new file mode 100644
index 00000000..53b948a6
--- /dev/null
+++ b/lib/l10n/app_fi.arb
@@ -0,0 +1,391 @@
+{
+ "guest": "Vieras",
+ "browse": "Selaa",
+ "search": "Hae",
+ "library": "Kirjasto",
+ "lyrics": "Lyriikat",
+ "settings": "Asetukset",
+ "genre_categories_filter": "Suodata kategorioita tai genrejä",
+ "genre": "Genre",
+ "personalized": "Personoidut",
+ "featured": "Esittelyssä",
+ "new_releases": "Uusi julkaisu",
+ "songs": "Laulut",
+ "playing_track": "Soitetaan {track}",
+ "queue_clear_alert": "Tämä tulee tyhjentämään jonon. {track_length} Kappaleita poistetaan\nHaluatko jatkaa?",
+ "load_more": "Lataa lisää",
+ "playlists": "Soittolistat",
+ "artists": "Artistit",
+ "albums": "Albumit",
+ "tracks": "Kappaleet",
+ "downloads": "Lataukset",
+ "filter_playlists": "Suodata soittolistasi...",
+ "liked_tracks": "Tykätyt kappaleet",
+ "liked_tracks_description": "Kaikki tykättysi kappaleet",
+ "create_playlist": "Luo soittolista",
+ "create_a_playlist": "Luo soittolista",
+ "update_playlist": "Päivitä soittolista",
+ "create": "Luo",
+ "cancel": "Peruuta",
+ "update": "Päivitä",
+ "playlist_name": "Soittolistan nimi",
+ "name_of_playlist": "Soittolistan nimi",
+ "description": "Kuvaus",
+ "public": "Julkinen",
+ "collaborative": "Collaborative",
+ "search_local_tracks": "Hae paikallisia lauluja...",
+ "play": "Soita",
+ "delete": "Poista",
+ "none": "Ei mitään",
+ "sort_a_z": "Suodata A-Z",
+ "sort_z_a": "Suodata Z-A",
+ "sort_artist": "Suodata Artistilta",
+ "sort_album": "Suodata Albumilta",
+ "sort_duration": "Suodata Pituudelta",
+ "sort_tracks": "Suodata Kappaleet",
+ "currently_downloading": "Ladataan ({tracks_length})",
+ "cancel_all": "Peru kaikki",
+ "filter_artist": "Suodata artistit...",
+ "followers": "{followers} Seuraajaa",
+ "add_artist_to_blacklist": "Lisää artisti mustalle listalle",
+ "top_tracks": "Suosituimmat kappaleet",
+ "fans_also_like": "Fanit myös tykkäsivät",
+ "loading": "Ladataan...",
+ "artist": "Artisti",
+ "blacklisted": "Mustalistattu",
+ "following": "Seurataan",
+ "follow": "Seuraa",
+ "artist_url_copied": "Aristin URL kopioitiin leikepöytään",
+ "added_to_queue": "Lisättiin {tracks} kappaletta jonoon",
+ "filter_albums": "Suodata albumit...",
+ "synced": "Synkronoitu",
+ "plain": "Tavallinen",
+ "shuffle": "Sekoita",
+ "search_tracks": "Hae kappaleita...",
+ "released": "Julkaistu",
+ "error": "Virhe {error}",
+ "title": "Otsikko",
+ "time": "Aika",
+ "more_actions": "Lisää toimintoja",
+ "download_count": "Lataa ({count})",
+ "add_count_to_playlist": "Lisää ({count}) Soittolistaasi",
+ "add_count_to_queue": "Lisää ({count}) Jonoon",
+ "play_count_next": "Soita ({count}) seuraavaksi",
+ "album": "Albumi",
+ "copied_to_clipboard": "Kopioitiin {data} leikepöytään",
+ "add_to_following_playlists": "Lisää {track} seuraaviin soittolistoihin",
+ "add": "Lisää",
+ "added_track_to_queue": "Lisättiin {track} jonoon",
+ "add_to_queue": "Lisää jonoon",
+ "track_will_play_next": "{track} Soitetaan seuraavaksi",
+ "play_next": "Soita seuraavaksi",
+ "removed_track_from_queue": "Poistettiin {track} jonosta",
+ "remove_from_queue": "Poista jonosta",
+ "remove_from_favorites": "Poista suosikeista",
+ "save_as_favorite": "Tallenna soittolistana",
+ "add_to_playlist": "Lisää soittolistaan",
+ "remove_from_playlist": "Poista soittolistasta",
+ "add_to_blacklist": "Lisää mustalle listalle",
+ "remove_from_blacklist": "Poista mustalistalta",
+ "share": "Jaa",
+ "mini_player": "Minisoitin",
+ "slide_to_seek": "Liu'uta mennäkseen eteenpäin tai taaksepäin",
+ "shuffle_playlist": "Sekoita soittolista",
+ "unshuffle_playlist": "Poista sekoitus soittolistasta",
+ "previous_track": "Äskeinen kappale",
+ "next_track": "Seuraava kappale",
+ "pause_playback": "Pysäytä soittolistan toisto",
+ "resume_playback": "Jatka soittolistan toistoa",
+ "loop_track": "Uudelleentoista kappale",
+ "repeat_playlist": "Toista soittolista uudelleen",
+ "queue": "Jono",
+ "alternative_track_sources": "Toinen kappale lähde",
+ "download_track": "Lataa kappale",
+ "tracks_in_queue": "{tracks} kappaletta jonossa",
+ "clear_all": "Tyhjennä kaikki",
+ "show_hide_ui_on_hover": "Näytä/Piilota UI leijumalla",
+ "always_on_top": "Aina päällimmäisenä",
+ "exit_mini_player": "Lähde minisoittimesta",
+ "download_location": "Lataus sijainti",
+ "account": "Käyttäjä",
+ "login_with_spotify": "Kirjaudu Spotify-käyttäjällä",
+ "connect_with_spotify": "Yhdistä Spotify:lla",
+ "logout": "Kirjaudu ulos",
+ "logout_of_this_account": "Kirjaudu ulos tältä käyttäjältä",
+ "language_region": "Kieli ja Maa",
+ "language": "Kieli",
+ "system_default": "Järjestelmän oletus",
+ "market_place_region": "Markkina-alue",
+ "recommendation_country": "Suositeltu maa",
+ "appearance": "Ulkomuto",
+ "layout_mode": "Asettelutila",
+ "override_layout_settings": "Jätä reagoiva asettelutila huomioimatta",
+ "adaptive": "Mukautuva",
+ "compact": "Kompakti",
+ "extended": "Laajennettu",
+ "theme": "Teema",
+ "dark": "Tumma",
+ "light": "Vaalea",
+ "system": "Järjestelmä",
+ "accent_color": "Korostusväri",
+ "sync_album_color": "Synkronoi albumin väri",
+ "sync_album_color_description": "Käyttää albumin kansitaiteen vallitsevaa väirä korostuvärinä",
+ "playback": "Toisto",
+ "audio_quality": "Äänenlaatu",
+ "high": "Korkea",
+ "low": "Matala",
+ "pre_download_play": "Esilataa ja soita",
+ "pre_download_play_description": "Audion suoratoiston sijaan, lataa tavut ja soita ne (Suositeltu korkeamman kaistanleveyden käyttäjille)",
+ "skip_non_music": "Ohita ei-musiikki kohdat (SponsorBlock)",
+ "blacklist_description": "Mustalistat kappaleet aja artistit",
+ "wait_for_download_to_finish": "Odota nykyisen latauksen lopetteluun",
+ "desktop": "Työpöytä",
+ "close_behavior": "Sulkemisen käyttäytyminen",
+ "close": "Sulje",
+ "minimize_to_tray": "Minimisoi tehtäväpalkkiin",
+ "show_tray_icon": "Näytä järjestelmäkuvake",
+ "about": "Tietoa",
+ "u_love_spotube": "Tiedämme että rakastat Spotubea",
+ "check_for_updates": "Tarkista päivitykset",
+ "about_spotube": "Tietoa Spotube:sta",
+ "blacklist": "Mustalista",
+ "please_sponsor": "Sponsoroi/Lahjoita, kiitos",
+ "spotube_description": "Spotube, kevyt, cross-platform, vapaa-kaikille spotify clientti",
+ "version": "Versio",
+ "build_number": "Rakennusnumero",
+ "founder": "Perustaja",
+ "repository": "Arkisto",
+ "bug_issues": "Bugit+Ongelmat",
+ "made_with": "Tehty ❤️ Bangladeshista 🇧🇩",
+ "kingkor_roy_tirtho": "Kingkor Roy Tirtho",
+ "copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
+ "license": "Lisenssi",
+ "add_spotify_credentials": "Lisää Spotify-tunnuksesi aloittaaksesi",
+ "credentials_will_not_be_shared_disclaimer": "Älä huoli, tunnuksiasi ei talleteta tai jaeta kenenkään kanssa",
+ "know_how_to_login": "Etkö tiedä miten tehdä tämä?",
+ "follow_step_by_step_guide": "Seuraa askel askeleelta opasta",
+ "spotify_cookie": "Spotify {name} Keksi",
+ "cookie_name_cookie": "{name} Keksi",
+ "fill_in_all_fields": "Täytä kaikki kentät",
+ "submit": "Lähetä",
+ "exit": "Poistu",
+ "previous": "Edellinen",
+ "next": "Seuraava",
+ "done": "Tehty",
+ "step_1": "Vaihe 1",
+ "first_go_to": "Ensiksi, mene",
+ "login_if_not_logged_in": "ja Kirjaudu/Tee tili jos et ole kirjautunut sisään",
+ "step_2": "Vaihe 2",
+ "step_2_steps": "1. Kun olet kirjautunut, paina F12 tai oikeaa hiiren näppäintä > Tarkista ja avaa selaimen kehittäjä työkalut.\n2. Mene sitten \"Application\"-välilehteen (Chrome, Edge, Brave jne..) tai \"Storage\"-välilehteen (Firefox, Palemoon jne..)\n3. Mene \"Cookies\"-osastoon, sitten \"https://accounts.spotify.com\" alakohtaan.",
+ "step_3": "Vaihe 3",
+ "step_3_steps": "Kopioi Keksin \"sp_dc\" arvo",
+ "success_emoji": "Onnistuit🥳",
+ "success_message": "Olet nyt kirjautunut sisään Spotify-käyttäjällesi. Hyvää työtä toveri!",
+ "step_4": "Vaihe 4",
+ "step_4_steps": "Liitä kopioitu \"sp_dc\" arvo",
+ "something_went_wrong": "Jotain meni pieleen",
+ "piped_instance": "Johdettu palvelinesiintymä",
+ "piped_description": "Johdettu palvelinesiintymä Kappale täsmäyksiin",
+ "piped_warning": "Jotkut niistä eivät toimi hyvin, käytä siis omalla vastuullasi",
+ "generate_playlist": "Tuota soittolista",
+ "track_exists": "Kappale {track} on jo olemassa!",
+ "replace_downloaded_tracks": "Korvaa kaikki ladatut kappaleet",
+ "skip_download_tracks": "Ohita ladattujen laulujen lataaminen",
+ "do_you_want_to_replace": "Haluatko korvata olemassa olevan kappaleen??",
+ "replace": "Korvaa",
+ "skip": "Ohita",
+ "select_up_to_count_type": "Valitse enintään {count} {type}",
+ "select_genres": "Valitse Genret",
+ "add_genres": "Lisää Genrejä",
+ "country": "Maa",
+ "number_of_tracks_generate": "Numero tuotettavia kappaleita",
+ "acousticness": "Akustisuus",
+ "danceability": "Tanssittavuus",
+ "energy": "Energia",
+ "instrumentalness": "Instrumentaalisuus",
+ "liveness": "Elävyyttä",
+ "loudness": "Äänekkyys",
+ "speechiness": "Puheisuus",
+ "valence": "Valenssi",
+ "popularity": "Suosio",
+ "key": "Sävellaji",
+ "duration": "Pituus (s)",
+ "tempo": "Tempo (BPM)",
+ "mode": "Tila",
+ "time_signature": "Aikamerkki",
+ "short": "Lyhyt",
+ "medium": "Keskikokoinen",
+ "long": "Pitkä",
+ "min": "Minimi",
+ "max": "Maximi",
+ "target": "Kohde",
+ "moderate": "Kohtalainen",
+ "deselect_all": "Poista kaikki valinnat",
+ "select_all": "Valitse kaikki",
+ "are_you_sure": "Oletko varma?",
+ "generating_playlist": "Luodaan mukautettua soittolistoa...",
+ "selected_count_tracks": "Valittu {count} kappaletta",
+ "download_warning": "Jos lataat kaikki laulut kerrällä olet selkeästi Piratoimassa ja aiheuttamassa vahinkoa musiikin luovaan yhteiskuntaan. Toivottavasti olet tietoinen tästä. Yritä aina kunnioittaa ja tukea Artistin kovaa työtä.",
+ "download_ip_ban_warning": "BTW, YouTube voi estää IP-Osoitteesi tavallista liiallisten latauspyyntöjen takia. IP-Osoitteen esto tarkoittaa sitä, ettet voi käyttää YouTubea (vaikka olisit kirjautunut) vähintään 2-3kk aikana kyseiseltä laitteelta. Spotube ei kanna yhtään vastuuta jos se tapahtuu.",
+ "by_clicking_accept_terms": "Painamalla 'hyväksy' hyväksyt seuraaviin ehtoihin:",
+ "download_agreement_1": "Tiedän että Piratoin musiikkia. Olen paha.",
+ "download_agreement_2": "Tuen Artisteja silloin kun pystyn, ja teen tämän vain koska minulla ei ole rahaa ostaa heidän taidetta",
+ "download_agreement_3": "Ymmärrän että minun YouTube voi estää IP-Osoitteeni ja en pidä Spotubea tai omistajiinsa/avustajia vastuullisena mistään omista teoistsani",
+ "decline": "Hylkää",
+ "accept": "Hyväksy",
+ "details": "Yksityiskohdat",
+ "youtube": "YouTube",
+ "channel": "Kanava",
+ "likes": "Tykkäykset",
+ "dislikes": "Epä-tykkäykset",
+ "views": "Näyttökerrat",
+ "streamUrl": "Suoratoiston URL",
+ "stop": "Lopeta",
+ "sort_newest": "Suodata uusimmista",
+ "sort_oldest": "Suodata vanhimmista",
+ "sleep_timer": "Uniajastin",
+ "mins": "{minutes} Minuuttia",
+ "hours": "{hours} Tuntia",
+ "hour": "{hours} Tunti",
+ "custom_hours": "Mukautetut tunnit",
+ "logs": "Lokit",
+ "developers": "Kehittäjät",
+ "not_logged_in": "Et ole kirjautunut sisään.",
+ "search_mode": "Hakutila",
+ "audio_source": "Äänilähde",
+ "ok": "Ok",
+ "failed_to_encrypt": "Salaaminen epäonnistui",
+ "encryption_failed_warning": "Spotube käyttää salausta tallentaakseen tietosi, mutta epäonnistui, joten se palaa epäturvalliseen tallennukseen\nJos käytät Linuxia, varmista että sinulla on turvallisuuspalvelu (gnome-keyring, kde-wallet, keepassxc jne) asennettu",
+ "querying_info": "Hankitaan tietoa...",
+ "piped_api_down": "Johdettu palvelinesiintymä on alhaalla",
+ "piped_down_error_instructions": "Johdettu palvelinesiintymä {pipedInstance} on alhaalla.\n\nVaihda joko ilmeytymä tia vahda 'API tyyppi' YouTuben viralliseen API\n\nKäynnistä sovellus uudestaan vaihdon jälkeen",
+ "you_are_offline": "Et ole yhdistetty verkkoon",
+ "connection_restored": "Verkkoyhteys palautettu",
+ "use_system_title_bar": "Käytä järjestelmäpalkkia",
+ "crunching_results": "Paloitellaan tuloksia...",
+ "search_to_get_results": "Hae saadakseen tuloksia",
+ "use_amoled_mode": "Pilkkopimeä tumma teema",
+ "pitch_dark_theme": "AMOLED Tila",
+ "normalize_audio": "Normalisoi audio",
+ "change_cover": "Vaihda koveri",
+ "add_cover": "Lisää koveri",
+ "restore_defaults": "Palauta oletukset",
+ "download_music_codec": "Ladatun musiikin codefc",
+ "streaming_music_codec": "Suoratoistetun musiikin codec",
+ "login_with_lastfm": "Kirjaudu sisään Last.fm:llä",
+ "connect": "Yhdistä",
+ "disconnect_lastfm": "Katkaise Last.fm",
+ "disconnect": "Katkaise",
+ "username": "Käyttäjänimi",
+ "password": "Salasana",
+ "login": "Kirjaudu",
+ "login_with_your_lastfm": "Kirjaudu Last.fm käyttäjälläsi",
+ "scrobble_to_lastfm": "Scrobble Last.fm:ään",
+ "go_to_album": "Mene albumiin",
+ "discord_rich_presence": "Discord Rich Presence",
+ "browse_all": "Selaa kaikki",
+ "genres": "Genret",
+ "explore_genres": "Seikkaile genrejä",
+ "friends": "Kaverit",
+ "no_lyrics_available": "Anteeksi, emme löytäneet lyriikoita tälle laululle",
+ "start_a_radio": "Aloita Radio",
+ "how_to_start_radio": "Kuinka haluat aloittaa radion?",
+ "replace_queue_question": "Haluatko korvata nykyisen jonon vai lisätä siihen?",
+ "endless_playback": "Loputon toisto",
+ "delete_playlist": "Poista soittolista",
+ "delete_playlist_confirmation": "Oletko varma että haluat poistaa tämän soittolistan?",
+ "local_tracks": "Paikalliset kappaleet",
+ "song_link": "Laulun linkki",
+ "skip_this_nonsense": "Ohita tämä hölynpöly",
+ "freedom_of_music": "“Musiikin vapaus”",
+ "freedom_of_music_palm": "“Musiikin vapaus käsissäsi”",
+ "get_started": "Aloitetaan",
+ "youtube_source_description": "Suositeltu ja toimii parhaiten.",
+ "piped_source_description": "Tuntuuko vapaalta? Sama kuin YouTube mutta paljon vapautta",
+ "jiosaavn_source_description": "Paras Etelä-Aasian alueelle.",
+ "highest_quality": "Korkein laatu: {quality}",
+ "select_audio_source": "Valitse äänilähde",
+ "endless_playback_description": "Lisää automaattisesti uusia lauluja\njonon perään",
+ "choose_your_region": "Valitse alueesi",
+ "choose_your_region_description": "Tämä auttaa Spotube näyttämään sinulle oikeaa sisältöä\nsijaintiasi varten.",
+ "choose_your_language": "Valitse kielesi",
+ "help_project_grow": "Auta tätä projektia kasvamaan",
+ "help_project_grow_description": "Spotube projekti minkä lähdekoodi on julkisesti saatavilla. Voit autta tätä projektia kasvamaan muutoksilla, ilmoittamalla bugeista, tai ehdottamalla uusia ominaisuuksia.",
+ "contribute_on_github": "Auta GitHub:ssa",
+ "donate_on_open_collective": "Lahjoita avoimessa kollektiivissa",
+ "browse_anonymously": "Selaa anonyyminä",
+ "enable_connect": "Ota käyttöön yhdistäminen",
+ "enable_connect_description": "Ohjaa Spotubea toiselta laitteelta",
+ "devices": "Laitteet",
+ "select": "Valitse",
+ "connect_client_alert": "{client} ohjaa sinua",
+ "this_device": "Tämä laite",
+ "remote": "Etä",
+ "local_library": "Paikallinen kirjasto",
+ "add_library_location": "Lisää kirjastoon",
+ "remove_library_location": "Poista kirjastosta",
+ "local_tab": "Paikallinen",
+ "stats": "Tilastot",
+ "and_n_more": "ja {count} lisää",
+ "recently_played": "Äskettäin soitetut",
+ "browse_more": "Selaa lisää",
+ "no_title": "Ei otsikkoa",
+ "not_playing": "Ei soi",
+ "epic_failure": "Epäonnistuminen!",
+ "added_num_tracks_to_queue": "Lisätty {tracks_length} kappaletta jonoon",
+ "spotube_has_an_update": "Spotubella on päivitys",
+ "download_now": "Lataa nyt",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} on julkaistu",
+ "release_version": "Spotube v{version} on julkaistu",
+ "read_the_latest": "Lue viimeisimmät",
+ "release_notes": "julkaisumuistiinpanot",
+ "pick_color_scheme": "Valitse värimaailma",
+ "save": "Tallenna",
+ "choose_the_device": "Valitse laite:",
+ "multiple_device_connected": "Useita laitteita on kytketty.\nValitse laite, jossa haluat toiminnon suorittaa",
+ "nothing_found": "Ei tuloksia",
+ "the_box_is_empty": "Laatikko on tyhjä",
+ "top_artists": "Suosituimmat artistit",
+ "top_albums": "Suosituimmat albumit",
+ "this_week": "Tällä viikolla",
+ "this_month": "Tässä kuussa",
+ "last_6_months": "Viimeiset 6 kuukautta",
+ "this_year": "Tänä vuonna",
+ "last_2_years": "Viimeiset 2 vuotta",
+ "all_time": "Kaikki ajat",
+ "powered_by_provider": "Tuottanut {providerName}",
+ "email": "Sähköposti",
+ "profile_followers": "Seuraajat",
+ "birthday": "Syntymäpäivä",
+ "subscription": "Tilaus",
+ "not_born": "Ei syntynyt",
+ "hacker": "Hakkeri",
+ "profile": "Profiili",
+ "no_name": "Ei nimeä",
+ "edit": "Muokkaa",
+ "user_profile": "Käyttäjäprofiili",
+ "count_plays": "{count} toistoa",
+ "streaming_fees_hypothetical": "Suoratoiston maksut (hypoteettinen)",
+ "minutes_listened": "Kuunneltuja minuutteja",
+ "streamed_songs": "Suoratoistettuja kappaleita",
+ "count_streams": "{count} suoratoistoa",
+ "owned_by_you": "Sinun omistama",
+ "copied_shareurl_to_clipboard": "{shareUrl} kopioitu leikepöydälle",
+ "spotify_hipotetical_calculation": "*Tämä on laskettu Spotifyn suoratoiston\nmaksun perusteella, joka on 0,003–0,005 dollaria.\nTämä on hypoteettinen laskelma, joka antaa käyttäjälle käsityksen\nsiitä, kuinka paljon he olisivat maksaneet artisteille,\njollei heidän kappaleensa olisi kuunneltu Spotifyssa.",
+ "count_mins": "{minutes} min",
+ "summary_minutes": "minuuttia",
+ "summary_listened_to_music": "Kuunneltu musiikkia",
+ "summary_songs": "kappaletta",
+ "summary_streamed_overall": "Suoratoistettu yhteensä",
+ "summary_owed_to_artists": "Maksettava artisteille\nTässä kuussa",
+ "summary_artists": "artisti",
+ "summary_music_reached_you": "Musiikki saavutti sinut",
+ "summary_full_albums": "täydet albumit",
+ "summary_got_your_love": "Sai rakkautesi",
+ "summary_playlists": "soittolistat",
+ "summary_were_on_repeat": "Olivat toistossa",
+ "total_money": "Yhteensä {money}",
+ "webview_not_found": "Webview ei löydy",
+ "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",
+ "unsupported_platform": "Ei tuettu alusta"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_fr.arb b/lib/l10n/app_fr.arb
index 5c24d0fe..522a2af4 100644
--- a/lib/l10n/app_fr.arb
+++ b/lib/l10n/app_fr.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Copiez la valeur du cookie \"sp_dc\"",
"step_4_steps": "Collez la valeur copiée de \"sp_dc\"",
"friends": "Amis",
- "no_lyrics_available": "Désolé, impossible de trouver les paroles de cette piste"
+ "no_lyrics_available": "Désolé, impossible de trouver les paroles de cette piste",
+ "sort_duration": "Trier par durée",
+ "start_a_radio": "Démarrer une radio",
+ "how_to_start_radio": "Comment voulez-vous démarrer la radio ?",
+ "replace_queue_question": "Voulez-vous remplacer la file d'attente actuelle ou y ajouter ?",
+ "endless_playback": "Lecture sans fin",
+ "delete_playlist": "Supprimer la playlist",
+ "delete_playlist_confirmation": "Êtes-vous sûr de vouloir supprimer cette playlist ?",
+ "local_tracks": "Titres locaux",
+ "song_link": "Lien de la chanson",
+ "skip_this_nonsense": "Passer cette absurdité",
+ "freedom_of_music": "“Liberté de la musique”",
+ "freedom_of_music_palm": "“Liberté de la musique dans la paume de votre main”",
+ "get_started": "Commençons",
+ "youtube_source_description": "Recommandé et fonctionne mieux.",
+ "piped_source_description": "Vous vous sentez libre ? Comme YouTube mais beaucoup plus gratuit.",
+ "jiosaavn_source_description": "Le meilleur pour la région d'Asie du Sud.",
+ "highest_quality": "Meilleure qualité : {quality}",
+ "select_audio_source": "Sélectionner la source audio",
+ "endless_playback_description": "Ajouter automatiquement de nouvelles chansons à la fin de la file d'attente",
+ "choose_your_region": "Choisissez votre région",
+ "choose_your_region_description": "Cela aidera Spotube à vous montrer le bon contenu pour votre emplacement.",
+ "choose_your_language": "Choisissez votre langue",
+ "help_project_grow": "Aidez ce projet à grandir",
+ "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.",
+ "contribute_on_github": "Contribuer sur GitHub",
+ "donate_on_open_collective": "Faire un don sur Open Collective",
+ "browse_anonymously": "Naviguer anonymement",
+ "enable_connect": "Activer la connexion",
+ "enable_connect_description": "Contrôlez Spotube depuis d'autres appareils",
+ "devices": "Appareils",
+ "select": "Sélectionner",
+ "connect_client_alert": "Vous êtes contrôlé par {client}",
+ "this_device": "Cet appareil",
+ "remote": "À distance",
+ "local_library": "Bibliothèque locale",
+ "add_library_location": "Ajouter à la bibliothèque",
+ "remove_library_location": "Retirer de la bibliothèque",
+ "local_tab": "Local",
+ "stats": "Statistiques",
+ "and_n_more": "et {count} de plus",
+ "recently_played": "Récemment joué",
+ "browse_more": "Parcourir plus",
+ "no_title": "Sans titre",
+ "not_playing": "Non joué",
+ "epic_failure": "Échec épique!",
+ "added_num_tracks_to_queue": "{tracks_length} morceaux ajoutés à la file d'attente",
+ "spotube_has_an_update": "Spotube a une mise à jour",
+ "download_now": "Télécharger maintenant",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} a été publié",
+ "release_version": "Spotube v{version} a été publié",
+ "read_the_latest": "Lisez les dernières ",
+ "release_notes": "notes de version",
+ "pick_color_scheme": "Choisissez le schéma de couleurs",
+ "save": "Sauvegarder",
+ "choose_the_device": "Choisissez l'appareil:",
+ "multiple_device_connected": "Plusieurs appareils sont connectés.\nChoisissez l'appareil sur lequel vous souhaitez effectuer cette action",
+ "nothing_found": "Rien trouvé",
+ "the_box_is_empty": "La boîte est vide",
+ "top_artists": "Meilleurs artistes",
+ "top_albums": "Meilleurs albums",
+ "this_week": "Cette semaine",
+ "this_month": "Ce mois-ci",
+ "last_6_months": "Les 6 derniers mois",
+ "this_year": "Cette année",
+ "last_2_years": "Les 2 dernières années",
+ "all_time": "De tous les temps",
+ "powered_by_provider": "Propulsé par {providerName}",
+ "email": "Email",
+ "profile_followers": "Abonnés",
+ "birthday": "Anniversaire",
+ "subscription": "Abonnement",
+ "not_born": "Non né",
+ "hacker": "Hacker",
+ "profile": "Profil",
+ "no_name": "Sans nom",
+ "edit": "Modifier",
+ "user_profile": "Profil utilisateur",
+ "count_plays": "{count} lectures",
+ "streaming_fees_hypothetical": "Frais de streaming (hypothétiques)",
+ "minutes_listened": "Minutes écoutées",
+ "streamed_songs": "Morceaux diffusés",
+ "count_streams": "{count} streams",
+ "owned_by_you": "Possédé par vous",
+ "copied_shareurl_to_clipboard": "{shareUrl} copié dans le presse-papier",
+ "spotify_hipotetical_calculation": "*Cela est calculé en fonction du\npaiement par stream de Spotify de 0,003 $ à 0,005 $.\nIl s'agit d'un calcul hypothétique pour donner\nune idée de combien vous auriez\npayé aux artistes si vous aviez\nécouté leur chanson sur Spotify.",
+ "count_mins": "{minutes} minutes",
+ "summary_minutes": "minutes",
+ "summary_listened_to_music": "A écouté de la musique",
+ "summary_songs": "morceaux",
+ "summary_streamed_overall": "Diffusé en général",
+ "summary_owed_to_artists": "Dû aux artistes\nCe mois-ci",
+ "summary_artists": "artistes",
+ "summary_music_reached_you": "La musique vous a atteint",
+ "summary_full_albums": "albums complets",
+ "summary_got_your_love": "A obtenu votre amour",
+ "summary_playlists": "playlists",
+ "summary_were_on_repeat": "Était en répétition",
+ "total_money": "Total {money}",
+ "webview_not_found": "Webview non trouvé",
+ "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",
+ "unsupported_platform": "Plateforme non prise en charge"
}
\ No newline at end of file
diff --git a/lib/l10n/app_hi.arb b/lib/l10n/app_hi.arb
index 1cf62398..ce01aebe 100644
--- a/lib/l10n/app_hi.arb
+++ b/lib/l10n/app_hi.arb
@@ -286,5 +286,106 @@
"step_3_steps": "\"sp_dc\" कुकी का मूल्य कॉपी करें",
"step_4_steps": "कॉपी किए गए \"sp_dc\" मूल्य को पेस्ट करें",
"friends": "दोस्त",
- "no_lyrics_available": "क्षमा करें, इस ट्रैक के लिए गाने नहीं मिल सके"
+ "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": "ओपन कलेक्टिव पर दान करें",
+ "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गणना है जो उपयोगकर्ता को यह जानकारी देती है कि वे कितना भुगतान\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}",
+ "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 मा सुनेमा\nभुक्तान गर्नुपर्ने थियो।",
+ "webview_not_found": "वेबव्यू नहीं मिला",
+ "webview_not_found_description": "आपके डिवाइस पर वेबव्यू रनटाइम इंस्टॉल नहीं है।\nअगर इंस्टॉल है, तो सुनिश्चित करें कि यह environment PATH में है\n\nइंस्टॉल करने के बाद, ऐप को पुनः शुरू करें",
+ "unsupported_platform": "असमर्थित प्लेटफार्म"
}
\ No newline at end of file
diff --git a/lib/l10n/app_id.arb b/lib/l10n/app_id.arb
new file mode 100644
index 00000000..121695f4
--- /dev/null
+++ b/lib/l10n/app_id.arb
@@ -0,0 +1,391 @@
+{
+ "guest": "Tamu",
+ "browse": "Jelajahi",
+ "search": "Cari",
+ "library": "Pustaka",
+ "lyrics": "Lirik",
+ "settings": "Pengaturan",
+ "genre_categories_filter": "Urutkan kategori atau genre...",
+ "genre": "Genre",
+ "personalized": "Dipersonalisasi",
+ "featured": "Unggulan",
+ "new_releases": "Rilis Terbaru",
+ "songs": "Lagu",
+ "playing_track": "Memutar {track}",
+ "queue_clear_alert": "Ini akan menghapus antrian saat ini This will clear the current queue. {track_length} trek akan dihapus\nAnda ingin melanjutkan?",
+ "load_more": "Lebih Banyak",
+ "playlists": "Daftar Putar",
+ "artists": "Artis",
+ "albums": "Album",
+ "tracks": "Trek",
+ "downloads": "Unduhan",
+ "filter_playlists": "Urutkan daftar putar Anda...",
+ "liked_tracks": "Lagu Yang Disukai",
+ "liked_tracks_description": "Semua lagu yang Anda sukai",
+ "create_playlist": "Buat Daftar Putar",
+ "create_a_playlist": "Buat daftar putar",
+ "update_playlist": "Ubah daftar putar",
+ "create": "Buat",
+ "cancel": "Batal",
+ "update": "Ubah",
+ "playlist_name": "Nama Daftar Putar",
+ "name_of_playlist": "Nama daftar putar",
+ "description": "Deskripsi",
+ "public": "Publik",
+ "collaborative": "Kolaboratif",
+ "search_local_tracks": "Cari trek lokal...",
+ "play": "Putar",
+ "delete": "Hapus",
+ "none": "Tidak Ada",
+ "sort_a_z": "Urutkan berdasarkan A-Z",
+ "sort_z_a": "Urutkan berdasarkan Z-A",
+ "sort_artist": "Urutkan berdasarkan Artis",
+ "sort_album": "Urutkan berdasarkan Album",
+ "sort_duration": "Urutkan berdasarkan Durasi",
+ "sort_tracks": "Urutkan trek",
+ "currently_downloading": "Sedang Mengunduh ({tracks_length})",
+ "cancel_all": "Batalkan Semua",
+ "filter_artist": "Urutkan artis...",
+ "followers": "{followers} Pengikut",
+ "add_artist_to_blacklist": "Tambah artis ke daftar hitam",
+ "top_tracks": "Lagu Teratas",
+ "fans_also_like": "Penggemar juga menyukainya",
+ "loading": "Memuat...",
+ "artist": "Artis",
+ "blacklisted": "Masuk Daftar Hitam",
+ "following": "Mengikuti",
+ "follow": "Ikuti",
+ "artist_url_copied": "URL artis telah disalin",
+ "added_to_queue": "Menambah trek {tracks} ke antrean",
+ "filter_albums": "Urutkan album...",
+ "synced": "Disinkronkan",
+ "plain": "Normal",
+ "shuffle": "Acak",
+ "search_tracks": "Cari trek...",
+ "released": "Dirilis",
+ "error": "Kesalahan {error}",
+ "title": "Judul",
+ "time": "Waktu",
+ "more_actions": "Tindakan Lainnya",
+ "download_count": "Unduhan ({count})",
+ "add_count_to_playlist": "Menambah ({count}) ke Daftar Putar",
+ "add_count_to_queue": "Menambah ({count}) ke Antrian",
+ "play_count_next": "Mainkan ({count}) selanjutnya",
+ "album": "Album",
+ "copied_to_clipboard": "{data} telah disalin",
+ "add_to_following_playlists": "Menambah {track} ke Daftar Putar berikut",
+ "add": "Tambah",
+ "added_track_to_queue": "Menambah {track} ke antrian",
+ "add_to_queue": "Tambah ke antrian",
+ "track_will_play_next": "{track} akan diputar berikutnya",
+ "play_next": "Mainkan selanjutnya",
+ "removed_track_from_queue": "Menghapus {track} dari antrian",
+ "remove_from_queue": "Hapus dari antrian",
+ "remove_from_favorites": "Hapus dari favorit",
+ "save_as_favorite": "Simpan sebagai favorit",
+ "add_to_playlist": "Tambah ke daftar putar",
+ "remove_from_playlist": "Hapus dari daftar putar",
+ "add_to_blacklist": "Tambah ke daftar hitam",
+ "remove_from_blacklist": "Hapus dari daftar hitam",
+ "share": "Bagikan",
+ "mini_player": "Pemutar Mini",
+ "slide_to_seek": "Geser untuk maju atau mundur",
+ "shuffle_playlist": "Acak daftar putar",
+ "unshuffle_playlist": "Batalkan pengacakan daftar putar",
+ "previous_track": "Lagu sebelumnya",
+ "next_track": "Lagu berikutnya",
+ "pause_playback": "Jeda Pemutaran",
+ "resume_playback": "Lanjutkan Pemutaran",
+ "loop_track": "Ulangi Pemutaran",
+ "repeat_playlist": "Ulangi daftar putar",
+ "queue": "Antrian",
+ "alternative_track_sources": "Sumber trek alternatif",
+ "download_track": "Unduh lagu",
+ "tracks_in_queue": "{tracks} trek dalam antrian",
+ "clear_all": "Bersihkan semua",
+ "show_hide_ui_on_hover": "Tampil/Sembunyikan UI saat mengarahkan kursor",
+ "always_on_top": "Selalu di atas",
+ "exit_mini_player": "Keluar Pemutar Mini",
+ "download_location": "Lokasi unduhan",
+ "account": "Akun",
+ "login_with_spotify": "Masuk dengan Spotify",
+ "connect_with_spotify": "Hubungkan dengan Spotify",
+ "logout": "Keluar",
+ "logout_of_this_account": "Keluar dari akun",
+ "language_region": "Bahasa & Wilayah",
+ "language": "Bahasa",
+ "system_default": "Bawaan Sistem",
+ "market_place_region": "Wilayah Pasar",
+ "recommendation_country": "Negara Rekomendasi",
+ "appearance": "Tampilan",
+ "layout_mode": "Mode Tata Letak",
+ "override_layout_settings": "Ganti pengaturan mode tata letak responsif",
+ "adaptive": "Adaptif",
+ "compact": "Ringkas",
+ "extended": "Diperluas",
+ "theme": "Tema",
+ "dark": "Gelap",
+ "light": "Terang",
+ "system": "Sistem",
+ "accent_color": "Warna Aksen",
+ "sync_album_color": "Sinkronkan warna album",
+ "sync_album_color_description": "Menggunakan warna dominan sampul album sebagai warna aksen",
+ "playback": "Pemutaran",
+ "audio_quality": "Kualitas Suara",
+ "high": "Tinggi",
+ "low": "Rendah",
+ "pre_download_play": "Unduh dan putar",
+ "pre_download_play_description": "Daripada streaming audio, unduh byte dan mainkan (Direkomendasikan untuk pengguna bandwidth yang lebih tinggi)",
+ "skip_non_music": "Lewati segmen non-musik (SponsorBlock)",
+ "blacklist_description": "Lagu dan artis di daftar hitam",
+ "wait_for_download_to_finish": "Tunggu hingga unduhan saat ini selesai",
+ "desktop": "Desktop",
+ "close_behavior": "Tutup Perilaku",
+ "close": "Tutup",
+ "minimize_to_tray": "Perkecil ke tray",
+ "show_tray_icon": "Tampilkan tray ikon sistem",
+ "about": "Tentang",
+ "u_love_spotube": "Kami tahu Anda menyukai Spotube",
+ "check_for_updates": "Periksa pembaruan",
+ "about_spotube": "Tentang Spotube",
+ "blacklist": "Daftar Hitam",
+ "please_sponsor": "Silakan Sponsor/Menyumbang",
+ "spotube_description": "Spotube, klien Spotify yang ringan, lintas platform, dan gratis untuk semua",
+ "version": "Versi",
+ "build_number": "Nomor Pembuatan",
+ "founder": "Pendiri",
+ "repository": "Repositori",
+ "bug_issues": "Bug+Masalah",
+ "made_with": "Dibuat dengan ❤️ di Bangladesh🇧🇩",
+ "kingkor_roy_tirtho": "Kingkor Roy Tirtho",
+ "copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
+ "license": "Lisensi",
+ "add_spotify_credentials": "Tambahkan kredensial Spotify Anda untuk memulai",
+ "credentials_will_not_be_shared_disclaimer": "Jangan khawatir, kredensial Anda tidak akan dikumpulkan atau dibagikan kepada siapa pun",
+ "know_how_to_login": "Tidak tahu bagaimana melakukan ini?",
+ "follow_step_by_step_guide": "Ikuti panduan Langkah demi Langkah",
+ "spotify_cookie": "Spotify {name} Cookie",
+ "cookie_name_cookie": "{name} Cookie",
+ "fill_in_all_fields": "Silakan isi semua kolom",
+ "submit": "Kirim",
+ "exit": "Keluar",
+ "previous": "Sebelumnya",
+ "next": "Berikutnya",
+ "done": "Selesai",
+ "step_1": "Langkah 1",
+ "first_go_to": "Pertama, Pergi ke",
+ "login_if_not_logged_in": "dan Masuk/Daftar jika Anda belum masuk",
+ "step_2": "Langkah 2",
+ "step_2_steps": "1. Setelah Anda masuk, tekan F12 atau Klik Kanan Mouse > Buka Browser Devtools.\n2. Lalu buka Tab \"Aplikasi\" (Chrome, Edge, Brave, dll.) atau Tab \"Penyimpanan\" (Firefox, Palemoon, dll.)\n3. Buka bagian \"Cookie\" lalu subbagian \"https://accounts.spotify.com\"",
+ "step_3": "Langkah 3",
+ "step_3_steps": "Salin nilai Cookie \"sp_dc\" ",
+ "success_emoji": "Berhasil🥳",
+ "success_message": "Sekarang Anda telah berhasil Masuk dengan akun Spotify Anda. Kerja bagus, sobat!",
+ "step_4": "Langkah 4",
+ "step_4_steps": "Tempel nilai \"sp_dc\" yang disalin",
+ "something_went_wrong": "Terjadi kesalahan",
+ "piped_instance": "Piped Server Instance",
+ "piped_description": "The Piped server instance untuk digunakan sebagai pencocokan trek",
+ "piped_warning": "Beberapa di antaranya mungkin tidak berfungsi dengan baik. Jadi gunakan dengan risiko Anda sendiri",
+ "generate_playlist": "Hasilkan Daftar Putar",
+ "track_exists": "Lagu {track} sudah ada",
+ "replace_downloaded_tracks": "Ganti semua trek yang diunduh",
+ "skip_download_tracks": "Lewati pengunduhan semua trek yang diunduh",
+ "do_you_want_to_replace": "Apakah Anda ingin mengganti track yang ada?",
+ "replace": "Ganti",
+ "skip": "Lewati",
+ "select_up_to_count_type": "Pilih hingga {count} {type}",
+ "select_genres": "Pilih Genre",
+ "add_genres": "Tambah Genre",
+ "country": "Negara",
+ "number_of_tracks_generate": "Jumlah trek yang akan dihasilkan",
+ "acousticness": "Akustik",
+ "danceability": "Menari",
+ "energy": "Energi",
+ "instrumentalness": "Instrumentalitas",
+ "liveness": "Kehidupan",
+ "loudness": "Kekerasan",
+ "speechiness": "Berbicara",
+ "valence": "Valensi",
+ "popularity": "Popularitas",
+ "key": "Kunci",
+ "duration": "Durasi (s)",
+ "tempo": "Tempo (BPM)",
+ "mode": "Mode",
+ "time_signature": "Tanda Tangan Waktu",
+ "short": "Pendek",
+ "medium": "Sedang",
+ "long": "Panjang",
+ "min": "Minimal",
+ "max": "Maksimal",
+ "target": "Target",
+ "moderate": "Sedang",
+ "deselect_all": "Batalkan Semua",
+ "select_all": "Pilih Semua",
+ "are_you_sure": "Anda yakin?",
+ "generating_playlist": "Menghasilkan daftar putar khusus Anda...",
+ "selected_count_tracks": "{count} lagu yang dipilih",
+ "download_warning": "Jika Anda mengunduh semua Lagu secara massal, Anda jelas membajak Musik & menyebabkan kerusakan pada masyarakat kreatif Musik. Saya harap Anda menyadari hal ini. Selalu berusaha menghormati & mendukung kerja keras Artis",
+ "download_ip_ban_warning": "BTW, IP Anda bisa diblokir di YouTube karena permintaan unduhan yang berlebihan dari biasanya. Blokir IP berarti Anda tidak dapat menggunakan YouTube (meskipun Anda masuk) setidaknya selama 2-3 bulan dari perangkat IP tersebut. Dan Spotube tidak bertanggung jawab jika hal ini terjadi",
+ "by_clicking_accept_terms": "Dengan mengklik 'terima' Anda menyetujui ketentuan berikut:",
+ "download_agreement_1": "Saya tahu saya membajak Musik. Saya buruk",
+ "download_agreement_2": "Saya akan mendukung Artis di mana pun saya bisa dan saya melakukan ini hanya karena saya tidak punya uang untuk membeli karya seni mereka",
+ "download_agreement_3": "Saya sepenuhnya menyadari bahwa IP saya dapat diblokir di YouTube & saya tidak menganggap Spotube atau pemilik/kontributornya bertanggung jawab atas kecelakaan apa pun yang disebabkan oleh tindakan saya saat ini",
+ "decline": "Menolak",
+ "accept": "Setuju",
+ "details": "Detail",
+ "youtube": "YouTube",
+ "channel": "Channel",
+ "likes": "Suka",
+ "dislikes": "Tidak Suka",
+ "views": "Dilihat",
+ "streamUrl": "URL Stream",
+ "stop": "Berhenti",
+ "sort_newest": "Urutkan yang baru ditambah",
+ "sort_oldest": "Urutkan yang paling lama ditambah",
+ "sleep_timer": "Pengatur Waktu Tidur",
+ "mins": "{minutes} Menit",
+ "hours": "{hours} Jam",
+ "hour": "{hours} Jam",
+ "custom_hours": "Jam Kostum",
+ "logs": "Log",
+ "developers": "Pengembang",
+ "not_logged_in": "Anda belum masuk",
+ "search_mode": "Mode Pencarian",
+ "audio_source": "Sumber Suara",
+ "ok": "OK",
+ "failed_to_encrypt": "Gagal mengenkripsi",
+ "encryption_failed_warning": "Spotube menggunakan enkripsi untuk menyimpan data Anda dengan aman. Namun gagal melakukannya. Jadi itu akan kembali ke penyimpanan yang tidak aman\nJika Anda menggunakan linux, pastikan Anda telah menginstal layanan rahasia (gnome-keyring, kde-wallet, keepassxc, dll)",
+ "querying_info": "Mencari informasi...",
+ "piped_api_down": "Piped API tidak aktif",
+ "piped_down_error_instructions": "Piped Instance {pipedInstance} saat ini tidak aktif\n\nUbah instance atau ubah 'jenis API' menjadi API YouTube resmi\n\nPastikan untuk memulai ulang aplikasi setelah perubahan",
+ "you_are_offline": "Anda sedang offline",
+ "connection_restored": "Koneksi internet Anda telah pulih",
+ "use_system_title_bar": "Gunakan bilah judul sistem",
+ "crunching_results": "Mengolah hasil...",
+ "search_to_get_results": "Cari untuk mendapatkan hasil",
+ "use_amoled_mode": "Tema gelap gulita",
+ "pitch_dark_theme": "Mode AMOLED",
+ "normalize_audio": "Normalisasi audio",
+ "change_cover": "Ganti sampul",
+ "add_cover": "Tambah sampul",
+ "restore_defaults": "Kembalikan semula",
+ "download_music_codec": "Unduh codec musik",
+ "streaming_music_codec": "Streaming codec musik",
+ "login_with_lastfm": "Masuk dengan Last.fm",
+ "connect": "Hubungkan",
+ "disconnect_lastfm": "Memutuskan Last.fm",
+ "disconnect": "Memutuskan",
+ "username": "Username",
+ "password": "Password",
+ "login": "Masuk",
+ "login_with_your_lastfm": "Masuk dengan Last.fm Anda",
+ "scrobble_to_lastfm": "Scrobble ke Last.fm",
+ "go_to_album": "Pergi ke Album",
+ "discord_rich_presence": "Discord Rich Presence",
+ "browse_all": "Lihat Semua",
+ "genres": "Genre",
+ "explore_genres": "Jelajahi Genre",
+ "friends": "Daftar Teman",
+ "no_lyrics_available": "Maaf, tidak dapat menemukan lirik untuk lagu ini",
+ "start_a_radio": "Putar Radio",
+ "how_to_start_radio": "Bagaimana Anda ingin memutar radio?",
+ "replace_queue_question": "Apakah Anda ingin mengganti antrean saat ini atau menambahkannya?",
+ "endless_playback": "Pemutaran Tanpa Akhir",
+ "delete_playlist": "Hapus Daftar Putar",
+ "delete_playlist_confirmation": "Anda yakin ingin menghapus daftar putar ini?",
+ "local_tracks": "Trek Lokal",
+ "song_link": "Tautan Lagu",
+ "skip_this_nonsense": "Lewati omong kosong ini",
+ "freedom_of_music": "“Kebebasan Musik”",
+ "freedom_of_music_palm": "“Kebebasan Musik di telapak tangan Anda”",
+ "get_started": "Mari kita mulai",
+ "youtube_source_description": "Direkomendasikan dan berfungsi paling baik.",
+ "piped_source_description": "Merasa bebas? Sama seperti YouTube tetapi banyak yang gratis.",
+ "jiosaavn_source_description": "Terbaik untuk wilayah Asia Selatan.",
+ "highest_quality": "Kualitas Terbaik: {quality}",
+ "select_audio_source": "Pilih Sumber Suara",
+ "endless_playback_description": "Tambahkan lagu baru secara otomatis\nke akhir antrean",
+ "choose_your_region": "Pilih wilayah Anda",
+ "choose_your_region_description": "Ini akan membantu Spotube menampilkan konten yang tepat\nuntuk lokasi Anda.",
+ "choose_your_language": "Pilih bahasa Anda",
+ "help_project_grow": "Bantu proyek ini berkembang",
+ "help_project_grow_description": "Spotube adalah proyek sumber terbuka. Anda dapat membantu proyek ini berkembang dengan berkontribusi pada proyek, melaporkan bug, atau menyarankan fitur baru.",
+ "contribute_on_github": "Berkontribusi di GitHub",
+ "donate_on_open_collective": "Donasi di Open Collective",
+ "browse_anonymously": "Jelajahi Secara Anonim",
+ "enable_connect": "Aktifkan Hubungkan",
+ "enable_connect_description": "Kontrol Spotube dari perangkat lain",
+ "devices": "Perangkat",
+ "select": "Pilih",
+ "connect_client_alert": "Anda dikendalikan oleh {client}",
+ "this_device": "Perangkat Ini",
+ "remote": "Remot",
+ "local_library": "Perpustakaan lokal",
+ "add_library_location": "Tambahkan ke perpustakaan",
+ "remove_library_location": "Hapus dari perpustakaan",
+ "local_tab": "Lokal",
+ "stats": "Statistik",
+ "and_n_more": "dan {count} lainnya",
+ "recently_played": "Baru saja diputar",
+ "browse_more": "Telusuri lebih banyak",
+ "no_title": "Tanpa judul",
+ "not_playing": "Tidak diputar",
+ "epic_failure": "Kegagalan epik!",
+ "added_num_tracks_to_queue": "Menambahkan {tracks_length} trek ke antrean",
+ "spotube_has_an_update": "Spotube memiliki pembaruan",
+ "download_now": "Unduh sekarang",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} telah dirilis",
+ "release_version": "Spotube v{version} telah dirilis",
+ "read_the_latest": "Baca yang terbaru ",
+ "release_notes": "catatan rilis",
+ "pick_color_scheme": "Pilih skema warna",
+ "save": "Simpan",
+ "choose_the_device": "Pilih perangkat:",
+ "multiple_device_connected": "Beberapa perangkat terhubung.\nPilih perangkat tempat Anda ingin melakukan tindakan ini",
+ "nothing_found": "Tidak ditemukan apa pun",
+ "the_box_is_empty": "Kotak kosong",
+ "top_artists": "Artis Teratas",
+ "top_albums": "Album Teratas",
+ "this_week": "Minggu ini",
+ "this_month": "Bulan ini",
+ "last_6_months": "6 bulan terakhir",
+ "this_year": "Tahun ini",
+ "last_2_years": "2 tahun terakhir",
+ "all_time": "Sepanjang waktu",
+ "powered_by_provider": "Didukung oleh {providerName}",
+ "email": "Email",
+ "profile_followers": "Pengikut",
+ "birthday": "Ulang Tahun",
+ "subscription": "Langganan",
+ "not_born": "Belum lahir",
+ "hacker": "Hacker",
+ "profile": "Profil",
+ "no_name": "Tanpa nama",
+ "edit": "Edit",
+ "user_profile": "Profil pengguna",
+ "count_plays": "{count} pemutaran",
+ "streaming_fees_hypothetical": "Biaya streaming (hipotetis)",
+ "minutes_listened": "Menit didengarkan",
+ "streamed_songs": "Lagu yang disiarkan",
+ "count_streams": "{count} streams",
+ "owned_by_you": "Dimiliki oleh Anda",
+ "copied_shareurl_to_clipboard": "{shareUrl} disalin ke clipboard",
+ "spotify_hipotetical_calculation": "*Ini dihitung berdasarkan pembayaran\nper stream Spotify dari $0,003 hingga $0,005.\nIni adalah perhitungan hipotetis untuk memberi\npengguna gambaran tentang berapa banyak\nmereka akan membayar kepada artis jika\nmereka mendengarkan lagu mereka di Spotify.",
+ "count_mins": "{minutes} menit",
+ "summary_minutes": "menit",
+ "summary_listened_to_music": "Mendengarkan musik",
+ "summary_songs": "lagu",
+ "summary_streamed_overall": "Disiarkan secara keseluruhan",
+ "summary_owed_to_artists": "Terhutang kepada artis\nBulan ini",
+ "summary_artists": "artis",
+ "summary_music_reached_you": "Musik mencapai Anda",
+ "summary_full_albums": "album lengkap",
+ "summary_got_your_love": "Mendapatkan cinta Anda",
+ "summary_playlists": "daftar putar",
+ "summary_were_on_repeat": "Sedang diulang",
+ "total_money": "Total {money}",
+ "webview_not_found": "Webview tidak ditemukan",
+ "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",
+ "unsupported_platform": "Platform tidak didukung"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_it.arb b/lib/l10n/app_it.arb
index ec76b914..3a2c57c3 100644
--- a/lib/l10n/app_it.arb
+++ b/lib/l10n/app_it.arb
@@ -287,5 +287,106 @@
"step_3_steps": "Copia il valore del cookie \"sp_dc\"",
"step_4_steps": "Incolla il valore copiato di \"sp_dc\"",
"friends": "Amici",
- "no_lyrics_available": "Spiacente, impossibile trovare il testo di questa traccia"
+ "no_lyrics_available": "Spiacente, impossibile trovare il testo di questa traccia",
+ "sort_duration": "Ordina per Durata",
+ "start_a_radio": "Avvia una Radio",
+ "how_to_start_radio": "Come vuoi avviare la radio?",
+ "replace_queue_question": "Vuoi sostituire la coda attuale o aggiungerla?",
+ "endless_playback": "Riproduzione Infinita",
+ "delete_playlist": "Elimina Playlist",
+ "delete_playlist_confirmation": "Sei sicuro di voler eliminare questa playlist?",
+ "local_tracks": "Tracce Locali",
+ "song_link": "Link della Canzone",
+ "skip_this_nonsense": "Salta questa sciocchezza",
+ "freedom_of_music": "“Libertà della Musica”",
+ "freedom_of_music_palm": "“Libertà della Musica nel palmo della tua mano”",
+ "get_started": "Cominciamo",
+ "youtube_source_description": "Consigliato e funziona meglio.",
+ "piped_source_description": "Ti senti libero? Come YouTube ma molto più gratuito.",
+ "jiosaavn_source_description": "Il migliore per la regione dell'Asia meridionale.",
+ "highest_quality": "Massima Qualità: {quality}",
+ "select_audio_source": "Seleziona Sorgente Audio",
+ "endless_playback_description": "Aggiungi automaticamente nuove canzoni alla fine della coda",
+ "choose_your_region": "Scegli la tua regione",
+ "choose_your_region_description": "Questo aiuterà Spotube a mostrarti il contenuto giusto per la tua posizione.",
+ "choose_your_language": "Scegli la tua lingua",
+ "help_project_grow": "Aiuta questo progetto a crescere",
+ "help_project_grow_description": "Spotube è un progetto open-source. Puoi aiutare questo progetto a crescere contribuendo al progetto, segnalando bug o suggerendo nuove funzionalità.",
+ "contribute_on_github": "Contribuisci su GitHub",
+ "donate_on_open_collective": "Dona su Open Collective",
+ "browse_anonymously": "Naviga in modo anonimo",
+ "enable_connect": "Abilita connessione",
+ "enable_connect_description": "Controlla Spotube da altri dispositivi",
+ "devices": "Dispositivi",
+ "select": "Seleziona",
+ "connect_client_alert": "Stai venendo controllato da {client}",
+ "this_device": "Questo dispositivo",
+ "remote": "Remoto",
+ "local_library": "Biblioteca locale",
+ "add_library_location": "Aggiungi alla biblioteca",
+ "remove_library_location": "Rimuovi dalla biblioteca",
+ "local_tab": "Locale",
+ "stats": "Statistiche",
+ "and_n_more": "e {count} in più",
+ "recently_played": "Riprodotti di recente",
+ "browse_more": "Esplora di più",
+ "no_title": "Nessun titolo",
+ "not_playing": "Non in riproduzione",
+ "epic_failure": "Fallimento epico!",
+ "added_num_tracks_to_queue": "Aggiunti {tracks_length} brani alla coda",
+ "spotube_has_an_update": "Spotube ha un aggiornamento",
+ "download_now": "Scarica ora",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} è stato rilasciato",
+ "release_version": "Spotube v{version} è stato rilasciato",
+ "read_the_latest": "Leggi l'ultimo ",
+ "release_notes": "note di rilascio",
+ "pick_color_scheme": "Scegli uno schema di colori",
+ "save": "Salva",
+ "choose_the_device": "Scegli il dispositivo:",
+ "multiple_device_connected": "Sono collegati più dispositivi.\nScegli il dispositivo su cui vuoi che venga eseguita questa azione",
+ "nothing_found": "Nessun risultato",
+ "the_box_is_empty": "La scatola è vuota",
+ "top_artists": "Artisti Top",
+ "top_albums": "Album Top",
+ "this_week": "Questa settimana",
+ "this_month": "Questo mese",
+ "last_6_months": "Ultimi 6 mesi",
+ "this_year": "Quest'anno",
+ "last_2_years": "Ultimi 2 anni",
+ "all_time": "Di tutti i tempi",
+ "powered_by_provider": "Sostenuto da {providerName}",
+ "email": "Email",
+ "profile_followers": "Follower",
+ "birthday": "Compleanno",
+ "subscription": "Abbonamento",
+ "not_born": "Non nato",
+ "hacker": "Hacker",
+ "profile": "Profilo",
+ "no_name": "Nessun nome",
+ "edit": "Modifica",
+ "user_profile": "Profilo utente",
+ "count_plays": "{count} riproduzioni",
+ "streaming_fees_hypothetical": "Spese di streaming (ipotetico)",
+ "minutes_listened": "Minuti ascoltati",
+ "streamed_songs": "Brani in streaming",
+ "count_streams": "{count} streaming",
+ "owned_by_you": "Di tua proprietà",
+ "copied_shareurl_to_clipboard": "Copiato {shareUrl} negli appunti",
+ "spotify_hipotetical_calculation": "*Questo è calcolato in base al pagamento per streaming di Spotify\nche va da $0.003 a $0.005. Questo è un calcolo ipotetico\nper dare all'utente un'idea di quanto avrebbe pagato agli artisti se avesse ascoltato\ne loro canzoni su Spotify.",
+ "count_mins": "{minutes} min",
+ "summary_minutes": "minuti",
+ "summary_listened_to_music": "Musica ascoltata",
+ "summary_songs": "brani",
+ "summary_streamed_overall": "Streaming complessivo",
+ "summary_owed_to_artists": "Dovuto agli artisti\nquesto mese",
+ "summary_artists": "dell'artista",
+ "summary_music_reached_you": "La musica ti ha raggiunto",
+ "summary_full_albums": "album completi",
+ "summary_got_your_love": "Ha ricevuto il tuo amore",
+ "summary_playlists": "playlist",
+ "summary_were_on_repeat": "Erano in ripetizione",
+ "total_money": "Totale {money}",
+ "webview_not_found": "Webview non trovato",
+ "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",
+ "unsupported_platform": "Piattaforma non supportata"
}
\ No newline at end of file
diff --git a/lib/l10n/app_ja.arb b/lib/l10n/app_ja.arb
index d16708d7..ed779478 100644
--- a/lib/l10n/app_ja.arb
+++ b/lib/l10n/app_ja.arb
@@ -286,5 +286,106 @@
"step_3_steps": "\"sp_dc\" Cookieの値をコピー",
"step_4_steps": "コピーした\"sp_dc\"の値を貼り付け",
"friends": "友達",
- "no_lyrics_available": "申し訳ありませんが、このトラックの歌詞を見つけることができません"
+ "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ランタイムがインストールされていません。\nインストールされている場合は、environment PATHにあることを確認してください\n\nインストール後、アプリを再起動してください",
+ "unsupported_platform": "サポートされていないプラットフォーム"
}
\ No newline at end of file
diff --git a/lib/l10n/app_ka.arb b/lib/l10n/app_ka.arb
new file mode 100644
index 00000000..888dbb6f
--- /dev/null
+++ b/lib/l10n/app_ka.arb
@@ -0,0 +1,391 @@
+{
+ "guest": "სტუმარი",
+ "browse": "ნახვა",
+ "search": "ძებნა",
+ "library": "ბიბლიოთეკა",
+ "lyrics": "ტექსტები",
+ "settings": "კონფიგურაციები",
+ "genre_categories_filter": "კატეგორიების ან ჟანრების ფილტრი...",
+ "genre": "ჟანრი",
+ "personalized": "პეერსონალიზებული",
+ "featured": "გამორჩეული",
+ "new_releases": "ახალი გამოცემები",
+ "songs": "სიმღერები",
+ "playing_track": "უკრავს {track}",
+ "queue_clear_alert": "ეს გაასუფთავებს მიმდინარე რიგს. {track_length} ტრეკი წაიშლება\nᲒინდა გააგრძელო?",
+ "load_more": "მეტის ჩატვირთვა",
+ "playlists": "ფლეილისტები",
+ "artists": "არტისტები",
+ "albums": "ალბომები",
+ "tracks": "ტრეკები",
+ "downloads": "ჩამოტვირთვები",
+ "filter_playlists": "ფლეილისტების გაფილტვრა...",
+ "liked_tracks": "მოწონებული ტრეკები",
+ "liked_tracks_description": "ყველა შენი მოწონებული ტრეკი",
+ "create_playlist": "ფლეილისტის შექმნა",
+ "create_a_playlist": "ფლეილისტის შექმნა",
+ "update_playlist": "ფლეილისტის განახლება",
+ "create": "შექმნა",
+ "cancel": "გაუქმება",
+ "update": "განახლება",
+ "playlist_name": "ფლეილისტის სახელი",
+ "name_of_playlist": "ფლეილისტის სახელი",
+ "description": "აღწერა",
+ "public": "საჯარო",
+ "collaborative": "კოლაბორაციული",
+ "search_local_tracks": "ლოცალური ტრეკების ძებნა...",
+ "play": "დაკვრა",
+ "delete": "წაშლა",
+ "none": "არცერთი",
+ "sort_a_z": "დალაგება A-Z-ს მიხედვით",
+ "sort_z_a": "დალაგება Z-A-ს მიხედვით",
+ "sort_artist": "დალაგება არტისტის მიხედვით",
+ "sort_album": "დალაგება ალბომის მიხედვით",
+ "sort_duration": "დალაგება ხანგრძლივობის მიხედვით",
+ "sort_tracks": "ტრეკების დალაგება",
+ "currently_downloading": "მიმდინარეობს ჩამოტვირთვა ({tracks_length})",
+ "cancel_all": "ყველას გაუქმება",
+ "filter_artist": "არტისტების ფილტრი...",
+ "followers": "{followers} ფოლოვერები",
+ "add_artist_to_blacklist": "არტისტის შავ სიაში დამატება",
+ "top_tracks": "ტოპ ტრეკები",
+ "fans_also_like": "ფანებს ასევე მოსწონთ",
+ "loading": "იტვირთება...",
+ "artist": "არტისტი",
+ "blacklisted": "შავ სიაში მყოფი",
+ "following": "ფოლოვინგი",
+ "follow": "დაფოლოვება",
+ "artist_url_copied": "არტისტის ლინკი დაკოპირებულია",
+ "added_to_queue": "{tracks} ტრეკი დაემატა რიგში",
+ "filter_albums": "ალბომების გაფილტვრა...",
+ "synced": "სინქრონიზებული",
+ "plain": "Plain",
+ "shuffle": "რიგის არევა",
+ "search_tracks": "ტრეკების ძებნა...",
+ "released": "გამოშვებული",
+ "error": "შეცდომა {error}",
+ "title": "სათაური",
+ "time": "დრო",
+ "more_actions": "მეტი მოქმედებები",
+ "download_count": "გადმოწერა ({count})",
+ "add_count_to_playlist": "ფლეილისტში ({count})-ის დამატება",
+ "add_count_to_queue": "რიგში ({count})-ის დამატება",
+ "play_count_next": "შემდეგი ({count})-ის დაკვრა",
+ "album": "ალბომი",
+ "copied_to_clipboard": "{data} დაკოპირებულია",
+ "add_to_following_playlists": "დაამატე {track} ამ ფლეილისტებში",
+ "add": "დამატება",
+ "added_track_to_queue": "რიგში დაემატა {track}",
+ "add_to_queue": "რიგში დამატება",
+ "track_will_play_next": "{track} დაუკრავს შემდეგს",
+ "play_next": "შემდეგის დაკვრა",
+ "removed_track_from_queue": "რიგიდან წაიშალა {track}",
+ "remove_from_queue": "რიგიდან წაშლა",
+ "remove_from_favorites": "ფავორიტებიდან წაშლა",
+ "save_as_favorite": "ფავორიტებში დამატება",
+ "add_to_playlist": "ფლეილისტში დამატება",
+ "remove_from_playlist": "ფლეილისტიდან წაშლა",
+ "add_to_blacklist": "შავ სიაში დამატება",
+ "remove_from_blacklist": "შავი სიიდან წაშლა",
+ "share": "გაზიარება",
+ "mini_player": "მინი დამკვრელი",
+ "slide_to_seek": "გადახვევისთვის გაასრიალეთ წინ ან უკან",
+ "shuffle_playlist": "ფლეილისტის არევა",
+ "unshuffle_playlist": "ფლეილისტის დალაგება",
+ "previous_track": "წინა ტრეკი",
+ "next_track": "შემდეგი ტრეკი",
+ "pause_playback": "დაკვრის გაჩერება",
+ "resume_playback": "დაკვრის გაგრძელება",
+ "loop_track": "ტრეკის ლუპზე დაკვრა",
+ "repeat_playlist": "ფლეილისტის გამეორება",
+ "queue": "რიგი",
+ "alternative_track_sources": "ალტერნატიული ტრეკების წყაროები",
+ "download_track": "გადმოწერე ტრეკი",
+ "tracks_in_queue": "{tracks} ტრეკი რიგში",
+ "clear_all": "ყველას წაშლა",
+ "show_hide_ui_on_hover": "UI-ის ჩვენება/დამალვა ჰოვერზე",
+ "always_on_top": "ტოველთვის ზემოდან",
+ "exit_mini_player": "მინი დამკვრელიდან გამოსვლა",
+ "download_location": "ჩამოტვირთვის მდებარეობა",
+ "account": "ანგარიში",
+ "login_with_spotify": "შედით თქვენი Spotify ანგარიშით",
+ "connect_with_spotify": "დაუკავშირდით Spotify-ს",
+ "logout": "გასვლა",
+ "logout_of_this_account": "ანგარიშიდან გასვლა",
+ "language_region": "ენა და რეგიონი",
+ "language": "ენა",
+ "system_default": "სისტემის ნაგულისხმევი",
+ "market_place_region": "მარკეტფლეისის რეგიონი",
+ "recommendation_country": "რეკომენდირებული ქვეყანა",
+ "appearance": "გარეგნობა",
+ "layout_mode": "განლაგების რეჟიმი",
+ "override_layout_settings": "რესფონსივ განლაგების რეჟიმის კონფიგურაციაზე გადაწერა",
+ "adaptive": "ადაპტირებული",
+ "compact": "კომპაქტური",
+ "extended": "გაფართოებული",
+ "theme": "თემა",
+ "dark": "ბნელი",
+ "light": "ღია",
+ "system": "სისტემის",
+ "accent_color": "აქცენტის ფერი",
+ "sync_album_color": "ალბომის ფერის სინქრონიზაცია",
+ "sync_album_color_description": "დომინანტური ალბომის ფერის აქცენტის ფერად გამოყენება",
+ "playback": "დაკვრა",
+ "audio_quality": "აუდიოს ხარისხი",
+ "high": "მაღალი",
+ "low": "დაბალი",
+ "pre_download_play": "წინასწარ ჩამოტვირთვა და დაკვრა",
+ "pre_download_play_description": "აუდიოს სტრიმინგის ნაცვლად, ბაიტების ჩამოტვირთვა და დაკვრა (რეკომენდებულია უფრო მაღალი გამტარუნარიანობის მომხმარებლებისთვის)",
+ "skip_non_music": "არა მუსიკალური ნაწილის გამოტოვება (სპონსორის ბლოკი)",
+ "blacklist_description": "შავ სიაში მყოფი არტისტები და ტრეკები",
+ "wait_for_download_to_finish": "გთხოვთ, დაელოდოთ მიმდინარე ჩამოტვირთვის დასრულებას",
+ "desktop": "დესკტოპი",
+ "close_behavior": "დახურვის ქცევა",
+ "close": "დახურვა",
+ "minimize_to_tray": "მინიმიზაცია",
+ "show_tray_icon": "სისტემის აიკონის ჩვენება",
+ "about": "ჩვენს შესახებ",
+ "u_love_spotube": "We know you love Spotube",
+ "check_for_updates": "განახლებების შემოწმება",
+ "about_spotube": "Spotube-ს შესახებ",
+ "blacklist": "შავი სია",
+ "please_sponsor": "გთხოვთ დაგვასპონსოროთ",
+ "spotube_description": "Spotube, a lightweight, cross-platform, free-for-all spotify client",
+ "version": "ვერსია",
+ "build_number": "Build Number",
+ "founder": "დამფუძნებელი",
+ "repository": "რეპოზიტორია",
+ "bug_issues": "Bug+Issues",
+ "made_with": "Made with ❤️ in Bangladesh🇧🇩",
+ "kingkor_roy_tirtho": "Kingkor Roy Tirtho",
+ "copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
+ "license": "ლიცენზია",
+ "add_spotify_credentials": "დასაწყებად დაამატეთ თქვენი Spotify მონაცემები",
+ "credentials_will_not_be_shared_disclaimer": "არ ინერვიულოთ, თქვენი მონაცემები არ იქნება შეგროვებული ან გაზიარებული ვინმესთან",
+ "know_how_to_login": "არ იცით როგორ გააკეთოთ ეს?",
+ "follow_step_by_step_guide": "მიჰყევით ნაბიჯ-ნაბიჯ სახელმძღვანელოს",
+ "spotify_cookie": "Spotify {name} ქუქი",
+ "cookie_name_cookie": "{name} ქუქი",
+ "fill_in_all_fields": "გთხოვთ შეავსოთ ყველა ველი",
+ "submit": "გაგზავნა",
+ "exit": "გამოსვლა",
+ "previous": "წინა",
+ "next": "შემდეგი",
+ "done": "მზადაა",
+ "step_1": "ნაბიჯი 1",
+ "first_go_to": "პირველი, გადადით",
+ "login_if_not_logged_in": "და შესვლა/რეგისტრაცია, თუ არ ხართ შესული",
+ "step_2": "ნაბიჯი 2",
+ "step_2_steps": "1. როცა შეხვალთ, დააჭირეთ F12-ს ან მაუსის მარჯვენა ღილაკს > Inspect to Open the Browser devtools.\n2. შემდეგ გახსენით \"Application\" განყოფილება (Chrome, Edge, Brave etc..) ან \"Storage\" განყოფილება (Firefox, Palemoon etc..)\n3. შედით \"Cookies\" სექციაში და შემდეგ \"https://accounts.spotify.com\" სუბსექციაში",
+ "step_3": "ნაბიჯი 3",
+ "step_3_steps": "დააკოპირეთ \"sp_dc\" ქუქი-ფაილის მნიშვნელობა",
+ "success_emoji": "წარმატება🥳",
+ "success_message": "თქვენ წარმატებით შეხვედით თქვენი Spotify ანგარიშით.",
+ "step_4": "ნაბიჯი 4",
+ "step_4_steps": "ჩასვით კოპირებული \"sp_dc\" მნიშვნელობა",
+ "something_went_wrong": "Რაღაც არასწორად წავიდა",
+ "piped_instance": "Piped Server Instance",
+ "piped_description": "The Piped server instance to use for track matching",
+ "piped_warning": "ზოგიერთი მათგანმა შეიძლება კარგად არ იმუშაოს. ",
+ "generate_playlist": "ფლეილისტის დაგენერირება",
+ "track_exists": "ტრეკი {track} უკვე არსებობს",
+ "replace_downloaded_tracks": "ყველა ჩამოტვირთული ტრეკის შეცვლა",
+ "skip_download_tracks": "ყველა ჩამოტვირთული ტრეკის გამოტოვება",
+ "do_you_want_to_replace": "გსურთ შეცვალოთ არსებული ტრეკი??",
+ "replace": "შეცვლა",
+ "skip": "გამოტოვება",
+ "select_up_to_count_type": "აირჩიე {count}-მდე {type}",
+ "select_genres": "ჟანრების არჩევა",
+ "add_genres": "ჟანრების დამატება",
+ "country": "ქვეყანა",
+ "number_of_tracks_generate": "დასაგენერირებელი ტრეკების რაოდენობა",
+ "acousticness": "Acousticness",
+ "danceability": "Danceability",
+ "energy": "Energy",
+ "instrumentalness": "Instrumentalness",
+ "liveness": "Liveness",
+ "loudness": "Loudness",
+ "speechiness": "Speechiness",
+ "valence": "Valence",
+ "popularity": "Popularity",
+ "key": "Key",
+ "duration": "Duration (s)",
+ "tempo": "Tempo (BPM)",
+ "mode": "Mode",
+ "time_signature": "Time Signature",
+ "short": "Short",
+ "medium": "საშუალო",
+ "long": "გრძელი",
+ "min": "მინიმალური",
+ "max": "მაქსიმალური",
+ "target": "სამიზნე",
+ "moderate": "საშუალო",
+ "deselect_all": "ყველა მონიშვნის გაუქმება",
+ "select_all": "ყველას მონიშვნა",
+ "are_you_sure": "Დარწმუნებული ხართ?",
+ "generating_playlist": "მიმდინარეობს თქვენი მორგებული ფლეილისტის გენერირება...",
+ "selected_count_tracks": "არჩეულია {count} ტრეკი",
+ "download_warning": "If you download all Tracks at bulk you're clearly pirating Music & causing damage to the creative society of Music. I hope you are aware of this. Always, try respecting & supporting Artist's hard work",
+ "download_ip_ban_warning": "BTW, your IP can get blocked on YouTube due excessive download requests than usual. IP block means you can't use YouTube (even if you're logged in) for at least 2-3 months from that IP device. And Spotube doesn't hold any responsibility if this ever happens",
+ "by_clicking_accept_terms": "By clicking 'accept' you agree to following terms:",
+ "download_agreement_1": "I know I'm pirating Music. I'm bad",
+ "download_agreement_2": "I'll support the Artist wherever I can and I'm only doing this because I don't have money to buy their art",
+ "download_agreement_3": "I'm completely aware that my IP can get blocked on YouTube & I don't hold Spotube or his owners/contributors responsible for any accidents caused by my current action",
+ "decline": "უარყოფა",
+ "accept": "დათანხმება",
+ "details": "დეტალები",
+ "youtube": "YouTube",
+ "channel": "Channel",
+ "likes": "მოწონებები",
+ "dislikes": "არ მოწონებები",
+ "views": "ნახვები",
+ "streamUrl": "სტრიმის ლინკი",
+ "stop": "გაჩერება",
+ "sort_newest": "ფალაგება სიახლის მიხედიტ",
+ "sort_oldest": "დალაგება სიძველის მიხედვით",
+ "sleep_timer": "ძილის ტაიმერი",
+ "mins": "{minutes} წუთი",
+ "hours": "{hours} საათი",
+ "hour": "{hours} საათი",
+ "custom_hours": "მორგებული საათები",
+ "logs": "ლოგები",
+ "developers": "დეველოპერები",
+ "not_logged_in": "არ ხარ დალოგინებული",
+ "search_mode": "ძებნის რეჟიმი",
+ "audio_source": "აუდიოს წყარო",
+ "ok": "ოკ",
+ "failed_to_encrypt": "დაშიფვრა ვერ მოხერხდა",
+ "encryption_failed_warning": "Spotube uses encryption to securely store your data. But failed to do so. So it'll fallback to insecure storage\nIf you're using linux, please make sure you've any secret-service (gnome-keyring, kde-wallet, keepassxc etc) installed",
+ "querying_info": "Querying info...",
+ "piped_api_down": "Piped API is down",
+ "piped_down_error_instructions": "The Piped instance {pipedInstance} is currently down\n\nEither change the instance or change the 'API type' to official YouTube API\n\nMake sure to restart the app after change",
+ "you_are_offline": "ამჟამად ხაზგარეშე ხართ",
+ "connection_restored": "თქვენი ინტერნეტ კავშირი აღდგა",
+ "use_system_title_bar": "სისტემის სათაურის ზოლის გამოყენება",
+ "crunching_results": "იტვირთება შედეგები...",
+ "search_to_get_results": "მოძებნეთ შედეგების მისაღებად",
+ "use_amoled_mode": "Pitch black dark theme",
+ "pitch_dark_theme": "AMOLED Mode",
+ "normalize_audio": "აუდიოს ნორმალიზება",
+ "change_cover": "Ქავერის შეცვლა",
+ "add_cover": "Ქავერის ფოტოს დამატება",
+ "restore_defaults": "ნაგულისხმევი პარამეტრების აღდგენა",
+ "download_music_codec": "მუსიკის კოდეკის გადმოწერა",
+ "streaming_music_codec": "სტრიმინგ მუსიკის კოდეკი",
+ "login_with_lastfm": "Last.fm-ით შესვლა",
+ "connect": "დაკავშირება",
+ "disconnect_lastfm": "Last.fm-იდან გამოსვლა",
+ "disconnect": "გამოსვლა",
+ "username": "მომხმარებელი",
+ "password": "პაროლი",
+ "login": "შესვლა",
+ "login_with_your_lastfm": "Last.fm ანგარიშით შესვლა",
+ "scrobble_to_lastfm": "Scrobble to Last.fm",
+ "go_to_album": "ალბომზე გადასვლა",
+ "discord_rich_presence": "Discord Rich Presence",
+ "browse_all": "ყველას ნახვა",
+ "genres": "ჟანრები",
+ "explore_genres": "შეისწავლეთ ჟანრები",
+ "friends": "მეგობრები",
+ "no_lyrics_available": "უკაცრავად, ამ ტრეკისთვის ტექსტის პოვნა შეუძლებელია",
+ "start_a_radio": "რადიოს ჩართვა",
+ "how_to_start_radio": "როგორ გნებავთ რადიოს ჩართვა?",
+ "replace_queue_question": "გნებავთ ჩაანაცვლოთ არსებული რიგი თუ დაამატოთ მასზე?",
+ "endless_playback": "დაუსრულებელი დაკვრა",
+ "delete_playlist": "ფლეილისტის წაშლა",
+ "delete_playlist_confirmation": "დარწმუნებული ხართ რომ გნებავთ ფლეილისტის წაშლა?",
+ "local_tracks": "ლოკალური ტრეკები",
+ "song_link": "ტრეკის ლინკი",
+ "skip_this_nonsense": "ამ სისულელის გამოტოვება",
+ "freedom_of_music": "“მუსიკის თავისუფლება”",
+ "freedom_of_music_palm": "“მუსიკის თავისუფლება შენს ხელის გულზე”",
+ "get_started": "დავიწყოთ",
+ "youtube_source_description": "რეკომენდებულია და მუშაობს საუკეთესოდ.",
+ "piped_source_description": "თავისუფლად გრძნობთ თავს? იგივეა, რაც YouTube, მაგრამ ბევრი თავისუფალი.",
+ "jiosaavn_source_description": "საუკეთესოა სამხრეთ აზიის რეგიონისთვის.",
+ "highest_quality": "საუკეთესო ხარისხი: {quality}",
+ "select_audio_source": "აუდიოს წყაროს არჩევა",
+ "endless_playback_description": "ახალი სიმთერების ავტომატურად რიგის ბოლოში დამატება",
+ "choose_your_region": "აირჩიე შენი რეგიონი",
+ "choose_your_region_description": "This will help Spotube show you the right content\nfor your location.",
+ "choose_your_language": "აირჩიე ენა",
+ "help_project_grow": "დაეხმარეთ ამ პროექტს განვითარებაში",
+ "help_project_grow_description": "Spotube is an open-source project. You can help this project grow by contributing to the project, reporting bugs, or suggesting new features.",
+ "contribute_on_github": "GitHub-ზე კონტრიბუცია",
+ "donate_on_open_collective": "Open Collective-ზე დონაცია",
+ "browse_anonymously": "ანონიმურად ნახვა",
+ "enable_connect": "დაკავშირების ჩართვა",
+ "enable_connect_description": "აკონტროლე Spotube სხვა მოწყობილობებიდან",
+ "devices": "მოწყობილობები",
+ "select": "არჩევა",
+ "connect_client_alert": "თქვენ კონტროლირებული ხართ {client} მოწყობილობით",
+ "this_device": "ეს მოწყობილობა",
+ "remote": "დისტანციური",
+ "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-ს ტრეკებს.",
+ "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}",
+ "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-ზე.",
+ "webview_not_found": "ვებვიუ ვერ მოიძებნა",
+ "webview_not_found_description": "თქვენს მოწყობილობაზე ვებვიუის შესრულების დრო არ არის დაყენებული.\nთუ დაყენებულია, დარწმუნდით, რომ ის environment PATH-შია\n\nდაყენების შემდეგ, გადატვირთეთ აპი",
+ "unsupported_platform": "მოუხერხებელი პლატფორმა"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_ko.arb b/lib/l10n/app_ko.arb
index dac5b72a..a71b59ae 100644
--- a/lib/l10n/app_ko.arb
+++ b/lib/l10n/app_ko.arb
@@ -287,5 +287,106 @@
"step_4_steps": "복사한 \"sp_dc\"값을 붙여넣기",
"friends": "친구",
"no_lyrics_available": "죄송하지만 이 곡의 가사를 찾지 못했습니다",
- "@@locale": "ko"
-}
+ "@@locale": "ko",
+ "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아티스트에게 지불했을 금액에 대한 통찰을 제공하기 위한\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의 스트림당 지불금 $0.003에서 $0.005까지의\n기준으로 계산되었습니다. 이는 사용자가 Spotify에서\n곡을 들을 때 아티스트에게 얼마를 지불했을지를\n알려주기 위한 가상의 계산입니다.",
+ "webview_not_found": "웹뷰를 찾을 수 없음",
+ "webview_not_found_description": "기기에 웹뷰 런타임이 설치되지 않았습니다.\n설치되어 있으면 environment PATH에 있는지 확인하십시오\n\n설치 후 앱을 다시 시작하세요",
+ "unsupported_platform": "지원되지 않는 플랫폼"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_ne.arb b/lib/l10n/app_ne.arb
index 2d20fc9c..9bcfebad 100644
--- a/lib/l10n/app_ne.arb
+++ b/lib/l10n/app_ne.arb
@@ -286,5 +286,106 @@
"genres": "शैलीहरू",
"explore_genres": "शैलीहरू अन्वेषण गर्नुहोस्",
"friends": "साथीहरू",
- "no_lyrics_available": "क्षमा गर्दैछौं, यस ट्र्याकका लागि गीतका शब्दहरू फेला परेन"
+ "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": "खुला संगठनमा दान गर्नुहोस्",
+ "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": "पछिल्लो ६ महिना",
+ "this_year": "यो वर्ष",
+ "last_2_years": "पछिल्लो २ वर्ष",
+ "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उपयोगकर्तालाई यो थाहा दिनको लागि कि उनीहरूले अर्टिस्टहरूलाई\nSpotify मा गीत सुनेको भए कति भुक्तानी गर्ने थिए।",
+ "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गणना हो जसले प्रयोगकर्तालाई देखाउँछ कि उनीहरूले कति\nअर्टिस्टहरूलाई तिनीहरूका गीतहरू Spotify मा सुनेमा\nभुक्तान गर्नुपर्ने थियो।",
+ "webview_not_found": "वेबभ्यू फेला परेन",
+ "webview_not_found_description": "तपाईंको उपकरणमा कुनै वेबभ्यू रनटाइम स्थापना गरिएको छैन।\nयदि स्थापना गरिएको छ भने, environment PATH मा छ कि छैन भनेर सुनिश्चित गर्नुहोस्\n\nस्थापना पछि, अनुप्रयोग पुनः सुरु गर्नुहोस्",
+ "unsupported_platform": "असमर्थित प्लेटफार्म"
}
\ No newline at end of file
diff --git a/lib/l10n/app_nl.arb b/lib/l10n/app_nl.arb
index 3bece8be..93ab02a1 100644
--- a/lib/l10n/app_nl.arb
+++ b/lib/l10n/app_nl.arb
@@ -286,5 +286,107 @@
"genres": "Genres",
"explore_genres": "Genres verkennen",
"friends": "Vrienden",
- "no_lyrics_available": "Sorry, geen teksten gevonden voor dit nummer"
-}
+ "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”",
+ "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",
+ "enable_connect": "Verbinding inschakelen",
+ "enable_connect_description": "Spotube bedienen vanaf andere apparaten",
+ "devices": "Apparaten",
+ "select": "Selecteren",
+ "connect_client_alert": "Je wordt gecontroleerd door {client}",
+ "this_device": "Dit apparaat",
+ "remote": "Afstandsbediening",
+ "local_library": "Lokale bibliotheek",
+ "add_library_location": "Toevoegen aan bibliotheek",
+ "remove_library_location": "Verwijderen uit bibliotheek",
+ "local_tab": "Lokaal",
+ "stats": "Statistieken",
+ "and_n_more": "en {count} meer",
+ "recently_played": "Onlangs afgespeeld",
+ "browse_more": "Meer bekijken",
+ "no_title": "Geen titel",
+ "not_playing": "Niet aan het afspelen",
+ "epic_failure": "Epische mislukking!",
+ "added_num_tracks_to_queue": "{tracks_length} nummers aan de wachtrij toegevoegd",
+ "spotube_has_an_update": "Spotube heeft een update",
+ "download_now": "Nu downloaden",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} is uitgebracht",
+ "release_version": "Spotube v{version} is uitgebracht",
+ "read_the_latest": "Lees de nieuwste ",
+ "release_notes": "release-opmerkingen",
+ "pick_color_scheme": "Kies kleurenschema",
+ "save": "Opslaan",
+ "choose_the_device": "Kies het apparaat:",
+ "multiple_device_connected": "Er zijn meerdere apparaten verbonden.\nKies het apparaat waarop je deze actie wilt uitvoeren",
+ "nothing_found": "Niets gevonden",
+ "the_box_is_empty": "De doos is leeg",
+ "top_artists": "Topartiesten",
+ "top_albums": "Topalbums",
+ "this_week": "Deze week",
+ "this_month": "Deze maand",
+ "last_6_months": "Laatste 6 maanden",
+ "this_year": "Dit jaar",
+ "last_2_years": "Laatste 2 jaar",
+ "all_time": "All time",
+ "powered_by_provider": "Aangedreven door {providerName}",
+ "email": "E-mail",
+ "profile_followers": "Volgers",
+ "birthday": "Verjaardag",
+ "subscription": "Abonnement",
+ "not_born": "Niet geboren",
+ "hacker": "Hacker",
+ "profile": "Profiel",
+ "no_name": "Geen naam",
+ "edit": "Bewerken",
+ "user_profile": "Gebruikersprofiel",
+ "count_plays": "{count} afspeelbeurten",
+ "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.",
+ "count_mins": "{minutes} min",
+ "summary_minutes": "minuten",
+ "summary_listened_to_music": "Beluisterde muziek",
+ "summary_songs": "nummers",
+ "summary_streamed_overall": "Totaal gestreamd",
+ "summary_owed_to_artists": "Te betalen aan artiesten\ndeze maand",
+ "summary_artists": "van de artiest",
+ "summary_music_reached_you": "Muziek heeft je bereikt",
+ "summary_full_albums": "volledige albums",
+ "summary_got_your_love": "Kreeg je liefde",
+ "summary_playlists": "afspeellijsten",
+ "summary_were_on_repeat": "Was op herhaling",
+ "total_money": "Totaal {money}",
+ "minutes_listened": "Luistertijd",
+ "streamed_songs": "Gestreamde nummers",
+ "count_streams": "{count} streams",
+ "owned_by_you": "Bezit door jou",
+ "copied_shareurl_to_clipboard": "{shareUrl} gekopieerd naar klembord",
+ "spotify_hipotetical_calculation": "*Dit is berekend op basis van Spotify's betaling per stream\nvan $0.003 tot $0.005. Dit is een hypothetische\nberekening om de gebruiker inzicht te geven in hoeveel ze\naan de artiesten zouden hebben betaald als ze hun liedjes op Spotify\nzouden luisteren.",
+ "webview_not_found": "Webview niet gevonden",
+ "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",
+ "unsupported_platform": "Niet ondersteund platform"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_pl.arb b/lib/l10n/app_pl.arb
index b7ce8923..c003ef08 100644
--- a/lib/l10n/app_pl.arb
+++ b/lib/l10n/app_pl.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Skopiuj wartość ciasteczka \"sp_dc\"",
"step_4_steps": "Wklej skopiowaną wartość \"sp_dc\"",
"friends": "Przyjaciele",
- "no_lyrics_available": "Przepraszamy, nie można znaleźć tekstu dla tego utworu"
+ "no_lyrics_available": "Przepraszamy, nie można znaleźć tekstu dla tego utworu",
+ "sort_duration": "Sortuj według Czasu Trwania",
+ "start_a_radio": "Uruchom radio",
+ "how_to_start_radio": "Jak chcesz uruchomić radio?",
+ "replace_queue_question": "Czy chcesz zastąpić bieżącą kolejkę czy dodać do niej?",
+ "endless_playback": "Nieskończona Odtwarzanie",
+ "delete_playlist": "Usuń Playlistę",
+ "delete_playlist_confirmation": "Czy na pewno chcesz usunąć tę listę odtwarzania?",
+ "local_tracks": "Lokalne Utwory",
+ "song_link": "Link do Utworu",
+ "skip_this_nonsense": "Pomiń tę bzdurę",
+ "freedom_of_music": "“Wolność Muzyki”",
+ "freedom_of_music_palm": "“Wolność Muzyki w Twojej dłoni”",
+ "get_started": "Zacznijmy",
+ "youtube_source_description": "Polecane i działa najlepiej.",
+ "piped_source_description": "Czujesz się wolny? To samo co YouTube, ale dużo za darmo.",
+ "jiosaavn_source_description": "Najlepszy dla regionu Azji Południowej.",
+ "highest_quality": "Najwyższa Jakość: {quality}",
+ "select_audio_source": "Wybierz Źródło Audio",
+ "endless_playback_description": "Automatycznie dodaj nowe utwory na koniec kolejki",
+ "choose_your_region": "Wybierz swoją region",
+ "choose_your_region_description": "To pomoże Spotube pokazać Ci odpowiednią treść dla Twojej lokalizacji.",
+ "choose_your_language": "Wybierz swój język",
+ "help_project_grow": "Pomóż temu projektowi rosnąć",
+ "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.",
+ "contribute_on_github": "Przyczyniaj się na GitHubie",
+ "donate_on_open_collective": "Dotuj na Open Collective",
+ "browse_anonymously": "Przeglądaj Anonimowo",
+ "enable_connect": "Włącz połączenie",
+ "enable_connect_description": "Kontroluj Spotube z innych urządzeń",
+ "devices": "Urządzenia",
+ "select": "Wybierz",
+ "connect_client_alert": "Jesteś sterowany przez {client}",
+ "this_device": "To urządzenie",
+ "remote": "Zdalny",
+ "local_library": "Biblioteka lokalna",
+ "add_library_location": "Dodaj do biblioteki",
+ "remove_library_location": "Usuń z biblioteki",
+ "local_tab": "Lokalny",
+ "stats": "Statystyki",
+ "and_n_more": "i {count} więcej",
+ "recently_played": "Ostatnio odtwarzane",
+ "browse_more": "Zobacz więcej",
+ "no_title": "Brak tytułu",
+ "not_playing": "Nie odtwarzane",
+ "epic_failure": "Epicka porażka!",
+ "added_num_tracks_to_queue": "Dodano {tracks_length} utworów do kolejki",
+ "spotube_has_an_update": "Spotube ma aktualizację",
+ "download_now": "Pobierz teraz",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} został wydany",
+ "release_version": "Spotube v{version} został wydany",
+ "read_the_latest": "Przeczytaj najnowsze ",
+ "release_notes": "notatki o wersji",
+ "pick_color_scheme": "Wybierz schemat kolorów",
+ "save": "Zapisz",
+ "choose_the_device": "Wybierz urządzenie:",
+ "multiple_device_connected": "Jest wiele urządzeń podłączonych.\nWybierz urządzenie, na którym chcesz wykonać tę akcję",
+ "nothing_found": "Nic nie znaleziono",
+ "the_box_is_empty": "Pudełko jest puste",
+ "top_artists": "Najlepsi artyści",
+ "top_albums": "Najlepsze albumy",
+ "this_week": "W tym tygodniu",
+ "this_month": "W tym miesiącu",
+ "last_6_months": "Ostatnie 6 miesięcy",
+ "this_year": "W tym roku",
+ "last_2_years": "Ostatnie 2 lata",
+ "all_time": "Wszystkie czasy",
+ "powered_by_provider": "Napędzane przez {providerName}",
+ "email": "E-mail",
+ "profile_followers": "Obserwujący",
+ "birthday": "Data urodzenia",
+ "subscription": "Subskrypcja",
+ "not_born": "Nie urodzony",
+ "hacker": "Haker",
+ "profile": "Profil",
+ "no_name": "Brak nazwy",
+ "edit": "Edytuj",
+ "user_profile": "Profil użytkownika",
+ "count_plays": "{count} odtworzeń",
+ "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.",
+ "count_mins": "{minutes} min",
+ "summary_minutes": "minuty",
+ "summary_listened_to_music": "Słuchana muzyka",
+ "summary_songs": "utwory",
+ "summary_streamed_overall": "Ogółem streamowane",
+ "summary_owed_to_artists": "Do zapłaty artystom\nw tym miesiącu",
+ "summary_artists": "artystów",
+ "summary_music_reached_you": "Muzyka dotarła do Ciebie",
+ "summary_full_albums": "pełne albumy",
+ "summary_got_your_love": "Otrzymał Twoją miłość",
+ "summary_playlists": "playlisty",
+ "summary_were_on_repeat": "Były na powtarzaniu",
+ "total_money": "Łącznie {money}",
+ "minutes_listened": "Minuty odsłuchane",
+ "streamed_songs": "Strumieniowane utwory",
+ "count_streams": "{count} strumieni",
+ "owned_by_you": "Własność Twoja",
+ "copied_shareurl_to_clipboard": "{shareUrl} skopiowano do schowka",
+ "spotify_hipotetical_calculation": "*Obliczone na podstawie płatności Spotify za strumień\nw zakresie od $0.003 do $0.005. Jest to hipotetyczne\nobliczenie mające na celu pokazanie użytkownikowi, ile\nzapłaciliby artystom, gdyby słuchali ich utworów na Spotify.",
+ "webview_not_found": "Nie znaleziono Webview",
+ "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ę",
+ "unsupported_platform": "Nieobsługiwana platforma"
}
\ No newline at end of file
diff --git a/lib/l10n/app_pt.arb b/lib/l10n/app_pt.arb
index 1c75f734..02772b1e 100644
--- a/lib/l10n/app_pt.arb
+++ b/lib/l10n/app_pt.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Copie o valor do cookie \"sp_dc\"",
"step_4_steps": "Cole o valor copiado de \"sp_dc\"",
"friends": "Amigos",
- "no_lyrics_available": "Desculpe, não foi possível encontrar a letra desta faixa"
+ "no_lyrics_available": "Desculpe, não foi possível encontrar a letra desta faixa",
+ "sort_duration": "Ordenar por Duração",
+ "start_a_radio": "Iniciar uma Rádio",
+ "how_to_start_radio": "Como você deseja iniciar a rádio?",
+ "replace_queue_question": "Você deseja substituir a fila atual ou acrescentar a ela?",
+ "endless_playback": "Reprodução sem fim",
+ "delete_playlist": "Excluir Lista de Reprodução",
+ "delete_playlist_confirmation": "Tem certeza de que deseja excluir esta lista de reprodução?",
+ "local_tracks": "Faixas Locais",
+ "song_link": "Link da Música",
+ "skip_this_nonsense": "Pular essa bobagem",
+ "freedom_of_music": "“Liberdade da Música”",
+ "freedom_of_music_palm": "“Liberdade da Música na palma da sua mão”",
+ "get_started": "Vamos começar",
+ "youtube_source_description": "Recomendado e funciona melhor.",
+ "piped_source_description": "Sentindo-se livre? Igual ao YouTube, mas muito mais grátis.",
+ "jiosaavn_source_description": "Melhor para a região da Ásia do Sul.",
+ "highest_quality": "Melhor Qualidade: {quality}",
+ "select_audio_source": "Selecionar Fonte de Áudio",
+ "endless_playback_description": "Adicionar automaticamente novas músicas\nao final da fila",
+ "choose_your_region": "Escolha sua região",
+ "choose_your_region_description": "Isso ajudará o Spotube a mostrar o conteúdo certo\npara sua localização.",
+ "choose_your_language": "Escolha seu idioma",
+ "help_project_grow": "Ajude este projeto a crescer",
+ "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.",
+ "contribute_on_github": "Contribuir no GitHub",
+ "donate_on_open_collective": "Doar no Open Collective",
+ "browse_anonymously": "Navegar Anonimamente",
+ "enable_connect": "Ativar conexão",
+ "enable_connect_description": "Controle o Spotube a partir de outros dispositivos",
+ "devices": "Dispositivos",
+ "select": "Selecionar",
+ "connect_client_alert": "Você está sendo controlado por {client}",
+ "this_device": "Este dispositivo",
+ "remote": "Remoto",
+ "local_library": "Biblioteca local",
+ "add_library_location": "Adicionar à biblioteca",
+ "remove_library_location": "Remover da biblioteca",
+ "local_tab": "Local",
+ "stats": "Estatísticas",
+ "and_n_more": "e {count} mais",
+ "recently_played": "Reproduzido Recentemente",
+ "browse_more": "Ver Mais",
+ "no_title": "Sem Título",
+ "not_playing": "Não está a reproduzir",
+ "epic_failure": "Fracasso épico!",
+ "added_num_tracks_to_queue": "Adicionados {tracks_length} faixas à fila",
+ "spotube_has_an_update": "Spotube tem uma atualização",
+ "download_now": "Baixar Agora",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} foi lançado",
+ "release_version": "Spotube v{version} foi lançado",
+ "read_the_latest": "Leia o mais recente ",
+ "release_notes": "notas de versão",
+ "pick_color_scheme": "Escolha o esquema de cores",
+ "save": "Salvar",
+ "choose_the_device": "Escolha o dispositivo:",
+ "multiple_device_connected": "Há vários dispositivos conectados.\nEscolha o dispositivo no qual deseja executar esta ação",
+ "nothing_found": "Nada encontrado",
+ "the_box_is_empty": "A caixa está vazia",
+ "top_artists": "Principais Artistas",
+ "top_albums": "Principais Álbuns",
+ "this_week": "Esta semana",
+ "this_month": "Este mês",
+ "last_6_months": "Últimos 6 meses",
+ "this_year": "Este ano",
+ "last_2_years": "Últimos 2 anos",
+ "all_time": "De todos os tempos",
+ "powered_by_provider": "Desenvolvido por {providerName}",
+ "email": "E-mail",
+ "profile_followers": "Seguidores",
+ "birthday": "Aniversário",
+ "subscription": "Assinatura",
+ "not_born": "Não nascido",
+ "hacker": "Hacker",
+ "profile": "Perfil",
+ "no_name": "Sem Nome",
+ "edit": "Editar",
+ "user_profile": "Perfil do Usuário",
+ "count_plays": "{count} reproduzidos",
+ "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.",
+ "count_mins": "{minutes} min",
+ "summary_minutes": "minutos",
+ "summary_listened_to_music": "Música ouvida",
+ "summary_songs": "faixas",
+ "summary_streamed_overall": "Total de streams",
+ "summary_owed_to_artists": "Devido aos artistas\neste mês",
+ "summary_artists": "artista",
+ "summary_music_reached_you": "A música chegou até você",
+ "summary_full_albums": "álbuns completos",
+ "summary_got_your_love": "Recebeu seu amor",
+ "summary_playlists": "playlists",
+ "summary_were_on_repeat": "Estavam em repetição",
+ "total_money": "Total {money}",
+ "minutes_listened": "Minutos ouvidos",
+ "streamed_songs": "Músicas transmitidas",
+ "count_streams": "{count} streams",
+ "owned_by_you": "De sua propriedade",
+ "copied_shareurl_to_clipboard": "{shareUrl} copiado para a área de transferência",
+ "spotify_hipotetical_calculation": "*Isso é calculado com base no pagamento por stream do Spotify\nque varia de $0.003 a $0.005. Esta é uma cálculo hipotético\npara dar ao usuário uma visão de quanto teriam pago aos artistas\nse eles ouvissem suas músicas no Spotify.",
+ "webview_not_found": "Webview não encontrado",
+ "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",
+ "unsupported_platform": "Plataforma não suportada"
}
\ No newline at end of file
diff --git a/lib/l10n/app_ru.arb b/lib/l10n/app_ru.arb
index 7ed67f4f..189e644f 100644
--- a/lib/l10n/app_ru.arb
+++ b/lib/l10n/app_ru.arb
@@ -8,16 +8,16 @@
"genre_categories_filter": "Фильтр по категориям или жанрам...",
"genre": "Жанр",
"personalized": "Персонализированный",
- "featured": "Будующий",
- "new_releases": "Новые",
- "songs": "Песни",
+ "featured": "Популярное",
+ "new_releases": "Новое",
+ "songs": "Треки",
"playing_track": "Играет {track}",
"queue_clear_alert": "Это удалит текущую очередь. {track_length} треков будет удалено. Вы хотите продолжить?",
"load_more": "Загрузить больше",
"playlists": "Плейлисты",
"artists": "Исполнители",
"albums": "Альбомы",
- "tracks": "Трек",
+ "tracks": "Треки",
"downloads": "Загрузки",
"filter_playlists": "Применить фильтры к вашим плейлистам...",
"liked_tracks": "Понравившиеся треки",
@@ -25,20 +25,22 @@
"create_playlist": "Создание плейлиста",
"create_a_playlist": "Создать плейлист",
"create": "Создать",
- "cancel": "Отменить",
+ "cancel": "Отмена",
+ "update": "Обновить",
"playlist_name": "Назвать плейлист",
"name_of_playlist": "Название плейлиста",
"description": "Описание",
- "public": "Публичные",
+ "public": "Публичный",
"collaborative": "Совместный",
"search_local_tracks": "Поиск песен на вашем устройстве...",
"play": "Играть",
"delete": "Удалить",
- "none": "Никто",
+ "none": "Пусто",
"sort_a_z": "Сортировка по алфавиту",
"sort_z_a": "Сортировка по алфавиту в обратную сторону",
"sort_artist": "Сортировать по исполнителю",
"sort_album": "Сортировать по альбомам",
+ "sort_duration": "Сортировать по длительности",
"sort_tracks": "Сортировать треки",
"currently_downloading": "Загружается ({tracks_length})",
"cancel_all": "Отменить все",
@@ -104,6 +106,9 @@
"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",
@@ -141,7 +146,7 @@
"close": "Закрыть",
"minimize_to_tray": "Свернуть",
"show_tray_icon": "Показать значок на панели задач",
- "about": "О",
+ "about": "О нас",
"u_love_spotube": "Мы знаем что вам нравится Spotube",
"check_for_updates": "Проверьте наличие обновлений",
"about_spotube": "О Spotube",
@@ -175,9 +180,11 @@
"step_2": "Шаг 2",
"step_2_steps": "1. После входа в систему нажмите F12 или щелкните правой кнопкой мыши > «Проверить», чтобы открыть инструменты разработчика браузера.\n2. Затем перейдите на вкладку \"Application\" (Chrome, Edge, Brave и т.д..) or \"Storage\" (Firefox, Palemoon и т.д..)\n3. Перейдите в раздел \"Cookies\", а затем в подраздел \"https://accounts.spotify.com\"",
"step_3": "Шаг 3",
- "success_emoji": "Успешно 🥳",
+ "step_3_steps": "Скопируйте значение Cookie \"sp_dc\"",
+ "success_emoji": "Успешно🥳",
"success_message": "Теперь вы успешно вошли в свою учетную запись Spotify. Отличная работа, приятель!",
"step_4": "Шаг 4",
+ "step_4_steps": "Вставьте скопированное значение \"sp_dc\"",
"something_went_wrong": "Что-то пошло не так",
"piped_instance": "Экземпляр сервера Piped",
"piped_description": "Серверный экземпляр Piped для сопоставления треков",
@@ -205,7 +212,7 @@
"popularity": "Популярность",
"key": "Ключ",
"duration": "Продолжительность (с)",
- "tempo": "Время (BPM)",
+ "tempo": "Темп (BPM)",
"mode": "Режим",
"time_signature": "Тактовый размер",
"short": "Короткий",
@@ -257,8 +264,6 @@
"you_are_offline": "Нет доступа к сети",
"connection_restored": "Ваше интернет-соединение восстановлено",
"use_system_title_bar": "Использовать системную панель заголовка",
- "update_playlist": "Обновить плейлист",
- "update": "Обновить",
"crunching_results": "Обработка результатов...",
"search_to_get_results": "Поиск для получения результатов",
"use_amoled_mode": "Режим AMOLED",
@@ -283,8 +288,104 @@
"browse_all": "Просмотреть все",
"genres": "Жанры",
"explore_genres": "Исследовать жанры",
- "step_3_steps": "Скопируйте значение файла cookie \"sp_dc\"",
- "step_4_steps": "Вставьте скопированное значение \"sp_dc\"",
"friends": "Друзья",
- "no_lyrics_available": "Извините, не удается найти текст для этого трека"
+ "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": "Лучший для Южно-Азиатского региона.",
+ "highest_quality": "Наивысшее качество: {quality}",
+ "select_audio_source": "Выберите аудиоисточник",
+ "endless_playback_description": "Автоматически добавляйте новые песни\nв конец очереди",
+ "choose_your_region": "Выберите ваш регион",
+ "choose_your_region_description": "Это поможет Spotube показать вам правильный контент\nдля вашего местоположения.",
+ "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": "Статистика",
+ "update_playlist": "Обновить плейлист",
+ "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расчет, чтобы показать пользователю, сколько бы он\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}",
+ "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.",
+ "webview_not_found": "Webview не найден",
+ "webview_not_found_description": "На вашем устройстве не установлена среда выполнения Webview.\nЕсли он установлен, убедитесь, что он находится в environment PATH\n\nПосле установки перезапустите приложение",
+ "unsupported_platform": "Платформа не поддерживается"
}
\ No newline at end of file
diff --git a/lib/l10n/app_th.arb b/lib/l10n/app_th.arb
new file mode 100644
index 00000000..27c05a5d
--- /dev/null
+++ b/lib/l10n/app_th.arb
@@ -0,0 +1,392 @@
+{
+ "guest": "ผู้มาเยือน",
+ "browse": "เรียกดู",
+ "search": "ค้นหา",
+ "library": "คลัง",
+ "lyrics": "เนื้อเพลง",
+ "settings": "ตั้งค่า",
+ "genre_categories_filter": "กรองประเภทหรือแนวเพลง...",
+ "genre": "ประเภท",
+ "personalized": "ปรับแต่ง",
+ "featured": "เด่น",
+ "new_releases": "เพิ่งปล่อยใหม่",
+ "songs": "เพลง",
+ "playing_track": "กำลังเล่น {track}",
+ "queue_clear_alert": "การดำเนินการนี้จะล้างคิวปัจจุบัน {track_length} แทร็ก จะถูกลบออก\nคุณต้องการดำเนินการต่อหรือไม่?",
+ "load_more": "โหลดเพิ่มเติม",
+ "playlists": "เพลย์ลิสต์",
+ "artists": "ศิลปิน",
+ "albums": "อัลบั้ม",
+ "tracks": "แทร็ก",
+ "downloads": "ดาวน์โหลด",
+ "filter_playlists": "กรองเพลย์ลิสต์...",
+ "liked_tracks": "เพลงที่ชอบ",
+ "liked_tracks_description": "เพลงที่คุณชื่นชอบทั้งหมด",
+ "create_playlist": "สร้างเพลย์ลิสต์",
+ "create_a_playlist": "สร้างเพลย์ลิสต์",
+ "update_playlist": "อัพเดทเพลย์ลิสต์",
+ "create": "สร้าง",
+ "cancel": "ยกเลิก",
+ "update": "อัพเดท",
+ "playlist_name": "ชื่อเพลย์ลิสต์",
+ "name_of_playlist": "ชื่อของเพลย์ลิสต์",
+ "description": "คำอธิบาย",
+ "public": "สาธารณะ",
+ "collaborative": "ร่วมมือกัน",
+ "search_local_tracks": "ค้นหาเพลงในเครื่อง...",
+ "play": "เล่น",
+ "delete": "ลบ",
+ "none": "ไม่มี",
+ "sort_a_z": "เรียงตาม A-Z",
+ "sort_z_a": "เรียงตาม Z-A",
+ "sort_artist": "เรียงตามศิลปิน",
+ "sort_album": "เรียงตามอัลบั้ม",
+ "sort_duration": "เรียงตามความยาว",
+ "sort_tracks": "เรียงตามเพลง",
+ "currently_downloading": "กำลังดาวน์โหลด ({tracks_length})",
+ "cancel_all": "ยกเลิกทั้งหมด",
+ "filter_artist": "กรองศิลปิน...",
+ "followers": "{followers} ผู้ติดตาม",
+ "add_artist_to_blacklist": "เพิ่มศิลปินในบัญชีดำ",
+ "top_tracks": "เพลงฮิต",
+ "fans_also_like": "แฟนๆ ยังชอบ",
+ "loading": "กำลังโหลด...",
+ "artist": "ศิลปิน",
+ "blacklisted": "อยู่ในบัญชีดำ",
+ "following": "กำลังติดตาม",
+ "follow": "ติดตาม",
+ "artist_url_copied": "คัดลอก 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": "วนเพลง",
+ "repeat_playlist": "ซ้ำเพลย์ลิสต์",
+ "queue": "คิว",
+ "alternative_track_sources": "แหล่งแทร็กอื่น",
+ "download_track": "ดาวน์โหลดแทร็ก",
+ "tracks_in_queue": "{tracks} แทร็กในคิว",
+ "clear_all": "ล้างทั้งหมด",
+ "show_hide_ui_on_hover": "แสดง/ซ่อน UI เมื่อโฮเวอร์",
+ "always_on_top": "อยู่ด้านบนเสมอ",
+ "exit_mini_player": "ออกจากมินิเพลย์เยอร์",
+ "download_location": "ตำแหน่งดาวน์โหลด",
+ "account": "บัญชี",
+ "login_with_spotify": "เข้าสู่ระบบด้วยบัญชี Spotify",
+ "connect_with_spotify": "เชื่อมต่อกับ Spotify",
+ "logout": "ออกจากระบบ",
+ "logout_of_this_account": "ออกจากระบบบัญชีนี้",
+ "language_region": "ภาษาและภูมิภาค",
+ "language": "ภาษา",
+ "system_default": "ค่าเริ่มต้นของระบบ",
+ "market_place_region": "ภูมิภาค Marketplace",
+ "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": "Kingkor Roy Tirtho",
+ "copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
+ "license": "ใบอนุญาต",
+ "add_spotify_credentials": "เพิ่มข้อมูลรับรอง Spotify ของคุณเพื่อเริ่มต้น",
+ "credentials_will_not_be_shared_disclaimer": "ไม่ต้องกังวล ข้อมูลรับรองใดๆ ของคุณจะไม่ถูกเก็บรวบรวมหรือแชร์กับใคร",
+ "know_how_to_login": "ไม่รู้จักวิธีดำเนินการนี้ใช่ไหม",
+ "follow_step_by_step_guide": "ทำตามคู่มือทีละขั้น",
+ "spotify_cookie": "คุกกี้ Spotify {name}",
+ "cookie_name_cookie": "คุกกี้ {name}",
+ "fill_in_all_fields": "กรุณากรอกข้อมูลทุกช่อง",
+ "submit": "ยื่น",
+ "exit": "ออก",
+ "previous": "ย้อนกลับ",
+ "next": "ถัดไป",
+ "done": "เสร็จ",
+ "step_1": "ขั้นที่ 1",
+ "first_go_to": "ก่อนอื่น ไปที่",
+ "login_if_not_logged_in": "ยังไม่ได้เข้าสู่ระบบ ให้เข้าสู่ระบบ/ลงทะเบียน",
+ "step_2": "ขั้นที่ 2",
+ "step_2_steps": "1. หลังจากเข้าสู่ระบบแล้ว กด F12 หรือ คลิกขวาที่เมาส์ > ตรวจสอบเพื่อเปิด Devtools เบราว์เซอร์\n2. จากนั้นไปที่แท็บ \"แอปพลิเคชัน\" (Chrome, Edge, Brave เป็นต้น) หรือแท็บ \"ที่เก็บข้อมูล\" (Firefox, Palemoon เป็นต้น)\n3. ไปที่ส่วน \"คุกกี้\" แล้วไปที่ subsection \"https: //accounts.spotify.com\"",
+ "step_3": "ขั้นที่ 3",
+ "step_3_steps": "คัดลอกค่าคุกกี้ \"sp_dc\"",
+ "success_emoji": "สำเร็จ",
+ "success_message": "ตอนนี้คุณเข้าสู่ระบบด้วยบัญชี Spotify ของคุณเรียบร้อยแล้ว ยอดเยี่ยม!",
+ "step_4": "ขั้นที่ 4",
+ "step_4_steps": "วางค่า \"sp_dc\" ที่คัดลอกมา",
+ "something_went_wrong": "มีอะไรผิดพลาด",
+ "piped_instance": "อินสแตนซ์เซิร์ฟเวอร์แบบ Pipe",
+ "piped_description": "อินสแตนซ์เซิร์ฟเวอร์แบบ Pipe ที่ใช้สำหรับการจับคู่แทร็ก",
+ "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": "ระยะเวลา (วินาที)",
+ "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 หมายความว่าคุณไม่สามารถใช้ YouTube (แม้ว่าคุณจะล็อกอินอยู่) เป็นเวลาอย่างน้อย 2-3 เดือนจากอุปกรณ์ IP นั้น และ 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หากคุณใช้ Linux โปรดตรวจสอบว่าคุณได้ติดตั้งบริการลับ (gnome-keyring, kde-wallet, keepassxc เป็นต้น)",
+ "querying_info": "กำลังดึงข้อมูล...",
+ "piped_api_down": "Piped API ไม่ทำงาน",
+ "piped_down_error_instructions": "Piped instance {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": "Scrobble ไปเป็น Last.fm",
+ "go_to_album": "ไปที่อัลบั้ม",
+ "discord_rich_presence": "Discord Rich Presence",
+ "browse_all": "เรียกดูทั้งหมด",
+ "genres": "ประเภท",
+ "explore_genres": "สำรวจประเภท",
+ "friends": "เพื่อน",
+ "no_lyrics_available": "ขออภัย ไม่พบเนื้อเพลงสำหรับเพลงนี้",
+ "start_a_radio": "เปิดวิทยุ",
+ "how_to_start_radio": "หากต้องการเปิดวิทยุฟังยังไง?",
+ "replace_queue_question": "คุณต้องการแทนที่คิวปัจจุบันหรือเพิ่มเข้าไปหรือไม่",
+ "endless_playback": "เล่นซ้ำ",
+ "delete_playlist": "ลบเพลย์ลิสต์",
+ "delete_playlist_confirmation": "คุณแน่ใจที่จะลบเพลย์ลิสต์นี้หรือไม่",
+ "local_tracks": "เพลงในเครื่อง",
+ "song_link": "ลิงค์เพลง",
+ "skip_this_nonsense": "ข้ามสิ่งไร้สาระนี้",
+ "freedom_of_music": "“เสรีภาพแห่งเสียงเพลง”",
+ "freedom_of_music_palm": "“เสรีภาพแห่งเสียงเพลง ในมือของคุณ”",
+ "get_started": "เริ่มต้น",
+ "youtube_source_description": "แนะนำและใช้งานได้ดีที่สุด",
+ "piped_source_description": "รู้สึกอิสระ? เหมือน YouTube แต่ฟรีกว่าเยอะ",
+ "jiosaavn_source_description": "ดีที่สุดสำหรับภูมิภาคเอเชียใต้",
+ "highest_quality": "คุณภาพสูงสุด: {quality}",
+ "select_audio_source": "เลือกแหล่งเสียง",
+ "endless_playback_description": "เพิ่มเพลงใหม่ลงในคิวโดยอัตโนมัติ",
+ "choose_your_region": "เลือกภูมิภาคของคุณ",
+ "choose_your_region_description": "สิ่งนี้จะช่วยให้ Spotube แสดงเนื้อหาที่เหมาะสมสำหรับคุณ",
+ "choose_ your_language": "เลือกภาษาของคุณ",
+ "help_project_grow": "ช่วยให้โครงการนี้เติบโต",
+ "help_project_grow_description": "Spotube เป็นโครงการโอเพนซอร์ส คุณสามารถช่วยให้โครงการนี้เติบโตได้โดยการมีส่วนร่วมในโครงการ รายงานข้อบกพร่อง หรือเสนอคุณสมบัติใหม่",
+ "contribute_on_github": "มีส่วนร่วมบน GitHub",
+ "donate_on_open_collective": "บริจาคบน Open Collective",
+ "browse_anonymously": "เรียกดูแบบไม่ระบุตัวตน",
+ "choose_your_language": "เลือกภาษาของคุณ",
+ "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เพื่อให้ข้อมูลแก่ผู้ใช้เกี่ยวกับจำนวนเงินที่พวกเขา\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}",
+ "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.",
+ "webview_not_found": "ไม่พบ Webview",
+ "webview_not_found_description": "ไม่พบ runtime ของ Webview บนอุปกรณ์ของคุณ\nหากติดตั้งแล้วตรวจสอบให้แน่ใจว่าอยู่ใน environment PATH\n\nหลังจากติดตั้งแล้ว ให้รีสตาร์ทแอป",
+ "unsupported_platform": "แพลตฟอร์มไม่รองรับ"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_tr.arb b/lib/l10n/app_tr.arb
index 4d9066fd..230f14e8 100644
--- a/lib/l10n/app_tr.arb
+++ b/lib/l10n/app_tr.arb
@@ -1,63 +1,64 @@
{
"guest": "Misafir",
- "browse": "Gözat",
+ "browse": "Göz at",
"search": "Ara",
"library": "Kütüphane",
- "lyrics": "Sözler",
+ "lyrics": "Şarkı sözleri",
"settings": "Ayarlar",
"genre_categories_filter": "Kategorileri veya türleri filtreleyin...",
"genre": "Tür",
"personalized": "Kişiselleştirilmiş",
- "featured": "Öne Çıkanlar",
- "new_releases": "Yeni Çıkanlar",
+ "featured": "Öne çıkanlar",
+ "new_releases": "Yeni çıkanlar",
"songs": "Şarkılar",
- "playing_track": "Oynatılıyor {track}",
- "queue_clear_alert": "Bu, mevcut kuyruğu temizleyecektir. {track_length} parçaları kaldırılacaktır\nDevam etmek istiyor musunuz?",
+ "playing_track": "{track} oynatılıyor",
+ "queue_clear_alert": "Bu, mevcut kuyruğu temizleyecektir. {track_length} parça kaldırılacak\nDevam etmek istiyor musunuz?",
"load_more": "Daha fazlasını yükle",
- "playlists": "Çalma Listeleri",
+ "playlists": "Oynatma listeleri",
"artists": "Sanatçılar",
"albums": "Albümler",
"tracks": "Parçalar",
- "downloads": "İndirmeler",
- "filter_playlists": "Çalma listelerinizi filtreleyin...",
- "liked_tracks": "Beğenilen Parçalar",
+ "downloads": "İndirilenler",
+ "filter_playlists": "Oynatma listelerinizi filtreleyin...",
+ "liked_tracks": "Beğenilen parçalar",
"liked_tracks_description": "Beğendiğiniz tüm parçalar",
- "create_playlist": "Çalma Listesi Oluştur",
- "create_a_playlist": "Bir çalma listesi oluştur",
- "update_playlist": "Çalma listesini güncelle",
+ "create_playlist": "Oynatma listesi oluştur",
+ "create_a_playlist": "Bir oynatma listesi oluştur",
+ "update_playlist": "Oynatma listesini güncelle",
"create": "Oluştur",
"cancel": "İptal",
"update": "Güncelle",
- "playlist_name": "Çalma Listesi Adı",
- "name_of_playlist": "Çalma listesi adı",
+ "playlist_name": "Oynatma listesi adı",
+ "name_of_playlist": "Oynatma listesinin adı",
"description": "Açıklama",
"public": "Halka açık",
"collaborative": "İşbirliği",
- "search_local_tracks": "Yerel parçaları arayın...",
+ "search_local_tracks": "Yerel parçaları ara...",
"play": "Oynat",
"delete": "Sil",
- "none": "Hiçbiri",
- "sort_a_z": "A'dan Z'ye sırala",
- "sort_z_a": "Z'dan A'ye sırala",
- "sort_artist": "Sanatçıya Göre Sırala",
- "sort_album": "Albüme Göre Sırala",
- "sort_tracks": "Parçaları Sırala",
- "currently_downloading": "Şu Anda İndiriliyor ({tracks_length})",
- "cancel_all": "Tümünü İptal Et",
- "filter_artist": "Sanatçıları filtrele...",
+ "none": "Yok",
+ "sort_a_z": "A - Z'ye göre sırala",
+ "sort_z_a": "Z - A'ya göre sırala",
+ "sort_artist": "Sanatçıya göre sırala",
+ "sort_album": "Albüme göre sırala",
+ "sort_duration": "Süreye göre sırala",
+ "sort_tracks": "Parçaları sırala",
+ "currently_downloading": "Şu anda indirilenler ({tracks_length})",
+ "cancel_all": "Tümünü iptal et",
+ "filter_artist": "Sanatçıları filtreleyin...",
"followers": "{followers} Takipçiler",
"add_artist_to_blacklist": "Sanatçıyı kara listeye ekle",
- "top_tracks": "En İyi Parçalar",
- "fans_also_like": "Hayranlar ayrıca şunları beğendi",
+ "top_tracks": "En iyi parçalar",
+ "fans_also_like": "Hayranlar ayrıca şunları da beğendi",
"loading": "Yükleniyor...",
"artist": "Sanatçı",
- "blacklisted": "Kara Listede",
- "following": "Takip Ediliyor",
- "follow": "Takip Et",
+ "blacklisted": "Kara listeye alındı",
+ "following": "Takip ediliyor",
+ "follow": "Takip et",
"artist_url_copied": "Sanatçı bağlantısı panoya kopyalandı",
- "added_to_queue": "Kuyruğa {tracks} parçaları eklendi",
- "filter_albums": "Albümleri filtrele...",
- "synced": "Eşitlendi",
+ "added_to_queue": "Kuyruğa {tracks} parçası eklendi",
+ "filter_albums": "Albümleri filtreleyin...",
+ "synced": "Senkronize edildi",
"plain": "Sade",
"shuffle": "Karıştır",
"search_tracks": "Parça ara...",
@@ -65,151 +66,153 @@
"error": "Hata {error}",
"title": "Başlık",
"time": "Zaman",
- "more_actions": "Daha fazla işlem",
+ "more_actions": "Daha fazla eylem",
"download_count": "İndir ({count})",
- "add_count_to_playlist": "Çalma Listesine ({count}) Ekle",
- "add_count_to_queue": "Sıraya ({count}) ekle",
- "play_count_next": "Oynat ({count}) sonraki",
+ "add_count_to_playlist": "Oynatma Listesine ekle ({count})",
+ "add_count_to_queue": "Kuyruğa ekle ({count})",
+ "play_count_next": "Sonrakini oynat ({count})",
"album": "Albüm",
- "copied_to_clipboard": "Panoya {data} kopyalandı",
- "add_to_following_playlists": "Aşağıdaki Çalma Listelerine {track} ekle",
+ "copied_to_clipboard": "{data} panoya kopyalandı",
+ "add_to_following_playlists": "{track} parçasını aşağıdaki oynatma listelerine ekle",
"add": "Ekle",
- "added_track_to_queue": "Sıraya {track} eklendi",
+ "added_track_to_queue": "{track} kuyruğa eklendi",
"add_to_queue": "Kuyruğa ekle",
- "track_will_play_next": "{track} sonraki çalacak",
- "play_next": "Sıradaki",
- "removed_track_from_queue": "Sıradan {track} kaldırıldı",
- "remove_from_queue": "Kuyruktan çıkar",
+ "track_will_play_next": "{track} bir sonraki çalacak",
+ "play_next": "Sonrakini oynat",
+ "removed_track_from_queue": "{track} kuyruktan kaldırıldı",
+ "remove_from_queue": "Kuyruktan kaldır",
"remove_from_favorites": "Favorilerden kaldır",
"save_as_favorite": "Favori olarak kaydet",
- "add_to_playlist": "Çalma listesine ekle",
- "remove_from_playlist": "Çalma listesinden kaldır",
+ "add_to_playlist": "Oynatma listesine ekle",
+ "remove_from_playlist": "Oynatma listesinden kaldır",
"add_to_blacklist": "Kara listeye ekle",
- "remove_from_blacklist": "Kara listeden çıkar",
+ "remove_from_blacklist": "Kara listeden kaldır",
"share": "Paylaş",
- "mini_player": "Mini Oynatıcı",
+ "mini_player": "Mini oynatıcı",
"slide_to_seek": "İleri veya geri arama yapmak için kaydırın",
- "shuffle_playlist": "Çalma listesini karıştır",
- "unshuffle_playlist": "Karışık çalma listesi",
+ "shuffle_playlist": "Oynatma listesini karıştır",
+ "unshuffle_playlist": "Oynatma listesinin karışıklığını kaldır",
"previous_track": "Önceki parça",
"next_track": "Sonraki parça",
- "pause_playback": "Çalmayı Duraklat",
- "resume_playback": "Çalmaya Devam Et",
+ "pause_playback": "Oynatmayı duraklat",
+ "resume_playback": "Oynatmayı sürdür",
"loop_track": "Döngü parçası",
- "repeat_playlist": "Çalma listesini tekrarla",
- "queue": "Sıra",
+ "repeat_playlist": "Oynatma listesini tekrarla",
+ "queue": "Kuyruk",
"alternative_track_sources": "Alternatif parça kaynakları",
"download_track": "Parçayı indir",
- "tracks_in_queue": "{tracks} sıradaki parçalar",
+ "tracks_in_queue": "{tracks} parça kuyrukta",
"clear_all": "Tümünü temizle",
- "show_hide_ui_on_hover": "Üzerine gelindiğinde kullanıcı arayüzünü göster/gizle",
- "always_on_top": "Her zaman en üstte",
+ "show_hide_ui_on_hover": "Fareyle üzerine gelindiğinde kullanıcı arayüzünü göster/gizle",
+ "always_on_top": "Her zaman üstte",
"exit_mini_player": "Mini oynatıcıdan çık",
"download_location": "İndirme konumu",
"account": "Hesap",
- "login_with_spotify": "Spotify hesabınız ile giriş yapın",
- "connect_with_spotify": "Spotify ile bağlantı kurun",
- "logout": "Çıkış Yap",
- "logout_of_this_account": "Bu hesaptan çıkış yap",
- "language_region": "Dil & Bölge",
- "language": "Dil",
- "system_default": "Sistem Varsayılanı",
- "market_place_region": "Mevcut Bölge",
- "recommendation_country": "Tavsiye Edilen Ülke",
+ "login_with_spotify": "Spotify hesabı ile giriş yap",
+ "connect_with_spotify": "Spotify ile bağlan",
+ "logout": "Çıkış yap",
+ "logout_of_this_account": "Hesaptan çıkış yap",
+ "language_region": "Dil ve bölge",
+ "language": "Tercih edilen dil",
+ "system_default": "Sistem varsayılanı",
+ "market_place_region": "Tercih edilen bölge",
+ "recommendation_country": "Tavsiye edilen ülke",
"appearance": "Görünüm",
- "layout_mode": "Düzen Modu",
+ "layout_mode": "Düzen modu",
"override_layout_settings": "Duyarlı düzen modu ayarlarını geçersiz kıl",
"adaptive": "Uyarlanabilir",
"compact": "Sıkıştırılmış",
"extended": "Genişletilmiş",
"theme": "Tema",
- "dark": "Karanlık",
- "light": "Aydınlık",
+ "dark": "Koyu",
+ "light": "Açık",
"system": "Sistem",
- "accent_color": "Vurgu Rengi",
- "sync_album_color": "Albüm rengini eşitle",
- "sync_album_color_description": "Albüm resminin baskın rengini vurgu rengi olarak kullanır",
- "playback": "Çalma",
- "audio_quality": "Ses Kalitesi",
+ "accent_color": "Vurgu rengi",
+ "sync_album_color": "Albüm rengini senkronize et",
+ "sync_album_color_description": "Vurgu rengi olarak albüm resminin baskın rengini kullanır",
+ "playback": "Oynatma",
+ "audio_quality": "Ses kalitesi",
"high": "Yüksek",
"low": "Düşük",
"pre_download_play": "Önceden indir ve oynat",
- "pre_download_play_description": "Ses akışı yerine, baytları indirin ve oynatın (Daha yüksek bant genişliği kullanıcıları için önerilir)",
- "skip_non_music": "Müzik olmayan bölümleri atla (SponsorBlock)",
+ "pre_download_play_description": "Ses akışı yerine baytları indir ve oynat (Daha yüksek bant genişliğine sahip kullanıcılar için önerilir)",
+ "skip_non_music": "Müzik olmayan bölümleri atlat (SponsorBlock)",
"blacklist_description": "Kara listeye alınan parçalar ve sanatçılar",
- "wait_for_download_to_finish": "Lütfen mevcut indirme işleminin bitmesini bekleyin",
+ "wait_for_download_to_finish": "Lütfen mevcut indirme işleminin tamamlanmasını bekleyin",
"desktop": "Masaüstü",
- "close_behavior": "Yakın Davranış",
+ "close_behavior": "Kapatma davranışı",
"close": "Kapat",
"minimize_to_tray": "Tepsiye küçült",
"show_tray_icon": "Sistem tepsisi simgesini göster",
"about": "Hakkında",
"u_love_spotube": "Spotube'u sevdiğinizi biliyoruz",
"check_for_updates": "Güncellemeleri kontrol et",
- "about_spotube": "Spotube Hakkında",
- "blacklist": "Kara Liste",
- "please_sponsor": "Lütfen Sponsor Olun/Bağış Yapın",
- "spotube_description": "Spotube, hafif, platformlar arası, herkesin kullanabileceği ücretsiz bir Spotify istemcisidir.",
+ "about_spotube": "Spotube hakkında",
+ "blacklist": "Kara liste",
+ "please_sponsor": "Sponsor Ol/Bağış Yap",
+ "spotube_description": "Spotube, hafif, platformlar arası uyumlu ve herkes için ücretsiz bir Spotify istemcisidir.",
"version": "Sürüm",
- "build_number": "Derleme Numarası",
- "founder": "Kurucu",
+ "build_number": "Derleme numarası",
+ "founder": "Geliştirici",
"repository": "Depo",
"bug_issues": "Hata + Sorunlar",
- "made_with": "❤️ ile Bangladesh🇧🇩 adresinde yapılmıştır.",
+ "made_with": "❤️ ile Bangladeş'te yapıldı",
"kingkor_roy_tirtho": "Kingkor Roy Tirtho",
"copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
"license": "Lisans",
- "add_spotify_credentials": "Başlamak için spotify bilgilerinizi ekleyin",
- "credentials_will_not_be_shared_disclaimer": "Endişelenmeyin, bilgileriniz toplanmayacak veya kimseyle paylaşılmayacak",
- "know_how_to_login": "Nasıl yapılacağını bilmiyor musunuz?",
- "follow_step_by_step_guide": "Adım Adım kılavuzu takip edin",
- "spotify_cookie": "Spotify {name} Çerez",
- "cookie_name_cookie": "{name} Çerez",
+ "add_spotify_credentials": "Başlamak için spotify kimlik bilgilerinizi ekleyin",
+ "credentials_will_not_be_shared_disclaimer": "Endişelenmeyin, kimlik bilgilerinizden hiçbiri toplanmayacak veya kimseyle paylaşılmayacak",
+ "know_how_to_login": "Bunu nasıl yapacağınızı bilmiyor musunuz?",
+ "follow_step_by_step_guide": "Adım adım kılavuzu takip edin",
+ "spotify_cookie": "Spotify {name} çerezi",
+ "cookie_name_cookie": "{name} çerezi",
"fill_in_all_fields": "Lütfen tüm alanları doldurun",
- "submit": "Gönder",
+ "submit": "Başvur",
"exit": "Çık",
"previous": "Önceki",
"next": "Sonraki",
"done": "Bitti",
"step_1": "1. Adım",
- "first_go_to": "İlk önce şu adrese gidin",
- "login_if_not_logged_in": "ve oturum açmadıysanız Giriş Yapın/Kaydolun",
+ "first_go_to": "İlk olarak şuraya gidin:",
+ "login_if_not_logged_in": "ve oturum açmadıysanız Oturum açın/Kaydolun",
"step_2": "2. Adım",
- "step_2_steps": "1. Giriş yaptıktan sonra, Tarayıcı devtools.\n2'yi açmak için F12'ye basın veya Fare Sağ Tıklaması > İncele'ye basın. Ardından \"Uygulama\" Sekmesine (Chrome, Edge, Brave vb.) veya \"Depolama\" Sekmesine (Firefox, Palemoon vb.) gidin\n3. \"Çerezler\" bölümüne ve ardından \"https://accounts.spotify.com\" alt bölümüne gidin",
+ "step_2_steps": "1. Oturum açtıktan sonra, tarayıcı geliştirme araçlarını açmak için F12'ye veya fareye sağ tıklayın > İncele'ye basın.\n2. Daha sonra \"Uygulama\" sekmesine (Chrome, Edge, Brave vb..) veya \"Depolama\" sekmesine (Firefox, Palemoon vb..) gidin\n3. \"Çerezler\" bölümüne, ardından \"https://accounts.spotify.com\" alt bölümüne gidin",
"step_3": "3. Adım",
+ "step_3_steps": "\"sp_dc\" Çerezinin değerini kopyalayın",
"success_emoji": "Başarılı🥳",
- "success_message": "Şimdi Spotify hesabınızla başarılı bir şekilde oturum açtınız. İyi iş, dostum!",
+ "success_message": "Artık Spotify hesabınızla başarıyla giriş yaptınız. Tebrik ederim!",
"step_4": "4. Adım",
- "something_went_wrong": "Bir şeyler ters gitti",
- "piped_instance": "Piped Sunucu Örneği",
+ "step_4_steps": "Kopyalanan \"sp_dc\" değerini yapıştırın",
+ "something_went_wrong": "Bir hata oluştu",
+ "piped_instance": "Piped sunucu örneği",
"piped_description": "Parça eşleştirme için kullanılacak Piped sunucu örneği",
- "piped_warning": "Bazıları iyi çalışmayabilir. Bu yüzden riski size ait olmak üzere kullanın",
- "generate_playlist": "Çalma Listesi Oluştur",
- "track_exists": "Track {track} zaten mevcut",
+ "piped_warning": "Bazıları iyi çalışmayabilir. Yani riski size ait olmak üzere kullanın",
+ "generate_playlist": "Oynatma listesi oluştur",
+ "track_exists": "{track} parçası zaten var",
"replace_downloaded_tracks": "İndirilen tüm parçaları değiştir",
"skip_download_tracks": "İndirilen tüm parçaları indirmeyi atla",
- "do_you_want_to_replace": "Mevcut parçayı değiştirmek mi istiyorsunuz?",
+ "do_you_want_to_replace": "Mevcut parçayı değiştirmek istiyor musunuz?",
"replace": "Değiştir",
"skip": "Atla",
"select_up_to_count_type": "En fazla {count} {type} seçin",
- "select_genres": "Tür Seç",
- "add_genres": "Tür Ekle",
+ "select_genres": "Türleri seç",
+ "add_genres": "Tür ekle",
"country": "Ülke",
"number_of_tracks_generate": "Oluşturulacak parça sayısı",
"acousticness": "Akustiklik",
- "danceability": "Dansedilebilirlik",
+ "danceability": "Dans Edilebilirlik",
"energy": "Enerji",
- "instrumentalness": "Enstrümansallık",
+ "instrumentalness": "Araçsallık",
"liveness": "Canlılık",
- "loudness": "Yükseklik",
+ "loudness": "Ses yüksekliği",
"speechiness": "Konuşkanlık",
- "valence": "Değerlilik",
+ "valence": "Değerlik",
"popularity": "Popülerlik",
"key": "Anahtar",
"duration": "Süre (sn)",
"tempo": "Tempo (BPM)",
"mode": "Mod",
- "time_signature": "Zaman İmzası",
+ "time_signature": "Zaman imzası",
"short": "Kısa",
"medium": "Orta",
"long": "Uzun",
@@ -217,74 +220,172 @@
"max": "Maks",
"target": "Hedef",
"moderate": "Orta",
- "deselect_all": "Tüm Seçimleri Kaldır",
- "select_all": "Tümünü Seç",
+ "deselect_all": "Tüm seçimleri kaldır",
+ "select_all": "Tümünü seç",
"are_you_sure": "Emin misiniz?",
- "generating_playlist": "Özel çalma listenizi oluşturun...",
- "selected_count_tracks": "Seçilen {count} parçalar",
- "download_warning": "Tüm Parçaları toplu olarak indirirseniz, açıkça Müzik korsanlığı yapmış ve yaratıcı Müzik toplumuna zarar vermiş olursunuz. Umarım bunun farkındasınızdır. Her zaman, Sanatçıların sıkı çalışmalarına saygı duymayı ve desteklemeyi deneyin",
- "download_ip_ban_warning": "Bu arada, normalden fazla indirme isteği nedeniyle IP adresiniz YouTube'da engellenebilir. IP engeli, o IP cihazından en az 2-3 ay boyunca YouTube'u (giriş yapmış olsanız bile) kullanamayacağınız anlamına gelir. Ve Spotube böyle bir durumda herhangi bir sorumluluk kabul etmez",
- "by_clicking_accept_terms": "'Kabul et' seçeneğine tıklayarak aşağıdaki şartları kabul etmiş olursunuz:",
- "download_agreement_1": "Müzik korsanlığı yaptığımı biliyorum. Ben malım.",
- "download_agreement_2": "Sanatçıları elimden geldiğince destekleyeceğim ve bunu sadece sanatlarını satın alacak param olmadığı için yapıyorum",
- "download_agreement_3": "IP adresimin YouTube'da engellenebileceğinin tamamen farkındayım ve mevcut eylemimin neden olduğu herhangi bir kazadan Spotube'u veya sahiplerini/dağıtıcılarını sorumlu tutmuyorum",
+ "generating_playlist": "Özel oynatma listeniz oluşturuluyor...",
+ "selected_count_tracks": "{count} parça seçildi",
+ "download_warning": "Tüm şarkıları toplu olarak indiriyorsanız, açıkça müzik korsanlığı yapıyorsunuz ve müzik dünyasının yaratıcı topluluğuna zarar veriyorsunuz demektir. Umuyorum bunun farkındasınızdır. Her zaman, sanatçıların emeğine saygı göstermeyi ve desteklemeyi deneyin.",
+ "download_ip_ban_warning": "Ayrıca, normalden fazla indirme istekleri nedeniyle YouTube'da IP'niz engellenebilir. IP engeli, en az 2-3 ay boyunca YouTube'u (hatta oturum açmış olsanız bile) o IP cihazından kullanamayacağınız anlamına gelir. Ve eğer böyle bir durum yaşanırsa, Spotube bundan hiçbir sorumluluk kabul etmez.",
+ "by_clicking_accept_terms": "\"Kabul et\" e tıklayarak aşağıdaki şartları kabul etmiş olursunuz:",
+ "download_agreement_1": "Müzik korsanlığı yaptığımı biliyorum. Ben fakir biriyim.",
+ "download_agreement_2": "Sanatçıyı elimden geldiğince destekleyeceğim ve bunu sadece sanatını satın alacak param olmadığı için yapıyorum",
+ "download_agreement_3": "YouTube'da IP'min engellenebileceğinin tamamen farkındayım ve mevcut eylemlerimden kaynaklanan herhangi bir kaza için Spotube'u veya sahiplerini/katkıda bulunanları sorumlu tutmuyorum.",
"decline": "Reddet",
"accept": "Kabul et",
"details": "Detaylar",
"youtube": "YouTube",
"channel": "Kanal",
- "likes": "Beğeniler",
- "dislikes": "Beğenmemeler",
+ "likes": "Beğenenler",
+ "dislikes": "Beğenmeyenler",
"views": "İzlenmeler",
- "streamUrl": "Yayın Bağlantısı",
- "stop": "Dur",
- "sort_newest": "En yeni eklenene göre sırala",
+ "streamUrl": "Akış bağlantısı",
+ "stop": "Durdur",
+ "sort_newest": "En yeni eklenene göre sırala.",
"sort_oldest": "En eski eklenene göre sırala",
"sleep_timer": "Uyku Zamanlayıcısı",
- "mins": "{minutes} Dakikalar",
- "hours": "{hours} Saat",
- "hour": "{hours} Saatler",
+ "mins": "{minutes} Dakika",
+ "hours": "{hours} Saatler",
+ "hour": "{hours} Saat",
"custom_hours": "Özel Saatler",
"logs": "Günlükler",
"developers": "Geliştiriciler",
"not_logged_in": "Giriş yapmadınız",
- "search_mode": "Arama Modu",
- "audio_source": "Ses Kaynağı",
+ "search_mode": "Arama modu",
+ "audio_source": "Ses kaynağı",
"ok": "Tamam",
"failed_to_encrypt": "Şifreleme başarısız oldu",
- "encryption_failed_warning": "Spotube, verilerinizi güvenli bir şekilde depolamak için şifreleme kullanır. Ancak bunu başaramadı. Bu nedenle güvensiz bir depolamaya geri dönecektir. Linux kullanıyorsanız, lütfen gnome-keyring, kde-wallet, keepassxc vb. gibi bir güvenlik hizmetinizin kurulu olduğundan emin olun.",
+ "encryption_failed_warning": "Spotube, verilerinizi güvenli bir şekilde depolamak için şifreleme kullanır. Ancak bunu başaramadı. Bu nedenle, güvensiz depolamaya geri dönecektir\nLinux kullanıyorsanız, lütfen gnome-keyring, kde-wallet, keepassxc vb. herhangi bir gizli servisin yüklü olduğundan emin olun.",
"querying_info": "Bilgi sorgulanıyor...",
"piped_api_down": "Piped API kapalı",
- "piped_down_error_instructions": "Piped örneği {pipedInstance} şu anda kapalı\n\nYa örneği değiştirin ya da '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",
+ "piped_down_error_instructions": "Piped örneği {pipedInstance} şu anda kapalı\n\nÖrneği değiştirin veya 'API türünü' resmi YouTube API'si olarak değiştirin\n\nDeğişiklikten sonra uygulamayı yeniden başlattığınızdan emin olun",
"you_are_offline": "Şu anda çevrimdışısınız",
- "connection_restored": "İnternet bağlantınız yeniden kuruldu",
+ "connection_restored": "İnternet bağlantınız geri yüklendi",
"use_system_title_bar": "Sistem başlık çubuğunu kullan",
- "crunching_results": "Sonuçlar kırılıyor...",
- "search_to_get_results": "Sonuç almak için arama yap",
- "use_amoled_mode": "AMOLED modunu kullan",
- "pitch_dark_theme": "Zifiri siyah dart teması",
+ "crunching_results": "Sonuçlar...",
+ "search_to_get_results": "Sonuç almak için arayın",
+ "use_amoled_mode": "AMOLED modu kullan",
+ "pitch_dark_theme": "Zifiri karanlık koyu tema",
"normalize_audio": "Sesi normalleştir",
"change_cover": "Kapağı değiştir",
"add_cover": "Kapak ekle",
"restore_defaults": "Varsayılanları geri yükle",
- "download_music_codec": "Müzik codec bileşenini indirin",
- "streaming_music_codec": "Müzik akışı codec bileşeni",
+ "download_music_codec": "Müzik codec bileşenini indir",
+ "streaming_music_codec": "Müzik codec'i akışı",
"login_with_lastfm": "Last.fm ile giriş yap",
"connect": "Bağlan",
"disconnect_lastfm": "Last.fm bağlantısını kes",
- "disconnect": "Bağlantıyı Kes",
- "username": "Kullanıcı Adı",
+ "disconnect": "Bağlantıyı kes",
+ "username": "Kullanıcı adı",
"password": "Şifre",
- "login": "Giriş Yap",
- "login_with_your_lastfm": "Last.fm hesabınız ile giriş yapın",
+ "login": "Giriş yap",
+ "login_with_your_lastfm": "Last.fm hesabınızla giriş yapın",
"scrobble_to_lastfm": "Last.fm için Scrobble",
- "go_to_album": "Albüme Git",
- "discord_rich_presence": "Discord Zengin Varlık",
- "browse_all": "Tümünü Gözat",
- "genres": "Müzik Türleri",
- "explore_genres": "Türleri Keşfet",
- "step_3_steps": "\"sp_dc\" Çerezinin değerini kopyala",
- "step_4_steps": "Kopyalanan \"sp_dc\" değerini yapıştır",
+ "go_to_album": "Albüme git",
+ "discord_rich_presence": "Discord zengin varlığı",
+ "browse_all": "Tümüne göz at",
+ "genres": "Müzik türleri",
+ "explore_genres": "Türleri keşfet",
"friends": "Arkadaşlar",
- "no_lyrics_available": "Üzgünüz, bu parça için şarkı sözleri bulunamıyor"
+ "no_lyrics_available": "Üzgünüz, bu parçanın sözleri bulunamıyor",
+ "start_a_radio": "Radyo başlat",
+ "how_to_start_radio": "Radyoyu nasıl başlatmak istersiniz?",
+ "replace_queue_question": "Mevcut kuyruğu değiştirmek mi yoksa eklemek mi istersiniz?",
+ "endless_playback": "Sonsuz olarak oynat",
+ "delete_playlist": "Oynatma listesini sil",
+ "delete_playlist_confirmation": "Bu oynatma listesini silmek istediğinizden emin misiniz?",
+ "local_tracks": "Yerel parçalar",
+ "song_link": "Şarkı bağlantısı",
+ "skip_this_nonsense": "Bu saçmalığı atla",
+ "freedom_of_music": "“Müzik özgürlüğü”",
+ "freedom_of_music_palm": "“Müzik özgürlüğü avucunuzun içinde”",
+ "get_started": "Haydi başlayalım",
+ "youtube_source_description": "Tavsiye edilir ve en iyi şekilde çalışır.",
+ "piped_source_description": "Özgür hissediyor musunuz? YouTube ile aynı, ama çok daha özgür.",
+ "jiosaavn_source_description": "Güney Asya bölgesi için en iyisi.",
+ "highest_quality": "En yüksek kalite: {quality}",
+ "select_audio_source": "Ses kaynağını seçin",
+ "endless_playback_description": "Yeni şarkıları otomatik olarak\nkuyruğun sonuna ekle",
+ "choose_your_region": "Bölgenizi seçin",
+ "choose_your_region_description": "Bu, Spotube'un konumunuza uygun içerikleri göstermesine yardımcı olacaktır.",
+ "choose_your_language": "Dilinizi seçin",
+ "help_project_grow": "Bu projenin büyümesine yardımcı olun",
+ "help_project_grow_description": "Spotube açık kaynaklı bir projedir. Projeye katkıda bulunarak, hataları bildirerek veya yeni özellikler önererek bu projenin büyümesine yardımcı olabilirsiniz.",
+ "contribute_on_github": "GitHub'da katkıda bulun",
+ "donate_on_open_collective": "Open Collective'de bağış yap",
+ "browse_anonymously": "Anonim olarak giriş yap",
+ "enable_connect": "Bağlanmayı etkinleştir",
+ "enable_connect_description": "Spotube'u diğer cihazlardan kontrol edin",
+ "devices": "Cihazlar",
+ "select": "Seç",
+ "connect_client_alert": "{client} tarafından kontrol ediliyorsun.",
+ "this_device": "Bu cihaz",
+ "remote": "Yönet",
+ "local_library": "Yerel kütüphane",
+ "add_library_location": "Kütüphaneye ekle",
+ "remove_library_location": "Kütüphaneden çıkar",
+ "local_tab": "Yerel",
+ "stats": "İstatistikler",
+ "and_n_more": "ve {count} daha",
+ "recently_played": "Son Çalınanlar",
+ "browse_more": "Daha Fazla Göz At",
+ "no_title": "Başlık Yok",
+ "not_playing": "Çalmıyor",
+ "epic_failure": "Efsanevi başarısızlık!",
+ "added_num_tracks_to_queue": "{tracks_length} şarkı sıraya eklendi",
+ "spotube_has_an_update": "Spotube bir güncelleme aldı",
+ "download_now": "Şimdi İndir",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} yayımlandı",
+ "release_version": "Spotube v{version} yayımlandı",
+ "read_the_latest": "Son haberleri oku",
+ "release_notes": "sürüm notları",
+ "pick_color_scheme": "Renk şeması seç",
+ "save": "Kaydet",
+ "choose_the_device": "Cihazı seçin:",
+ "multiple_device_connected": "Birden fazla cihaz bağlı.\nBu işlemi gerçekleştirmek istediğiniz cihazı seçin",
+ "nothing_found": "Hiçbir şey bulunamadı",
+ "the_box_is_empty": "Kutu boş",
+ "top_artists": "En İyi Sanatçılar",
+ "top_albums": "En İyi Albümler",
+ "this_week": "Bu hafta",
+ "this_month": "Bu ay",
+ "last_6_months": "Son 6 ay",
+ "this_year": "Bu yıl",
+ "last_2_years": "Son 2 yıl",
+ "all_time": "Tüm zamanlar",
+ "powered_by_provider": "{providerName} tarafından desteklenmektedir",
+ "email": "E-posta",
+ "profile_followers": "Takipçiler",
+ "birthday": "Doğum Günü",
+ "subscription": "Abonelik",
+ "not_born": "Henüz doğmadı",
+ "hacker": "Hacker",
+ "profile": "Profil",
+ "no_name": "İsim Yok",
+ "edit": "Düzenle",
+ "user_profile": "Kullanıcı Profili",
+ "count_plays": "{count} çalma",
+ "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.",
+ "count_mins": "{minutes} dk",
+ "summary_minutes": "dakika",
+ "summary_listened_to_music": "Dinlenen müzik",
+ "summary_songs": "şarkılar",
+ "summary_streamed_overall": "Genel olarak akış",
+ "summary_owed_to_artists": "Sanatçılara borç\nbu ay",
+ "summary_artists": "sanatçının",
+ "summary_music_reached_you": "Müzik sana ulaştı",
+ "summary_full_albums": "tam albümler",
+ "summary_got_your_love": "Sevgini aldı",
+ "summary_playlists": "çalma listeleri",
+ "summary_were_on_repeat": "Tekrarda vardı",
+ "total_money": "Toplam {money}",
+ "minutes_listened": "Dinlenilen Dakikalar",
+ "streamed_songs": "Yayınlanan Şarkılar",
+ "count_streams": "{count} yayın",
+ "owned_by_you": "Sahip olduğunuz",
+ "copied_shareurl_to_clipboard": "{shareUrl} panoya kopyalandı",
+ "spotify_hipotetical_calculation": "*Bu, Spotify'ın her yayın başına ödemenin\n$0.003 ile $0.005 arasında olduğu varsayımıyla hesaplanmıştır. Bu\nhipotetik bir hesaplamadır, kullanıcıya şarkılarını Spotify'da dinlediklerinde\nsanatçılara ne kadar ödeme yapacaklarını gösterir.",
+ "webview_not_found": "Webview bulunamadı",
+ "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",
+ "unsupported_platform": "Desteklenmeyen platform"
}
\ No newline at end of file
diff --git a/lib/l10n/app_uk.arb b/lib/l10n/app_uk.arb
index a4586a5e..0c65f756 100644
--- a/lib/l10n/app_uk.arb
+++ b/lib/l10n/app_uk.arb
@@ -286,5 +286,106 @@
"step_3_steps": "Скопіюйте значення cookie \"sp_dc\"",
"step_4_steps": "Вставте скопійоване значення \"sp_dc\"",
"friends": "Друзі",
- "no_lyrics_available": "Вибачте, не вдалося знайти текст для цього треку"
+ "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": "Автоматично додавати нові пісні\nв кінець черги",
+ "choose_your_region": "Виберіть ваш регіон",
+ "choose_your_region_description": "Це допоможе Spotube показати вам правильний контент\nдля вашого місцезнаходження.",
+ "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розрахунок, щоб дати уявлення користувачу про те, скільки б він\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}",
+ "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.",
+ "webview_not_found": "Webview не знайдено",
+ "webview_not_found_description": "На вашому пристрої не встановлено виконуване середовище Webview.\nЯкщо воно встановлено, переконайтеся, що воно знаходиться в environment PATH\n\nПісля встановлення перезапустіть програму",
+ "unsupported_platform": "Непідтримувана платформа"
}
\ No newline at end of file
diff --git a/lib/l10n/app_vi.arb b/lib/l10n/app_vi.arb
index d8d337c2..75dc1532 100644
--- a/lib/l10n/app_vi.arb
+++ b/lib/l10n/app_vi.arb
@@ -284,5 +284,108 @@
"discord_rich_presence": "Hiển thị trạng thái Discord",
"browse_all": "Duyệt tất cả",
"genres": "Thể loại",
- "explore_genres": "Khám phá Thể loại"
-}
+ "explore_genres": "Khám phá Thể loại",
+ "sort_duration": "Sắp xếp theo Thời lượng",
+ "start_a_radio": "Bắt đầu Một Đài phát thanh",
+ "how_to_start_radio": "Bạn muốn bắt đầu đài phát thanh như thế nào?",
+ "replace_queue_question": "Bạn muốn thay thế hàng đợi hiện tại hay thêm vào?",
+ "endless_playback": "Phát không giới hạn",
+ "delete_playlist": "Xóa Danh sách phát",
+ "delete_playlist_confirmation": "Bạn có chắc chắn muốn xóa danh sách phát này không?",
+ "local_tracks": "Bài hát Địa phương",
+ "song_link": "Liên kết Bài hát",
+ "skip_this_nonsense": "Bỏ qua bớt rối này",
+ "freedom_of_music": "“Sự Tự do của Âm nhạc”",
+ "freedom_of_music_palm": "“Sự Tự do của Âm nhạc trong lòng bàn tay của bạn”",
+ "get_started": "Bắt đầu thôi",
+ "youtube_source_description": "Được đề xuất và hoạt động tốt nhất.",
+ "piped_source_description": "Cảm thấy tự do? Giống như YouTube nhưng miễn phí hơn rất nhiều.",
+ "jiosaavn_source_description": "Tốt nhất cho khu vực Nam Á.",
+ "highest_quality": "Chất lượng Tốt nhất: {quality}",
+ "select_audio_source": "Chọn Nguồn Âm thanh",
+ "endless_playback_description": "Tự động thêm các bài hát mới\nvào cuối hàng đợi",
+ "choose_your_region": "Chọn khu vực của bạn",
+ "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.",
+ "choose_your_language": "Chọn ngôn ngữ của bạn",
+ "help_project_grow": "Hãy giúp dự án này phát triển",
+ "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.",
+ "contribute_on_github": "Đóng góp trên GitHub",
+ "donate_on_open_collective": "Quyên góp trên Open Collective",
+ "browse_anonymously": "Duyệt Anonymously",
+ "friends": "Bạn bè",
+ "no_lyrics_available": "Xin lỗi, không tìm thấy lời cho bài hát này",
+ "enable_connect": "Kích hoạt kết nối",
+ "enable_connect_description": "Điều khiển Spotube từ các thiết bị khác",
+ "devices": "Thiết bị",
+ "select": "Chọn",
+ "connect_client_alert": "Bạn đang được điều khiển bởi {client}",
+ "this_device": "Thiết bị này",
+ "remote": "Từ xa",
+ "local_library": "Thư viện địa phương",
+ "add_library_location": "Thêm vào thư viện",
+ "remove_library_location": "Xóa khỏi thư viện",
+ "local_tab": "Địa phương",
+ "stats": "Thống kê",
+ "and_n_more": "và {count} cái khác",
+ "recently_played": "Gần đây đã phát",
+ "browse_more": "Xem thêm",
+ "no_title": "Không có tiêu đề",
+ "not_playing": "Không phát",
+ "epic_failure": "Thất bại hoàn toàn!",
+ "added_num_tracks_to_queue": "Đã thêm {tracks_length} bài hát vào danh sách phát",
+ "spotube_has_an_update": "Spotube có bản cập nhật",
+ "download_now": "Tải về ngay",
+ "nightly_version": "Spotube Nightly {nightlyBuildNum} đã được phát hành",
+ "release_version": "Spotube v{version} đã được phát hành",
+ "read_the_latest": "Đọc tin mới nhất",
+ "release_notes": "ghi chú phát hành",
+ "pick_color_scheme": "Chọn chủ đề màu sắc",
+ "save": "Lưu",
+ "choose_the_device": "Chọn thiết bị:",
+ "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",
+ "nothing_found": "Không tìm thấy gì",
+ "the_box_is_empty": "Hộp trống",
+ "top_artists": "Những Nghệ Sĩ Hàng Đầu",
+ "top_albums": "Những Album Hàng Đầu",
+ "this_week": "Tuần này",
+ "this_month": "Tháng này",
+ "last_6_months": "6 tháng qua",
+ "this_year": "Năm nay",
+ "last_2_years": "2 năm qua",
+ "all_time": "Mọi thời đại",
+ "powered_by_provider": "Cung cấp bởi {providerName}",
+ "email": "Email",
+ "profile_followers": "Người theo dõi",
+ "birthday": "Ngày sinh",
+ "subscription": "Gói cước",
+ "not_born": "Chưa sinh",
+ "hacker": "Tin tặc",
+ "profile": "Hồ sơ",
+ "no_name": "Không có tên",
+ "edit": "Chỉnh sửa",
+ "user_profile": "Hồ sơ người dùng",
+ "count_plays": "{count} lần phát",
+ "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.",
+ "count_mins": "{minutes} phút",
+ "summary_minutes": "phút",
+ "summary_listened_to_music": "Đã nghe nhạc",
+ "summary_songs": "bài hát",
+ "summary_streamed_overall": "Stream tổng cộng",
+ "summary_owed_to_artists": "Nợ nghệ sĩ\ntrong tháng này",
+ "summary_artists": "nghệ sĩ",
+ "summary_music_reached_you": "Âm nhạc đã đến với bạn",
+ "summary_full_albums": "album đầy đủ",
+ "summary_got_your_love": "Nhận được tình yêu của bạn",
+ "summary_playlists": "danh sách phát",
+ "summary_were_on_repeat": "Đã được phát lại",
+ "total_money": "Tổng cộng {money}",
+ "minutes_listened": "Thời gian nghe",
+ "streamed_songs": "Bài hát đã phát",
+ "count_streams": "{count} lượt phát",
+ "owned_by_you": "Thuộc sở hữu của bạn",
+ "copied_shareurl_to_clipboard": "{shareUrl} đã sao chép vào bảng tạm",
+ "spotify_hipotetical_calculation": "*Được tính toán dựa trên khoản thanh toán của Spotify cho mỗi lượt phát\ntừ $0.003 đến $0.005. Đây là một tính toán giả định để\ncung cấp cho người dùng cái nhìn về số tiền họ sẽ phải trả\ncho các nghệ sĩ nếu họ nghe bài hát của họ trên Spotify.",
+ "webview_not_found": "Không tìm thấy Webview",
+ "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",
+ "unsupported_platform": "Nền tảng không được hỗ trợ"
+}
\ No newline at end of file
diff --git a/lib/l10n/app_zh.arb b/lib/l10n/app_zh.arb
index 20fdb329..c9bf35df 100644
--- a/lib/l10n/app_zh.arb
+++ b/lib/l10n/app_zh.arb
@@ -286,5 +286,106 @@
"step_3_steps": "复制\"sp_dc\" Cookie的值",
"step_4_steps": "粘贴复制的\"sp_dc\"值",
"friends": "朋友",
- "no_lyrics_available": "抱歉,无法找到此曲的歌词"
+ "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 运行时。\n如果已安装,请确保它在 environment PATH 中\n\n安装后,重新启动应用程序",
+ "unsupported_platform": "不支持的平台"
}
\ No newline at end of file
diff --git a/lib/l10n/l10n.dart b/lib/l10n/l10n.dart
index 7aec682a..ebdc4b61 100644
--- a/lib/l10n/l10n.dart
+++ b/lib/l10n/l10n.dart
@@ -7,10 +7,15 @@
/// TexturedPolak@github => Polish
/// yuri-val@github => Ukrainian
/// energywave@github, ncvescera@github, OpenCode@github => Italian
-/// mdksec@github => Turkish
+/// mikropsoft@github => Turkish
/// Stephan-P@github, SecularSteve@github => Dutch
/// doannc2212@github => Vietnamese
/// sappho192@github => Korean
+/// watchakorn-18k@github => Thai
+/// Microsoft Copilot, Tutislav@github => Czech
+
+library l10n;
+
import 'package:flutter/material.dart';
class L10n {
@@ -19,22 +24,28 @@ class L10n {
const Locale('ar', 'SA'),
const Locale('bn', 'BD'),
const Locale('ca', 'AD'),
+ const Locale('cs', 'CZ'),
const Locale('de', 'GE'),
const Locale('es', 'ES'),
- const Locale("fa", "IR"),
+ const Locale('fa', 'IR'),
+ const Locale('fi', 'FI'),
const Locale('fr', 'FR'),
const Locale('ne', 'NP'),
const Locale('hi', 'IN'),
+ const Locale('id', 'ID'),
const Locale('it', 'IT'),
const Locale('ja', 'JP'),
+ const Locale('ka', 'GE'),
const Locale('ko', 'KR'),
const Locale('nl', 'NL'),
const Locale('pl', 'PL'),
const Locale('pt', 'PT'),
const Locale('ru', 'RU'),
const Locale('uk', 'UA'),
+ const Locale('th', 'TH'),
const Locale('tr', 'TR'),
const Locale('zh', 'CN'),
const Locale('vi', 'VN'),
+ const Locale('eu', 'ES'),
];
}
diff --git a/lib/main.dart b/lib/main.dart
index 31c1da57..f13991e2 100644
--- a/lib/main.dart
+++ b/lib/main.dart
@@ -1,184 +1,139 @@
-import 'package:catcher_2/catcher_2.dart';
-import 'package:dart_discord_rpc/dart_discord_rpc.dart';
-import 'package:device_preview/device_preview.dart';
-import 'package:fl_query/fl_query.dart';
+import 'dart:async';
+import 'dart:ui';
+
+import 'package:desktop_webview_window/desktop_webview_window.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
-import 'package:flutter_desktop_tools/flutter_desktop_tools.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: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:shared_preferences/shared_preferences.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/intents.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';
+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/provider/audio_player/audio_player_streams.dart';
+import 'package:spotube/provider/database/database.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/models/logger.dart';
-import 'package:spotube/models/skip_segment.dart';
-import 'package:spotube/models/source_match.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';
-import 'package:spotube/services/connectivity_adapter.dart';
+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/persisted_state_notifier.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:spotube/hooks/configurators/use_init_sys_tray.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';
Future main(List rawArgs) async {
+ if (rawArgs.contains("web_view_title_bar")) {
+ WidgetsFlutterBinding.ensureInitialized();
+ if (runWebViewTitleBarWidget(rawArgs)) {
+ return;
+ }
+ }
final arguments = await startCLI(rawArgs);
+ AppLogger.initialize(arguments["verbose"]);
- final widgetsBinding = WidgetsFlutterBinding.ensureInitialized();
+ AppLogger.runZoned(() async {
+ final widgetsBinding = WidgetsFlutterBinding.ensureInitialized();
- await registerWindowsScheme("spotify");
+ await registerWindowsScheme("spotify");
- FlutterNativeSplash.preserve(widgetsBinding: widgetsBinding);
+ tz.initializeTimeZones();
- MediaKit.ensureInitialized();
+ FlutterNativeSplash.preserve(widgetsBinding: widgetsBinding);
- // force High Refresh Rate on some Android devices (like One Plus)
- if (DesktopTools.platform.isAndroid) {
- await FlutterDisplayMode.setHighRefreshRate();
- }
+ MediaKit.ensureInitialized();
- if (DesktopTools.platform.isDesktop) {
- await DesktopTools.window.setPreventClose(true);
- }
+ await migrateMacOsFromSandboxToNoSandbox();
- await SystemTheme.accentColor.load();
+ // force High Refresh Rate on some Android devices (like One Plus)
+ if (kIsAndroid) {
+ await FlutterDisplayMode.setHighRefreshRate();
+ }
- if (!kIsWeb) {
- MetadataGod.initialize();
- }
+ if (kIsDesktop) {
+ await windowManager.setPreventClose(true);
+ }
- if (DesktopTools.platform.isWindows || DesktopTools.platform.isLinux) {
- DiscordRPC.initialize();
- }
+ await SystemTheme.accentColor.load();
- await KVStoreService.initialize();
- KVStoreService.doneGettingStarted = false;
+ if (!kIsWeb) {
+ MetadataGod.initialize();
+ }
- final hiveCacheDir =
- kIsWeb ? null : (await getApplicationSupportDirectory()).path;
+ if (kIsDesktop) {
+ await FlutterDiscordRPC.initialize(Env.discordAppId);
+ }
- await QueryClient.initialize(
- cachePrefix: "oss.krtirtho.spotube",
- cacheDir: hiveCacheDir,
- connectivity: FlQueryInternetConnectionCheckerAdapter(),
- );
+ if (kIsWindows) {
+ await SMTCWindows.initialize();
+ }
- Hive.registerAdapter(SkipSegmentAdapter());
+ await KVStoreService.initialize();
+ await EncryptedKvStoreService.initialize();
- Hive.registerAdapter(SourceMatchAdapter());
- Hive.registerAdapter(SourceTypeAdapter());
+ final hiveCacheDir =
+ kIsWeb ? null : (await getApplicationSupportDirectory()).path;
- // Cache versioning entities with Adapter
- SourceMatch.version = 'v1';
- SkipSegment.version = 'v1';
+ Hive.init(hiveCacheDir);
- await Hive.openLazyBox(
- SourceMatch.boxName,
- path: hiveCacheDir,
- );
- await Hive.openLazyBox(
- SkipSegment.boxName,
- path: hiveCacheDir,
- );
- await PersistedStateNotifier.initializeBoxes(
- path: hiveCacheDir,
- );
+ final database = AppDatabase();
- await DesktopTools.ensureInitialized(
- DesktopWindowOptions(
- hideTitleBar: true,
- title: "Spotube",
- backgroundColor: Colors.transparent,
- minimumSize: const Size(300, 700),
- ),
- );
+ await migrateFromHiveToDrift(database);
- Catcher2(
- enableLogger: arguments["verbose"],
- debugConfig: Catcher2Options(
- SilentReportMode(),
- [
- ConsoleHandler(
- enableDeviceParameters: false,
- enableApplicationParameters: false,
- ),
- if (!kIsWeb) FileHandler(await getLogsPath(), printLogs: false),
- ],
- ),
- releaseConfig: Catcher2Options(
- SilentReportMode(),
- [
- if (arguments["verbose"] ?? false) ConsoleHandler(),
- if (!kIsWeb)
- FileHandler(
- await getLogsPath(),
- printLogs: false,
- ),
- ],
- ),
- runAppFunction: () {
- runApp(
- DevicePreview(
- availableLocales: L10n.all,
- enabled: false,
- data: const DevicePreviewData(
- isEnabled: false,
- orientation: Orientation.portrait,
- ),
- builder: (context) {
- return ProviderScope(
- child: QueryClientProvider(
- staleDuration: const Duration(minutes: 30),
- child: const Spotube(),
- ),
- );
- },
- ),
- );
- },
- );
+ if (kIsDesktop) {
+ await localNotifier.setup(appName: "Spotube");
+ await WindowManagerTools.initialize();
+ }
+
+ runApp(
+ ProviderScope(
+ overrides: [
+ databaseProvider.overrideWith((ref) => database),
+ ],
+ observers: const [
+ AppLoggerProviderObserver(),
+ ],
+ child: const Spotube(),
+ ),
+ );
+ });
}
-class Spotube extends StatefulHookConsumerWidget {
- const Spotube({Key? key}) : super(key: key);
+class Spotube extends HookConsumerWidget {
+ const Spotube({super.key});
@override
- SpotubeState createState() => SpotubeState();
-
- static SpotubeState of(BuildContext context) =>
- context.findAncestorStateOfType()!;
-}
-
-class SpotubeState extends ConsumerState {
- final logger = getLogger(Spotube);
- SharedPreferences? localStorage;
-
- @override
- void initState() {
- super.initState();
- SharedPreferences.getInstance().then(((value) => localStorage = value));
- }
-
- @override
- Widget build(BuildContext context) {
+ Widget build(BuildContext context, ref) {
final themeMode =
ref.watch(userPreferencesProvider.select((s) => s.themeMode));
final accentMaterialColor =
@@ -189,15 +144,23 @@ class SpotubeState extends ConsumerState {
final paletteColor =
ref.watch(paletteProvider.select((s) => s?.dominantColor?.color));
final router = ref.watch(routerProvider);
+ final hasTouchSupport = useHasTouch();
+ ref.listen(audioPlayerStreamListenersProvider, (_, __) {});
+ ref.listen(bonsoirProvider, (_, __) {});
+ ref.listen(connectClientsProvider, (_, __) {});
+ ref.listen(serverProvider, (_, __) {});
+ ref.listen(trayManagerProvider, (_, __) {});
+
+ useFixWindowStretching();
useDisableBatteryOptimizations();
- useInitSysTray(ref);
useDeepLinking(ref);
useCloseBehavior(ref);
useGetStoragePermissions(ref);
useEffect(() {
FlutterNativeSplash.remove();
+
return () {
/// For enabling hot reload for audio player
if (!kDebugMode) return;
@@ -231,12 +194,22 @@ class SpotubeState extends ConsumerState {
debugShowCheckedModeBanner: false,
title: 'Spotube',
builder: (context, child) {
- return DevicePreview.appBuilder(
- context,
- DesktopTools.platform.isDesktop
- ? DragToResizeArea(child: child!)
- : child,
+ child = ScrollConfiguration(
+ behavior: ScrollConfiguration.of(context).copyWith(
+ dragDevices: hasTouchSupport
+ ? {
+ PointerDeviceKind.touch,
+ PointerDeviceKind.stylus,
+ PointerDeviceKind.invertedStylus,
+ }
+ : null,
+ ),
+ child: child!,
);
+
+ if (kIsDesktop && !kIsMacOS) child = DragToResizeArea(child: child);
+
+ return child;
},
themeMode: themeMode,
theme: lightTheme,
diff --git a/lib/models/connect/connect.dart b/lib/models/connect/connect.dart
new file mode 100644
index 00000000..a70520ad
--- /dev/null
+++ b/lib/models/connect/connect.dart
@@ -0,0 +1,15 @@
+library connect;
+
+import 'dart:async';
+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/provider/audio_player/state.dart';
+
+part 'connect.freezed.dart';
+part 'connect.g.dart';
+
+part 'ws_event.dart';
+part 'load.dart';
diff --git a/lib/models/connect/connect.freezed.dart b/lib/models/connect/connect.freezed.dart
new file mode 100644
index 00000000..088cfbd1
--- /dev/null
+++ b/lib/models/connect/connect.freezed.dart
@@ -0,0 +1,610 @@
+// 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 'connect.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');
+
+WebSocketLoadEventData _$WebSocketLoadEventDataFromJson(
+ Map json) {
+ switch (json['runtimeType']) {
+ case 'playlist':
+ return WebSocketLoadEventDataPlaylist.fromJson(json);
+ case 'album':
+ return WebSocketLoadEventDataAlbum.fromJson(json);
+
+ default:
+ throw CheckedFromJsonException(
+ json,
+ 'runtimeType',
+ 'WebSocketLoadEventData',
+ 'Invalid union type "${json['runtimeType']}"!');
+ }
+}
+
+/// @nodoc
+mixin _$WebSocketLoadEventData {
+ @JsonKey(name: 'tracks', toJson: _tracksJson)
+ 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,
+ int? initialIndex)
+ playlist,
+ required TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex)
+ album,
+ }) =>
+ throw _privateConstructorUsedError;
+ @optionalTypeArgs
+ TResult? whenOrNull({
+ TResult? Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ PlaylistSimple? collection,
+ int? initialIndex)?
+ playlist,
+ TResult? Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex)?
+ album,
+ }) =>
+ throw _privateConstructorUsedError;
+ @optionalTypeArgs
+ TResult maybeWhen({
+ TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ PlaylistSimple? collection,
+ int? initialIndex)?
+ playlist,
+ TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex)?
+ album,
+ required TResult orElse(),
+ }) =>
+ throw _privateConstructorUsedError;
+ @optionalTypeArgs
+ TResult map({
+ required TResult Function(WebSocketLoadEventDataPlaylist value) playlist,
+ required TResult Function(WebSocketLoadEventDataAlbum value) album,
+ }) =>
+ throw _privateConstructorUsedError;
+ @optionalTypeArgs
+ TResult? mapOrNull({
+ TResult? Function(WebSocketLoadEventDataPlaylist value)? playlist,
+ TResult? Function(WebSocketLoadEventDataAlbum value)? album,
+ }) =>
+ throw _privateConstructorUsedError;
+ @optionalTypeArgs
+ TResult maybeMap({
+ TResult Function(WebSocketLoadEventDataPlaylist value)? playlist,
+ TResult Function(WebSocketLoadEventDataAlbum value)? album,
+ required TResult orElse(),
+ }) =>
+ throw _privateConstructorUsedError;
+ Map toJson() => throw _privateConstructorUsedError;
+ @JsonKey(ignore: true)
+ $WebSocketLoadEventDataCopyWith get copyWith =>
+ throw _privateConstructorUsedError;
+}
+
+/// @nodoc
+abstract class $WebSocketLoadEventDataCopyWith<$Res> {
+ factory $WebSocketLoadEventDataCopyWith(WebSocketLoadEventData value,
+ $Res Function(WebSocketLoadEventData) then) =
+ _$WebSocketLoadEventDataCopyWithImpl<$Res, WebSocketLoadEventData>;
+ @useResult
+ $Res call(
+ {@JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ int? initialIndex});
+}
+
+/// @nodoc
+class _$WebSocketLoadEventDataCopyWithImpl<$Res,
+ $Val extends WebSocketLoadEventData>
+ implements $WebSocketLoadEventDataCopyWith<$Res> {
+ _$WebSocketLoadEventDataCopyWithImpl(this._value, this._then);
+
+ // ignore: unused_field
+ final $Val _value;
+ // ignore: unused_field
+ final $Res Function($Val) _then;
+
+ @pragma('vm:prefer-inline')
+ @override
+ $Res call({
+ Object? tracks = null,
+ Object? initialIndex = freezed,
+ }) {
+ return _then(_value.copyWith(
+ tracks: null == tracks
+ ? _value.tracks
+ : tracks // ignore: cast_nullable_to_non_nullable
+ as List,
+ initialIndex: freezed == initialIndex
+ ? _value.initialIndex
+ : initialIndex // ignore: cast_nullable_to_non_nullable
+ as int?,
+ ) as $Val);
+ }
+}
+
+/// @nodoc
+abstract class _$$WebSocketLoadEventDataPlaylistImplCopyWith<$Res>
+ implements $WebSocketLoadEventDataCopyWith<$Res> {
+ factory _$$WebSocketLoadEventDataPlaylistImplCopyWith(
+ _$WebSocketLoadEventDataPlaylistImpl value,
+ $Res Function(_$WebSocketLoadEventDataPlaylistImpl) then) =
+ __$$WebSocketLoadEventDataPlaylistImplCopyWithImpl<$Res>;
+ @override
+ @useResult
+ $Res call(
+ {@JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ PlaylistSimple? collection,
+ int? initialIndex});
+}
+
+/// @nodoc
+class __$$WebSocketLoadEventDataPlaylistImplCopyWithImpl<$Res>
+ extends _$WebSocketLoadEventDataCopyWithImpl<$Res,
+ _$WebSocketLoadEventDataPlaylistImpl>
+ implements _$$WebSocketLoadEventDataPlaylistImplCopyWith<$Res> {
+ __$$WebSocketLoadEventDataPlaylistImplCopyWithImpl(
+ _$WebSocketLoadEventDataPlaylistImpl _value,
+ $Res Function(_$WebSocketLoadEventDataPlaylistImpl) _then)
+ : super(_value, _then);
+
+ @pragma('vm:prefer-inline')
+ @override
+ $Res call({
+ Object? tracks = null,
+ Object? collection = freezed,
+ Object? initialIndex = freezed,
+ }) {
+ return _then(_$WebSocketLoadEventDataPlaylistImpl(
+ tracks: null == tracks
+ ? _value._tracks
+ : tracks // ignore: cast_nullable_to_non_nullable
+ as List,
+ collection: freezed == collection
+ ? _value.collection
+ : collection // ignore: cast_nullable_to_non_nullable
+ as PlaylistSimple?,
+ initialIndex: freezed == initialIndex
+ ? _value.initialIndex
+ : initialIndex // ignore: cast_nullable_to_non_nullable
+ as int?,
+ ));
+ }
+}
+
+/// @nodoc
+@JsonSerializable()
+class _$WebSocketLoadEventDataPlaylistImpl
+ extends WebSocketLoadEventDataPlaylist {
+ _$WebSocketLoadEventDataPlaylistImpl(
+ {@JsonKey(name: 'tracks', toJson: _tracksJson)
+ required final List tracks,
+ this.collection,
+ this.initialIndex,
+ final String? $type})
+ : _tracks = tracks,
+ $type = $type ?? 'playlist',
+ super._();
+
+ factory _$WebSocketLoadEventDataPlaylistImpl.fromJson(
+ Map json) =>
+ _$$WebSocketLoadEventDataPlaylistImplFromJson(json);
+
+ final List _tracks;
+ @override
+ @JsonKey(name: 'tracks', toJson: _tracksJson)
+ List get tracks {
+ if (_tracks is EqualUnmodifiableListView) return _tracks;
+ // ignore: implicit_dynamic_type
+ return EqualUnmodifiableListView(_tracks);
+ }
+
+ @override
+ final PlaylistSimple? collection;
+ @override
+ final int? initialIndex;
+
+ @JsonKey(name: 'runtimeType')
+ final String $type;
+
+ @override
+ String toString() {
+ return 'WebSocketLoadEventData.playlist(tracks: $tracks, collection: $collection, initialIndex: $initialIndex)';
+ }
+
+ @override
+ bool operator ==(Object other) {
+ return identical(this, other) ||
+ (other.runtimeType == runtimeType &&
+ other is _$WebSocketLoadEventDataPlaylistImpl &&
+ const DeepCollectionEquality().equals(other._tracks, _tracks) &&
+ (identical(other.collection, collection) ||
+ other.collection == collection) &&
+ (identical(other.initialIndex, initialIndex) ||
+ other.initialIndex == initialIndex));
+ }
+
+ @JsonKey(ignore: true)
+ @override
+ int get hashCode => Object.hash(runtimeType,
+ const DeepCollectionEquality().hash(_tracks), collection, initialIndex);
+
+ @JsonKey(ignore: true)
+ @override
+ @pragma('vm:prefer-inline')
+ _$$WebSocketLoadEventDataPlaylistImplCopyWith<
+ _$WebSocketLoadEventDataPlaylistImpl>
+ get copyWith => __$$WebSocketLoadEventDataPlaylistImplCopyWithImpl<
+ _$WebSocketLoadEventDataPlaylistImpl>(this, _$identity);
+
+ @override
+ @optionalTypeArgs
+ TResult when({
+ required TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ PlaylistSimple? collection,
+ int? initialIndex)
+ playlist,
+ required TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex)
+ album,
+ }) {
+ return playlist(tracks, collection, initialIndex);
+ }
+
+ @override
+ @optionalTypeArgs
+ TResult? whenOrNull({
+ TResult? Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ PlaylistSimple? collection,
+ int? initialIndex)?
+ playlist,
+ TResult? Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex)?
+ album,
+ }) {
+ return playlist?.call(tracks, collection, initialIndex);
+ }
+
+ @override
+ @optionalTypeArgs
+ TResult maybeWhen({
+ TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ PlaylistSimple? collection,
+ int? initialIndex)?
+ playlist,
+ TResult Function(
+ @JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex)?
+ album,
+ required TResult orElse(),
+ }) {
+ if (playlist != null) {
+ return playlist(tracks, collection, initialIndex);
+ }
+ return orElse();
+ }
+
+ @override
+ @optionalTypeArgs
+ TResult map({
+ required TResult Function(WebSocketLoadEventDataPlaylist value) playlist,
+ required TResult Function(WebSocketLoadEventDataAlbum value) album,
+ }) {
+ return playlist(this);
+ }
+
+ @override
+ @optionalTypeArgs
+ TResult? mapOrNull({
+ TResult? Function(WebSocketLoadEventDataPlaylist value)? playlist,
+ TResult? Function(WebSocketLoadEventDataAlbum value)? album,
+ }) {
+ return playlist?.call(this);
+ }
+
+ @override
+ @optionalTypeArgs
+ TResult maybeMap({
+ TResult Function(WebSocketLoadEventDataPlaylist value)? playlist,
+ TResult Function(WebSocketLoadEventDataAlbum value)? album,
+ required TResult orElse(),
+ }) {
+ if (playlist != null) {
+ return playlist(this);
+ }
+ return orElse();
+ }
+
+ @override
+ Map toJson() {
+ return _$$WebSocketLoadEventDataPlaylistImplToJson(
+ this,
+ );
+ }
+}
+
+abstract class WebSocketLoadEventDataPlaylist extends WebSocketLoadEventData {
+ factory WebSocketLoadEventDataPlaylist(
+ {@JsonKey(name: 'tracks', toJson: _tracksJson)
+ required final List tracks,
+ final PlaylistSimple? collection,
+ final int? initialIndex}) = _$WebSocketLoadEventDataPlaylistImpl;
+ WebSocketLoadEventDataPlaylist._() : super._();
+
+ factory WebSocketLoadEventDataPlaylist.fromJson(Map json) =
+ _$WebSocketLoadEventDataPlaylistImpl.fromJson;
+
+ @override
+ @JsonKey(name: 'tracks', toJson: _tracksJson)
+ List get tracks;
+ @override
+ PlaylistSimple? get collection;
+ @override
+ int? get initialIndex;
+ @override
+ @JsonKey(ignore: true)
+ _$$WebSocketLoadEventDataPlaylistImplCopyWith<
+ _$WebSocketLoadEventDataPlaylistImpl>
+ get copyWith => throw _privateConstructorUsedError;
+}
+
+/// @nodoc
+abstract class _$$WebSocketLoadEventDataAlbumImplCopyWith<$Res>
+ implements $WebSocketLoadEventDataCopyWith<$Res> {
+ factory _$$WebSocketLoadEventDataAlbumImplCopyWith(
+ _$WebSocketLoadEventDataAlbumImpl value,
+ $Res Function(_$WebSocketLoadEventDataAlbumImpl) then) =
+ __$$WebSocketLoadEventDataAlbumImplCopyWithImpl<$Res>;
+ @override
+ @useResult
+ $Res call(
+ {@JsonKey(name: 'tracks', toJson: _tracksJson) List tracks,
+ AlbumSimple? collection,
+ int? initialIndex});
+}
+
+/// @nodoc
+class __$$WebSocketLoadEventDataAlbumImplCopyWithImpl<$Res>
+ extends _$WebSocketLoadEventDataCopyWithImpl<$Res,
+ _$WebSocketLoadEventDataAlbumImpl>
+ implements _$$WebSocketLoadEventDataAlbumImplCopyWith<$Res> {
+ __$$WebSocketLoadEventDataAlbumImplCopyWithImpl(
+ _$WebSocketLoadEventDataAlbumImpl _value,
+ $Res Function(_$WebSocketLoadEventDataAlbumImpl) _then)
+ : super(_value, _then);
+
+ @pragma('vm:prefer-inline')
+ @override
+ $Res call({
+ Object? tracks = null,
+ Object? collection = freezed,
+ Object? initialIndex = freezed,
+ }) {
+ return _then(_$WebSocketLoadEventDataAlbumImpl(
+ tracks: null == tracks
+ ? _value._tracks
+ : tracks // ignore: cast_nullable_to_non_nullable
+ as List,
+ collection: freezed == collection
+ ? _value.collection
+ : collection // ignore: cast_nullable_to_non_nullable
+ as AlbumSimple?,
+ initialIndex: freezed == initialIndex
+ ? _value.initialIndex
+ : initialIndex // ignore: cast_nullable_to_non_nullable
+ as int?,
+ ));
+ }
+}
+
+/// @nodoc
+@JsonSerializable()
+class _$WebSocketLoadEventDataAlbumImpl extends WebSocketLoadEventDataAlbum {
+ _$WebSocketLoadEventDataAlbumImpl(
+ {@JsonKey(name: 'tracks', toJson: _tracksJson)
+ required final List tracks,
+ this.collection,
+ this.initialIndex,
+ final String? $type})
+ : _tracks = tracks,
+ $type = $type ?? 'album',
+ super._();
+
+ factory _$WebSocketLoadEventDataAlbumImpl.fromJson(
+ Map