Compare commits
1 Commits
fix/androi
...
renovate/p
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
302055c415 |
@@ -1,25 +0,0 @@
|
|||||||
# Custom EAS Build config for Android phone APK (downloadable artifact).
|
|
||||||
# Same bun-forcing flow as android-production.yml, but builds an APK
|
|
||||||
# (assembleRelease) instead of an AAB — for sideloading / GitHub artifact.
|
|
||||||
# Referenced from eas.json: build.production-apk.android.config
|
|
||||||
build:
|
|
||||||
name: Android phone APK (bun)
|
|
||||||
steps:
|
|
||||||
- eas/checkout
|
|
||||||
|
|
||||||
- run:
|
|
||||||
name: Install dependencies (bun, frozen)
|
|
||||||
command: bun install --frozen-lockfile
|
|
||||||
|
|
||||||
- run:
|
|
||||||
name: Prebuild (Android, bun)
|
|
||||||
command: bunx expo prebuild --platform android --no-install
|
|
||||||
|
|
||||||
- eas/configure_android_version
|
|
||||||
- eas/inject_android_credentials
|
|
||||||
|
|
||||||
- eas/run_gradle:
|
|
||||||
inputs:
|
|
||||||
command: :app:assembleRelease
|
|
||||||
|
|
||||||
- eas/find_and_upload_build_artifacts
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
# Custom EAS Build config for Android TV APK (downloadable artifact).
|
|
||||||
# Same bun-forcing flow, with EXPO_TV=1 (set via the profile env in
|
|
||||||
# eas.json) so prebuild generates the TV variant. Builds an APK for
|
|
||||||
# sideloading onto Android TV devices.
|
|
||||||
# Referenced from eas.json: build.production-apk-tv.android.config
|
|
||||||
build:
|
|
||||||
name: Android TV APK (bun)
|
|
||||||
steps:
|
|
||||||
- eas/checkout
|
|
||||||
|
|
||||||
- run:
|
|
||||||
name: Install dependencies (bun, frozen)
|
|
||||||
command: bun install --frozen-lockfile
|
|
||||||
|
|
||||||
# EXPO_TV=1 comes from the profile env, so prebuild targets Android TV.
|
|
||||||
- run:
|
|
||||||
name: Prebuild (Android TV, bun)
|
|
||||||
command: bunx expo prebuild --platform android --no-install
|
|
||||||
|
|
||||||
- eas/configure_android_version
|
|
||||||
- eas/inject_android_credentials
|
|
||||||
|
|
||||||
- eas/run_gradle:
|
|
||||||
inputs:
|
|
||||||
command: :app:assembleRelease
|
|
||||||
|
|
||||||
- eas/find_and_upload_build_artifacts
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
# Custom EAS Build config for Android (production AAB).
|
|
||||||
#
|
|
||||||
# Why this exists: EAS's managed build can't detect Bun's text lockfile
|
|
||||||
# (bun.lock) and falls back to yarn, which breaks our install. The managed
|
|
||||||
# steps `eas/install_node_modules` and `eas/prebuild` both use "the package
|
|
||||||
# manager detected based on your project", so we replace them with explicit
|
|
||||||
# `bun` commands. Everything else uses EAS's built-in functions so we still
|
|
||||||
# get remote versioning, credentials, and artifact upload.
|
|
||||||
#
|
|
||||||
# Referenced from eas.json: build.production.android.config = android-production.yml
|
|
||||||
build:
|
|
||||||
name: Android production (bun)
|
|
||||||
steps:
|
|
||||||
- eas/checkout
|
|
||||||
|
|
||||||
- run:
|
|
||||||
name: Install dependencies (bun, frozen)
|
|
||||||
command: bun install --frozen-lockfile
|
|
||||||
|
|
||||||
# android/ is gitignored, so generate native code fresh. --no-install
|
|
||||||
# because deps are already installed above; bunx keeps it on bun.
|
|
||||||
- run:
|
|
||||||
name: Prebuild (Android, bun)
|
|
||||||
command: bunx expo prebuild --platform android --no-install
|
|
||||||
|
|
||||||
# Applies the EAS-resolved remote versionCode/versionName (autoIncrement
|
|
||||||
# in eas.json) into the freshly prebuilt android/ project.
|
|
||||||
- eas/configure_android_version
|
|
||||||
|
|
||||||
# Injects the remote Android keystore / signing config.
|
|
||||||
- eas/inject_android_credentials
|
|
||||||
|
|
||||||
# Build the Play Store app bundle (.aab).
|
|
||||||
- eas/run_gradle:
|
|
||||||
inputs:
|
|
||||||
command: :app:bundleRelease
|
|
||||||
|
|
||||||
- eas/find_and_upload_build_artifacts
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
# Custom EAS Build config for iOS + tvOS (App Store), forcing bun.
|
|
||||||
#
|
|
||||||
# Shared by both the iPhone profile (production) and the tvOS profile
|
|
||||||
# (production_tv). The profile decides the rest:
|
|
||||||
# - production_tv sets EXPO_TV=1 (env) so prebuild targets tvOS, and
|
|
||||||
# credentialsSource: local (EAS can't manage tvOS creds remotely).
|
|
||||||
# - production uses remote-managed iOS credentials.
|
|
||||||
#
|
|
||||||
# Like the Android configs, this replaces eas/install_node_modules and
|
|
||||||
# eas/prebuild (both auto-detect the wrong package manager) with explicit
|
|
||||||
# bun commands, and keeps EAS built-ins for credentials/version/fastlane.
|
|
||||||
build:
|
|
||||||
name: iOS/tvOS App Store (bun)
|
|
||||||
steps:
|
|
||||||
- eas/checkout
|
|
||||||
|
|
||||||
- run:
|
|
||||||
name: Install dependencies (bun, frozen)
|
|
||||||
command: bun install --frozen-lockfile
|
|
||||||
|
|
||||||
- eas/resolve_apple_team_id_from_credentials:
|
|
||||||
id: resolve_team
|
|
||||||
|
|
||||||
# android/ + ios/ are gitignored, so generate native code fresh.
|
|
||||||
# EXPO_TV (from the profile env) selects iPhone vs tvOS. --no-install
|
|
||||||
# skips JS + pod install; we install pods explicitly below with bun deps.
|
|
||||||
- run:
|
|
||||||
name: Prebuild (iOS/tvOS, bun)
|
|
||||||
command: bunx expo prebuild --platform ios --no-install
|
|
||||||
|
|
||||||
- run:
|
|
||||||
name: Install CocoaPods
|
|
||||||
working_directory: ./ios
|
|
||||||
command: pod install
|
|
||||||
|
|
||||||
- eas/configure_ios_credentials
|
|
||||||
- eas/configure_ios_version
|
|
||||||
|
|
||||||
- eas/generate_gymfile_from_template:
|
|
||||||
inputs:
|
|
||||||
credentials: ${ eas.job.secrets.buildCredentials }
|
|
||||||
|
|
||||||
- eas/run_fastlane
|
|
||||||
- eas/find_and_upload_build_artifacts
|
|
||||||
216
.github/workflows/release.yml
vendored
@@ -1,216 +0,0 @@
|
|||||||
name: 🚀 Release (EAS build + submit)
|
|
||||||
|
|
||||||
# On merge to main (gated by the `production` GitHub Environment approval),
|
|
||||||
# build all targets on EAS in parallel via custom bun build configs:
|
|
||||||
# 1. iOS phone → App Store (auto-submit)
|
|
||||||
# 2. tvOS → App Store (auto-submit)
|
|
||||||
# 3. Android AAB → Google Play (auto-submit)
|
|
||||||
# 4. Android phone APK→ downloadable artifact
|
|
||||||
# 5. Android TV APK → downloadable artifact
|
|
||||||
# Note: EAS queues builds based on your plan's concurrency; parallel jobs
|
|
||||||
# here just submit them — EAS may still run them serially.
|
|
||||||
|
|
||||||
concurrency:
|
|
||||||
group: release-${{ github.ref }}
|
|
||||||
cancel-in-progress: false
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [main]
|
|
||||||
workflow_dispatch:
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
approve:
|
|
||||||
name: 🔐 Approve release
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
environment: production
|
|
||||||
steps:
|
|
||||||
- name: ✅ Release approved
|
|
||||||
run: echo "Release approved for ${{ github.sha }}"
|
|
||||||
|
|
||||||
build:
|
|
||||||
name: 🚀 ${{ matrix.name }}
|
|
||||||
needs: approve
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
- name: 🍎 iOS
|
|
||||||
platform: ios
|
|
||||||
profile: production
|
|
||||||
submit: true
|
|
||||||
- name: 📺 tvOS
|
|
||||||
platform: ios
|
|
||||||
profile: production_tv
|
|
||||||
submit: true
|
|
||||||
- name: 🤖 Android AAB
|
|
||||||
platform: android
|
|
||||||
profile: production
|
|
||||||
submit: true
|
|
||||||
- name: 🤖 Android APK
|
|
||||||
platform: android
|
|
||||||
profile: production-apk
|
|
||||||
submit: false
|
|
||||||
artifact_name: streamyfin-android-phone-apk
|
|
||||||
- name: 📺 Android TV APK
|
|
||||||
platform: android
|
|
||||||
profile: production-apk-tv
|
|
||||||
submit: false
|
|
||||||
artifact_name: streamyfin-android-tv-apk
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: 📥 Checkout code
|
|
||||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
|
||||||
with:
|
|
||||||
fetch-depth: 0
|
|
||||||
submodules: recursive
|
|
||||||
show-progress: false
|
|
||||||
|
|
||||||
- name: 🍞 Setup Bun
|
|
||||||
uses: oven-sh/setup-bun@0c5077e51419868618aeaa5fe8019c62421857d6 # v2.2.0
|
|
||||||
with:
|
|
||||||
bun-version: latest
|
|
||||||
|
|
||||||
- name: 💾 Cache Bun dependencies
|
|
||||||
uses: actions/cache@27d5ce7f107fe9357f9df03efb73ab90386fccae # v5.0.5
|
|
||||||
with:
|
|
||||||
path: ~/.bun/install/cache
|
|
||||||
key: ${{ runner.os }}-bun-cache-${{ hashFiles('bun.lock') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-bun-cache
|
|
||||||
|
|
||||||
- name: 📦 Install dependencies and reload submodules
|
|
||||||
run: |
|
|
||||||
bun install --frozen-lockfile
|
|
||||||
bun run submodule-reload
|
|
||||||
|
|
||||||
- name: 🏗️ Setup EAS
|
|
||||||
uses: expo/expo-github-action@b184ff86a3c926240f1b6db41764c83a01c02eef # main
|
|
||||||
with:
|
|
||||||
eas-version: latest
|
|
||||||
token: ${{ secrets.EXPO_TOKEN }}
|
|
||||||
eas-cache: true
|
|
||||||
|
|
||||||
# tvOS uses credentialsSource: local — restore the gitignored
|
|
||||||
# credentials.json + cert + provisioning profiles from secrets.
|
|
||||||
- name: 🔐 Restore tvOS signing credentials
|
|
||||||
if: matrix.profile == 'production_tv'
|
|
||||||
env:
|
|
||||||
EAS_CREDENTIALS_JSON: ${{ secrets.EAS_CREDENTIALS_JSON }}
|
|
||||||
TVOS_DIST_CERT_P12_BASE64: ${{ secrets.TVOS_DIST_CERT_P12_BASE64 }}
|
|
||||||
TVOS_APP_PROFILE_BASE64: ${{ secrets.TVOS_APP_PROFILE_BASE64 }}
|
|
||||||
TVOS_TOPSHELF_PROFILE_BASE64: ${{ secrets.TVOS_TOPSHELF_PROFILE_BASE64 }}
|
|
||||||
run: |
|
|
||||||
mkdir -p certs profiles
|
|
||||||
printf '%s' "$EAS_CREDENTIALS_JSON" > credentials.json
|
|
||||||
echo "$TVOS_DIST_CERT_P12_BASE64" | base64 -d > certs/distribution.p12
|
|
||||||
echo "$TVOS_APP_PROFILE_BASE64" | base64 -d > profiles/Streamyfin_tvOS_App_Store.mobileprovision
|
|
||||||
echo "$TVOS_TOPSHELF_PROFILE_BASE64" | base64 -d > profiles/Streamyfin_TopShelf_tvOS_App_Store.mobileprovision
|
|
||||||
|
|
||||||
# Android Play submit needs the Google Play service account JSON.
|
|
||||||
- name: 🔐 Restore Google Play service account
|
|
||||||
if: matrix.platform == 'android' && matrix.submit
|
|
||||||
env:
|
|
||||||
GOOGLE_SERVICE_ACCOUNT_KEY: ${{ secrets.GOOGLE_SERVICE_ACCOUNT_KEY }}
|
|
||||||
run: printf '%s' "$GOOGLE_SERVICE_ACCOUNT_KEY" > google-service-account.json
|
|
||||||
|
|
||||||
# App Store Connect API key for iOS/tvOS submit (raw-PEM or base64).
|
|
||||||
- name: 🔐 Restore App Store Connect API key
|
|
||||||
if: matrix.platform == 'ios'
|
|
||||||
env:
|
|
||||||
APPLE_KEY_CONTENT: ${{ secrets.APPLE_KEY_CONTENT }}
|
|
||||||
run: |
|
|
||||||
if printf '%s' "$APPLE_KEY_CONTENT" | grep -q "BEGIN PRIVATE KEY"; then
|
|
||||||
printf '%s' "$APPLE_KEY_CONTENT" > "$RUNNER_TEMP/asc_api_key.p8"
|
|
||||||
else
|
|
||||||
printf '%s' "$APPLE_KEY_CONTENT" | base64 -d > "$RUNNER_TEMP/asc_api_key.p8"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# ── Submit builds: cloud build + auto-submit to the store ──
|
|
||||||
- name: 🚀 Build & submit (${{ matrix.name }})
|
|
||||||
if: matrix.submit
|
|
||||||
env:
|
|
||||||
EXPO_TOKEN: ${{ secrets.EXPO_TOKEN }}
|
|
||||||
EXPO_ASC_API_KEY_PATH: ${{ runner.temp }}/asc_api_key.p8
|
|
||||||
EXPO_ASC_KEY_ID: ${{ secrets.APPLE_KEY_ID }}
|
|
||||||
EXPO_ASC_ISSUER_ID: ${{ secrets.APPLE_KEY_ISSUER_ID }}
|
|
||||||
run: |
|
|
||||||
eas build \
|
|
||||||
--platform ${{ matrix.platform }} \
|
|
||||||
--profile ${{ matrix.profile }} \
|
|
||||||
--auto-submit \
|
|
||||||
--non-interactive \
|
|
||||||
--wait
|
|
||||||
|
|
||||||
# ── Artifact builds: cloud build, then download + upload the APK ──
|
|
||||||
- name: 🏗️ Build artifact (${{ matrix.name }})
|
|
||||||
if: ${{ !matrix.submit }}
|
|
||||||
env:
|
|
||||||
EXPO_TOKEN: ${{ secrets.EXPO_TOKEN }}
|
|
||||||
run: |
|
|
||||||
eas build \
|
|
||||||
--platform ${{ matrix.platform }} \
|
|
||||||
--profile ${{ matrix.profile }} \
|
|
||||||
--non-interactive \
|
|
||||||
--wait \
|
|
||||||
--json > build-result.json
|
|
||||||
URL=$(node -e "const b=require('./build-result.json'); const x=Array.isArray(b)?b[0]:b; console.log(x.artifacts.applicationArchiveUrl)")
|
|
||||||
echo "Downloading artifact: $URL"
|
|
||||||
curl -fL "$URL" -o "${{ matrix.artifact_name }}.apk"
|
|
||||||
|
|
||||||
- name: 📤 Upload APK artifact (${{ matrix.name }})
|
|
||||||
if: ${{ !matrix.submit }}
|
|
||||||
uses: actions/upload-artifact@043fb46d1a93c77aae656e7c1c64a875d1fc6a0a # v7.0.1
|
|
||||||
with:
|
|
||||||
name: ${{ matrix.artifact_name }}
|
|
||||||
path: ${{ matrix.artifact_name }}.apk
|
|
||||||
retention-days: 14
|
|
||||||
|
|
||||||
# Draft a GitHub Release with the two APKs attached. The tag comes from the
|
|
||||||
# merged-in app version (app.json → expo.version), NOT the auto-incremented
|
|
||||||
# build number — so cutting a release is a deliberate version bump via PR.
|
|
||||||
github-release:
|
|
||||||
name: 📦 Draft GitHub Release
|
|
||||||
needs: build
|
|
||||||
if: ${{ !cancelled() }}
|
|
||||||
runs-on: ubuntu-24.04
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
actions: read # required for `gh run download` to list/fetch this run's artifacts
|
|
||||||
steps:
|
|
||||||
- name: 📥 Checkout code
|
|
||||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6.0.2
|
|
||||||
with:
|
|
||||||
fetch-depth: 0
|
|
||||||
show-progress: false
|
|
||||||
|
|
||||||
- name: 📦 Download APK artifacts from this run
|
|
||||||
env:
|
|
||||||
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
run: |
|
|
||||||
mkdir -p apks
|
|
||||||
gh run download ${{ github.run_id }} --name streamyfin-android-phone-apk --dir apks
|
|
||||||
gh run download ${{ github.run_id }} --name streamyfin-android-tv-apk --dir apks
|
|
||||||
ls -la apks
|
|
||||||
|
|
||||||
- name: 📝 Draft release (tag = app.json version, not auto-bumped)
|
|
||||||
env:
|
|
||||||
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
run: |
|
|
||||||
VERSION=$(node -e "console.log(require('./app.json').expo.version)")
|
|
||||||
TAG="v$VERSION"
|
|
||||||
echo "Release tag from merged app version: $TAG"
|
|
||||||
if gh release view "$TAG" >/dev/null 2>&1; then
|
|
||||||
echo "Release $TAG exists — updating APK assets"
|
|
||||||
gh release upload "$TAG" apks/*.apk --clobber
|
|
||||||
else
|
|
||||||
echo "Creating draft release $TAG"
|
|
||||||
gh release create "$TAG" \
|
|
||||||
--draft \
|
|
||||||
--generate-notes \
|
|
||||||
--title "$TAG" \
|
|
||||||
apks/*.apk
|
|
||||||
fi
|
|
||||||
9
.gitignore
vendored
@@ -18,9 +18,6 @@ web-build/
|
|||||||
/androidmobile
|
/androidmobile
|
||||||
/androidtv
|
/androidtv
|
||||||
|
|
||||||
# Gradle caches (top-level + per-module native projects)
|
|
||||||
**/.gradle/
|
|
||||||
|
|
||||||
# Module-specific Builds
|
# Module-specific Builds
|
||||||
modules/mpv-player/android/build
|
modules/mpv-player/android/build
|
||||||
modules/player/android
|
modules/player/android
|
||||||
@@ -76,12 +73,6 @@ modules/background-downloader/android/build/*
|
|||||||
|
|
||||||
# ios:unsigned-build Artifacts
|
# ios:unsigned-build Artifacts
|
||||||
build/
|
build/
|
||||||
# but keep EAS custom build configs (the generic build/ rule above matches .eas/build/)
|
|
||||||
!.eas/build/
|
|
||||||
!.eas/build/**
|
|
||||||
.claude/
|
.claude/
|
||||||
.agents/skills/**
|
.agents/skills/**
|
||||||
skills-lock.json
|
skills-lock.json
|
||||||
|
|
||||||
# CI-injected Google Play service account key (written at build time)
|
|
||||||
google-service-account.json
|
|
||||||
|
|||||||
3
app.json
@@ -2,7 +2,7 @@
|
|||||||
"expo": {
|
"expo": {
|
||||||
"name": "Streamyfin",
|
"name": "Streamyfin",
|
||||||
"slug": "streamyfin",
|
"slug": "streamyfin",
|
||||||
"version": "0.54.1",
|
"version": "0.54.0",
|
||||||
"orientation": "default",
|
"orientation": "default",
|
||||||
"icon": "./assets/images/icon.png",
|
"icon": "./assets/images/icon.png",
|
||||||
"scheme": "streamyfin",
|
"scheme": "streamyfin",
|
||||||
@@ -36,6 +36,7 @@
|
|||||||
"appleTeamId": "MWD5K362T8"
|
"appleTeamId": "MWD5K362T8"
|
||||||
},
|
},
|
||||||
"android": {
|
"android": {
|
||||||
|
"versionCode": 93,
|
||||||
"adaptiveIcon": {
|
"adaptiveIcon": {
|
||||||
"foregroundImage": "./assets/images/icon-android-plain.png",
|
"foregroundImage": "./assets/images/icon-android-plain.png",
|
||||||
"monochromeImage": "./assets/images/icon-android-themed.png",
|
"monochromeImage": "./assets/images/icon-android-themed.png",
|
||||||
|
|||||||
@@ -1,9 +1,4 @@
|
|||||||
import {
|
import { BottomSheetModal } from "@gorhom/bottom-sheet";
|
||||||
BottomSheetBackdrop,
|
|
||||||
type BottomSheetBackdropProps,
|
|
||||||
BottomSheetModal,
|
|
||||||
BottomSheetView,
|
|
||||||
} from "@gorhom/bottom-sheet";
|
|
||||||
import { useNavigation } from "expo-router";
|
import { useNavigation } from "expo-router";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { useEffect, useMemo, useRef, useState } from "react";
|
import { useEffect, useMemo, useRef, useState } from "react";
|
||||||
@@ -12,7 +7,6 @@ import { Alert, Platform, ScrollView, View } from "react-native";
|
|||||||
import { Pressable } from "react-native-gesture-handler";
|
import { Pressable } from "react-native-gesture-handler";
|
||||||
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { toast } from "sonner-native";
|
import { toast } from "sonner-native";
|
||||||
import { Button } from "@/components/Button";
|
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { TouchableItemRouter } from "@/components/common/TouchableItemRouter";
|
import { TouchableItemRouter } from "@/components/common/TouchableItemRouter";
|
||||||
import ActiveDownloads from "@/components/downloads/ActiveDownloads";
|
import ActiveDownloads from "@/components/downloads/ActiveDownloads";
|
||||||
@@ -107,7 +101,7 @@ export default function DownloadsPage() {
|
|||||||
navigation.setOptions({
|
navigation.setOptions({
|
||||||
headerRight: () => (
|
headerRight: () => (
|
||||||
<Pressable
|
<Pressable
|
||||||
onPress={() => bottomSheetModalRef.current?.present()}
|
onPress={bottomSheetModalRef.current?.present}
|
||||||
className='px-2'
|
className='px-2'
|
||||||
>
|
>
|
||||||
<DownloadSize items={downloadedFiles?.map((f) => f.item) || []} />
|
<DownloadSize items={downloadedFiles?.map((f) => f.item) || []} />
|
||||||
@@ -122,7 +116,7 @@ export default function DownloadsPage() {
|
|||||||
}
|
}
|
||||||
}, [showMigration]);
|
}, [showMigration]);
|
||||||
|
|
||||||
const deleteMovies = () =>
|
const _deleteMovies = () =>
|
||||||
deleteFileByType("Movie")
|
deleteFileByType("Movie")
|
||||||
.then(() =>
|
.then(() =>
|
||||||
toast.success(
|
toast.success(
|
||||||
@@ -133,7 +127,7 @@ export default function DownloadsPage() {
|
|||||||
writeToLog("ERROR", reason);
|
writeToLog("ERROR", reason);
|
||||||
toast.error(t("home.downloads.toasts.failed_to_delete_all_movies"));
|
toast.error(t("home.downloads.toasts.failed_to_delete_all_movies"));
|
||||||
});
|
});
|
||||||
const deleteShows = () =>
|
const _deleteShows = () =>
|
||||||
deleteFileByType("Episode")
|
deleteFileByType("Episode")
|
||||||
.then(() =>
|
.then(() =>
|
||||||
toast.success(
|
toast.success(
|
||||||
@@ -144,7 +138,7 @@ export default function DownloadsPage() {
|
|||||||
writeToLog("ERROR", reason);
|
writeToLog("ERROR", reason);
|
||||||
toast.error(t("home.downloads.toasts.failed_to_delete_all_tvseries"));
|
toast.error(t("home.downloads.toasts.failed_to_delete_all_tvseries"));
|
||||||
});
|
});
|
||||||
const deleteOtherMedia = () =>
|
const _deleteOtherMedia = () =>
|
||||||
Promise.all(
|
Promise.all(
|
||||||
otherMedia
|
otherMedia
|
||||||
.filter((item) => item.item.Type)
|
.filter((item) => item.item.Type)
|
||||||
@@ -168,9 +162,6 @@ export default function DownloadsPage() {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
const deleteAllMedia = async () =>
|
|
||||||
await Promise.all([deleteMovies(), deleteShows(), deleteOtherMedia()]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<OfflineModeProvider isOffline={true}>
|
<OfflineModeProvider isOffline={true}>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
@@ -265,42 +256,6 @@ export default function DownloadsPage() {
|
|||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
<BottomSheetModal
|
|
||||||
ref={bottomSheetModalRef}
|
|
||||||
enableDynamicSizing
|
|
||||||
handleIndicatorStyle={{
|
|
||||||
backgroundColor: "white",
|
|
||||||
}}
|
|
||||||
backgroundStyle={{
|
|
||||||
backgroundColor: "#171717",
|
|
||||||
}}
|
|
||||||
backdropComponent={(props: BottomSheetBackdropProps) => (
|
|
||||||
<BottomSheetBackdrop
|
|
||||||
{...props}
|
|
||||||
disappearsOnIndex={-1}
|
|
||||||
appearsOnIndex={0}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<BottomSheetView>
|
|
||||||
<View className='p-4 space-y-4 mb-4'>
|
|
||||||
<Button color='purple' onPress={deleteMovies}>
|
|
||||||
{t("home.downloads.delete_all_movies_button")}
|
|
||||||
</Button>
|
|
||||||
<Button color='purple' onPress={deleteShows}>
|
|
||||||
{t("home.downloads.delete_all_tvseries_button")}
|
|
||||||
</Button>
|
|
||||||
{otherMedia.length > 0 && (
|
|
||||||
<Button color='purple' onPress={deleteOtherMedia}>
|
|
||||||
{t("home.downloads.delete_all_other_media_button")}
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
<Button color='red' onPress={deleteAllMedia}>
|
|
||||||
{t("home.downloads.delete_all_button")}
|
|
||||||
</Button>
|
|
||||||
</View>
|
|
||||||
</BottomSheetView>
|
|
||||||
</BottomSheetModal>
|
|
||||||
</OfflineModeProvider>
|
</OfflineModeProvider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,19 +59,17 @@ function SettingsMobile() {
|
|||||||
|
|
||||||
<QuickConnect className='mb-4' />
|
<QuickConnect className='mb-4' />
|
||||||
|
|
||||||
{Platform.OS !== "ios" && (
|
<View className='mb-4'>
|
||||||
<View className='mb-4'>
|
<ListGroup title={t("pairing.pair_with_phone_title")}>
|
||||||
<ListGroup title={t("pairing.pair_with_phone_title")}>
|
<ListItem
|
||||||
<ListItem
|
onPress={() =>
|
||||||
onPress={() =>
|
router.push("/(auth)/(tabs)/(home)/companion-login")
|
||||||
router.push("/(auth)/(tabs)/(home)/companion-login")
|
}
|
||||||
}
|
title={t("pairing.pair_with_phone")}
|
||||||
title={t("pairing.pair_with_phone")}
|
textColor='blue'
|
||||||
textColor='blue'
|
/>
|
||||||
/>
|
</ListGroup>
|
||||||
</ListGroup>
|
</View>
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<View className='mb-4'>
|
<View className='mb-4'>
|
||||||
<AppLanguageSelector />
|
<AppLanguageSelector />
|
||||||
|
|||||||
@@ -114,7 +114,7 @@ export default function StreamystatsPage() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleRefreshFromServer = useCallback(async () => {
|
const handleRefreshFromServer = useCallback(async () => {
|
||||||
const newPluginSettings = await refreshStreamyfinPluginSettings();
|
const newPluginSettings = await refreshStreamyfinPluginSettings(true);
|
||||||
// Update local state with new values
|
// Update local state with new values
|
||||||
const newUrl = newPluginSettings?.streamyStatsServerUrl?.value || "";
|
const newUrl = newPluginSettings?.streamyStatsServerUrl?.value || "";
|
||||||
setUrl(newUrl);
|
setUrl(newUrl);
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import {
|
|||||||
BottomSheetTextInput,
|
BottomSheetTextInput,
|
||||||
BottomSheetView,
|
BottomSheetView,
|
||||||
} from "@gorhom/bottom-sheet";
|
} from "@gorhom/bottom-sheet";
|
||||||
import type { BottomSheetModalMethods } from "@gorhom/bottom-sheet/lib/typescript/types";
|
|
||||||
import { useQuery } from "@tanstack/react-query";
|
import { useQuery } from "@tanstack/react-query";
|
||||||
import { Image } from "expo-image";
|
import { Image } from "expo-image";
|
||||||
import { useLocalSearchParams, useNavigation } from "expo-router";
|
import { useLocalSearchParams, useNavigation } from "expo-router";
|
||||||
@@ -77,7 +76,7 @@ const MobilePage: React.FC = () => {
|
|||||||
const [issueMessage, setIssueMessage] = useState<string>();
|
const [issueMessage, setIssueMessage] = useState<string>();
|
||||||
const [requestBody, _setRequestBody] = useState<MediaRequestBody>();
|
const [requestBody, _setRequestBody] = useState<MediaRequestBody>();
|
||||||
const [issueTypeDropdownOpen, setIssueTypeDropdownOpen] = useState(false);
|
const [issueTypeDropdownOpen, setIssueTypeDropdownOpen] = useState(false);
|
||||||
const advancedReqModalRef = useRef<BottomSheetModalMethods>(null);
|
const advancedReqModalRef = useRef<BottomSheetModal>(null);
|
||||||
const bottomSheetModalRef = useRef<BottomSheetModal>(null);
|
const bottomSheetModalRef = useRef<BottomSheetModal>(null);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
|
|||||||
@@ -166,7 +166,7 @@ export default function IndexLayout() {
|
|||||||
open={dropdownOpen}
|
open={dropdownOpen}
|
||||||
onOpenChange={setDropdownOpen}
|
onOpenChange={setDropdownOpen}
|
||||||
trigger={
|
trigger={
|
||||||
<View>
|
<View className='pl-1.5'>
|
||||||
<Ionicons
|
<Ionicons
|
||||||
name='ellipsis-horizontal-outline'
|
name='ellipsis-horizontal-outline'
|
||||||
size={24}
|
size={24}
|
||||||
|
|||||||
@@ -40,8 +40,6 @@ const Layout = () => {
|
|||||||
keyboardDismissMode='none'
|
keyboardDismissMode='none'
|
||||||
screenOptions={{
|
screenOptions={{
|
||||||
tabBarBounces: true,
|
tabBarBounces: true,
|
||||||
tabBarActiveTintColor: "#FFFFFF",
|
|
||||||
tabBarInactiveTintColor: "#9CA3AF",
|
|
||||||
tabBarLabelStyle: {
|
tabBarLabelStyle: {
|
||||||
fontSize: TAB_LABEL_FONT_SIZE,
|
fontSize: TAB_LABEL_FONT_SIZE,
|
||||||
fontWeight: "600",
|
fontWeight: "600",
|
||||||
|
|||||||
@@ -102,8 +102,8 @@ export default function TabLayout() {
|
|||||||
!settings?.streamyStatsServerUrl || settings?.hideWatchlistsTab,
|
!settings?.streamyStatsServerUrl || settings?.hideWatchlistsTab,
|
||||||
tabBarIcon:
|
tabBarIcon:
|
||||||
Platform.OS === "android"
|
Platform.OS === "android"
|
||||||
? (_e) => require("@/assets/icons/list.star.png")
|
? (_e) => require("@/assets/icons/list.png")
|
||||||
: (_e) => ({ sfSymbol: "list.star" }),
|
: (_e) => ({ sfSymbol: "list.bullet.rectangle" }),
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<NativeTabs.Screen
|
<NativeTabs.Screen
|
||||||
@@ -112,7 +112,7 @@ export default function TabLayout() {
|
|||||||
title: t("tabs.library"),
|
title: t("tabs.library"),
|
||||||
tabBarIcon:
|
tabBarIcon:
|
||||||
Platform.OS === "android"
|
Platform.OS === "android"
|
||||||
? (_e) => require("@/assets/icons/rectangle.stack.fill.png")
|
? (_e) => require("@/assets/icons/server.rack.png")
|
||||||
: (_e) => ({ sfSymbol: "rectangle.stack.fill" }),
|
: (_e) => ({ sfSymbol: "rectangle.stack.fill" }),
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
@@ -123,8 +123,8 @@ export default function TabLayout() {
|
|||||||
tabBarItemHidden: !settings?.showCustomMenuLinks,
|
tabBarItemHidden: !settings?.showCustomMenuLinks,
|
||||||
tabBarIcon:
|
tabBarIcon:
|
||||||
Platform.OS === "android"
|
Platform.OS === "android"
|
||||||
? (_e) => require("@/assets/icons/link.png")
|
? (_e) => require("@/assets/icons/list.png")
|
||||||
: (_e) => ({ sfSymbol: "link" }),
|
: (_e) => ({ sfSymbol: "list.dash.fill" }),
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<NativeTabs.Screen
|
<NativeTabs.Screen
|
||||||
@@ -134,7 +134,7 @@ export default function TabLayout() {
|
|||||||
tabBarItemHidden: !Platform.isTV,
|
tabBarItemHidden: !Platform.isTV,
|
||||||
tabBarIcon:
|
tabBarIcon:
|
||||||
Platform.OS === "android"
|
Platform.OS === "android"
|
||||||
? (_e) => require("@/assets/icons/gearshape.fill.png")
|
? (_e) => require("@/assets/icons/gear.png") //Should maybe use other libraries to have it uniform
|
||||||
: (_e) => ({ sfSymbol: "gearshape.fill" }),
|
: (_e) => ({ sfSymbol: "gearshape.fill" }),
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -274,11 +274,6 @@ export default function DirectPlayerPage() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
if (itemId) {
|
if (itemId) {
|
||||||
setItem(null);
|
|
||||||
setDownloadedItem(null);
|
|
||||||
// Clear the previous episode's stream so the loader gate stays closed
|
|
||||||
// until the new item's stream resolves (avoids a stale MPV source frame).
|
|
||||||
setStream(null);
|
|
||||||
fetchItemData();
|
fetchItemData();
|
||||||
}
|
}
|
||||||
}, [itemId, offline, api, user?.Id]);
|
}, [itemId, offline, api, user?.Id]);
|
||||||
@@ -321,12 +316,6 @@ export default function DirectPlayerPage() {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure item matches the current itemId to avoid race conditions
|
|
||||||
if (item.Id !== itemId) {
|
|
||||||
setStreamStatus({ isLoading: false, isError: false });
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
let result: Stream | null = null;
|
let result: Stream | null = null;
|
||||||
if (offline && downloadedItem?.mediaSource) {
|
if (offline && downloadedItem?.mediaSource) {
|
||||||
const url = downloadedItem.videoFilePath;
|
const url = downloadedItem.videoFilePath;
|
||||||
@@ -399,7 +388,6 @@ export default function DirectPlayerPage() {
|
|||||||
item,
|
item,
|
||||||
user?.Id,
|
user?.Id,
|
||||||
downloadedItem,
|
downloadedItem,
|
||||||
offline,
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -439,15 +427,21 @@ export default function DirectPlayerPage() {
|
|||||||
if (!item?.Id || !stream?.sessionId || offline || !api) return;
|
if (!item?.Id || !stream?.sessionId || offline || !api) return;
|
||||||
|
|
||||||
const currentTimeInTicks = msToTicks(progress.get());
|
const currentTimeInTicks = msToTicks(progress.get());
|
||||||
await getPlaystateApi(api).reportPlaybackStopped({
|
await getPlaystateApi(api).onPlaybackStopped({
|
||||||
playbackStopInfo: {
|
itemId: item.Id,
|
||||||
ItemId: item.Id,
|
mediaSourceId: mediaSourceId,
|
||||||
MediaSourceId: mediaSourceId,
|
positionTicks: currentTimeInTicks,
|
||||||
PositionTicks: currentTimeInTicks,
|
playSessionId: stream.sessionId,
|
||||||
PlaySessionId: stream.sessionId,
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
}, [api, item, mediaSourceId, stream, progress, offline]);
|
}, [
|
||||||
|
api,
|
||||||
|
item,
|
||||||
|
mediaSourceId,
|
||||||
|
stream,
|
||||||
|
progress,
|
||||||
|
offline,
|
||||||
|
revalidateProgressCache,
|
||||||
|
]);
|
||||||
|
|
||||||
const stop = useCallback(() => {
|
const stop = useCallback(() => {
|
||||||
// Update URL with final playback position before stopping
|
// Update URL with final playback position before stopping
|
||||||
@@ -465,10 +459,9 @@ export default function DirectPlayerPage() {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const beforeRemoveListener = navigation.addListener("beforeRemove", stop);
|
const beforeRemoveListener = navigation.addListener("beforeRemove", stop);
|
||||||
return () => {
|
return () => {
|
||||||
reportPlaybackStopped();
|
|
||||||
beforeRemoveListener();
|
beforeRemoveListener();
|
||||||
};
|
};
|
||||||
}, [navigation, stop, reportPlaybackStopped]);
|
}, [navigation, stop]);
|
||||||
|
|
||||||
const currentPlayStateInfo = useCallback(():
|
const currentPlayStateInfo = useCallback(():
|
||||||
| PlaybackProgressInfo
|
| PlaybackProgressInfo
|
||||||
|
|||||||
BIN
assets/icons/gear.png
Normal file
|
After Width: | Height: | Size: 26 KiB |
|
Before Width: | Height: | Size: 11 KiB |
|
Before Width: | Height: | Size: 7.9 KiB After Width: | Height: | Size: 15 KiB |
BIN
assets/icons/heart.png
Normal file
|
After Width: | Height: | Size: 22 KiB |
|
Before Width: | Height: | Size: 8.5 KiB After Width: | Height: | Size: 112 KiB |
118
assets/icons/jellyseerr-logo.svg
Normal file
|
After Width: | Height: | Size: 22 KiB |
|
Before Width: | Height: | Size: 13 KiB |
BIN
assets/icons/list.png
Normal file
|
After Width: | Height: | Size: 2.3 KiB |
|
Before Width: | Height: | Size: 7.0 KiB |
|
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 20 KiB |
|
Before Width: | Height: | Size: 3.6 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" width="96" height="96" fill="none" viewBox="0 0 96 96"><path fill="url(#paint0_linear)" fill-rule="evenodd" d="M48 96C74.5097 96 96 74.5097 96 48C96 21.4903 74.5097 0 48 0C21.4903 0 0 21.4903 0 48C0 74.5097 21.4903 96 48 96ZM80.0001 52C80.0001 67.464 67.4641 80 52.0001 80C36.5361 80 24.0001 67.464 24.0001 52C24.0001 49.1303 24.4318 46.3615 25.2338 43.7548C27.4288 48.6165 32.3194 52 38.0001 52C45.7321 52 52.0001 45.732 52.0001 38C52.0001 32.3192 48.6166 27.4287 43.755 25.2337C46.3616 24.4317 49.1304 24 52.0001 24C67.4641 24 80.0001 36.536 80.0001 52Z" clip-rule="evenodd"/><path fill="#131928" fill-rule="evenodd" d="M80.0002 52C80.0002 67.464 67.4642 80 52.0002 80C36.864 80 24.5329 67.9897 24.017 52.9791C24.0057 53.318 24 53.6583 24 54C24 70.5685 37.4315 84 54 84C70.5685 84 84 70.5685 84 54C84 37.4315 70.5685 24 54 24C53.6597 24 53.3207 24.0057 52.9831 24.0169C67.9919 24.5347 80.0002 36.865 80.0002 52Z" clip-rule="evenodd" opacity=".2"/><path fill="url(#paint1_linear)" fill-rule="evenodd" d="M48 12C28.1177 12 12 28.1177 12 48C12 50.2091 10.2091 52 8 52C5.79086 52 4 50.2091 4 48C4 23.6995 23.6995 4 48 4C50.2091 4 52 5.79086 52 8C52 10.2091 50.2091 12 48 12Z" clip-rule="evenodd"/><defs><linearGradient id="paint0_linear" x1="48" x2="117.5" y1="0" y2="69.5" gradientUnits="userSpaceOnUse"><stop stop-color="#C395FC"/><stop offset="1" stop-color="#4F65F5"/></linearGradient><linearGradient id="paint1_linear" x1="28" x2="28" y1="8" y2="48" gradientUnits="userSpaceOnUse"><stop stop-color="#fff" stop-opacity=".4"/><stop offset="1" stop-color="#fff" stop-opacity="0"/></linearGradient></defs></svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.6 KiB |
BIN
assets/icons/server.rack.png
Normal file
|
After Width: | Height: | Size: 14 KiB |
65
assets/images/not-rotten-tomatoes.svg
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
<svg
|
||||||
|
type="certified"
|
||||||
|
viewBox="0 0 80 80"
|
||||||
|
preserveAspectRatio="xMidYMid"
|
||||||
|
version="1.1"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||||
|
>
|
||||||
|
<g transform="translate(2.29, 0)">
|
||||||
|
<path
|
||||||
|
d="M42.1942857,18.8022857 C44.3794286,18.608 49.1565714,18.7177143 51.4902857,21.0057143 C51.6297143,21.1451429 51.5085714,21.4605714 51.3097143,21.408 C47.8902857,20.4868571 42.5577143,25.0217143 39.1017143,22.0891429 C39.008,22.9485714 38.2331429,27.0857143 32.3314286,26.4731429 C32.192,26.4594286 32.1371429,26.304 32.24,26.2171429 C33.1542857,25.44 34.2765714,23.2891429 33.3142857,21.9154286 C30.3108571,23.9085714 28.7565714,23.9954286 23.2182857,21.5954286 C23.0377143,21.5177143 23.1451429,21.2228571 23.3577143,21.1748571 C24.5074286,20.9165714 27.2434286,19.9222857 29.696,19.4582857 C30.1645714,19.3691429 30.624,19.3165714 31.0674286,19.312 C28.528,18.7062857 27.4217143,18.1805714 25.7485714,18.1874286 C25.5657143,18.1897143 25.4742857,17.9611429 25.6068571,17.8354286 C28.224,15.3188571 32.9691429,15.1885714 35.2548571,17.0628571 L33.2068571,12.7862857 L35.696,12.4114286 C35.696,12.4114286 36.3451429,14.6925714 36.9257143,16.7428571 C39.5177143,13.904 43.5268571,14.192 44.8777143,16.672 C44.9577143,16.8182857 44.8251429,16.992 44.6605714,16.9622857 C43.3005714,16.7314286 42.3702857,17.8628571 42.1737143,18.7977143 L42.1942857,18.8022857"
|
||||||
|
id="Fill-2"
|
||||||
|
fill="#00912D"
|
||||||
|
></path>
|
||||||
|
<mask id="mask-2" fill="white">
|
||||||
|
<polygon
|
||||||
|
points="0.137142857 0.921142857 75.0534777 0.921142857 75.0534777 79.8628571 0.137142857 79.8628571"
|
||||||
|
></polygon>
|
||||||
|
</mask>
|
||||||
|
<path
|
||||||
|
d="M13.0491429,59.1817143 C9.90628571,55.3554286 7.86971429,50.576 7.51771429,44.9622857 C6.912,35.2342857 10.2354286,26.0845714 23.1794286,21.4834286 C23.1908571,21.5245714 23.1725714,21.5748571 23.2182857,21.5954286 C23.0377143,21.5177143 23.1451429,21.2228571 23.3577143,21.1748571 C24.5074286,20.9165714 27.2434286,19.92 29.696,19.4582857 C30.1645714,19.3691429 30.624,19.3165714 31.0674286,19.3097143 C28.528,18.7062857 27.4217143,18.1805714 25.7485714,18.1874286 C25.5657143,18.1897143 25.4742857,17.9611429 25.6068571,17.8331429 C28.224,15.3165714 32.9691429,15.1885714 35.2548571,17.0628571 L33.2068571,12.784 L35.696,12.4114286 C35.696,12.4114286 36.3451429,14.6902857 36.9257143,16.7428571 C39.5177143,13.904 43.5268571,14.192 44.8777143,16.672 C44.9577143,16.8182857 44.8251429,16.992 44.6605714,16.9622857 C43.3005714,16.7314286 42.3702857,17.8628571 42.1737143,18.7977143 L42.1942857,18.8022857 C44.3794286,18.608 49.1565714,18.7177143 51.4902857,21.0057143 C51.328,20.8502857 51.1337143,20.7245714 50.9508571,20.5874286 C60.2765714,23.504 66.7474286,30.1531429 67.44,41.2251429 C67.8811429,48.2948571 65.5702857,54.3885714 61.568,59.1154286 C62.784,59.2891429 63.9931429,59.4925714 65.2045714,59.6937143 C70.304,53.4537143 73.2502857,45.5428571 73.2502857,37.056 C73.2502857,17.7165714 57.5337143,2.56685714 37.472,2.56685714 C17.4102857,2.56685714 1.69371429,17.7165714 1.69371429,37.056 C1.69371429,45.5565714 4.64,53.472 9.744,59.7097143 C10.8434286,59.5268571 11.9451429,59.3462857 13.0491429,59.1817143"
|
||||||
|
fill="#FFD700"
|
||||||
|
mask="url(#mask-2)"
|
||||||
|
></path>
|
||||||
|
<path
|
||||||
|
d="M9.744,59.7097143 C4.64,53.472 1.69371429,45.5565714 1.69371429,37.056 C1.69371429,17.7165714 17.4102857,2.56685714 37.472,2.56685714 C57.5337143,2.56685714 73.2502857,17.7165714 73.2502857,37.056 C73.2502857,45.5428571 70.304,53.4537143 65.2045714,59.6937143 C65.8125714,59.7942857 66.4205714,59.8742857 67.0285714,59.984 C71.9497143,53.6457143 74.8937143,45.6982857 74.8937143,37.056 C74.8937143,16.3862857 58.1394286,0.921142857 37.472,0.921142857 C16.8022857,0.921142857 0.048,16.3862857 0.048,37.056 C0.048,45.7074286 2.99885714,53.6594286 7.92914286,59.9977143 C8.53257143,59.8902857 9.13828571,59.8102857 9.744,59.7097143"
|
||||||
|
fill="#FA6E0F"
|
||||||
|
mask="url(#mask-2)"
|
||||||
|
></path>
|
||||||
|
<path
|
||||||
|
d="M58.2857143,74.9394286 C62.3748571,75.1954286 65.7874286,77.2137143 67.8468571,79.9474286 C67.9131429,80.0182857 68.0114286,80.016 68.0411429,79.9382857 C68.7451429,77.0971429 68.9394286,74.0662857 68.5851429,71.0125714 C68.5874286,70.9805714 68.6125714,70.9577143 68.6537143,70.9485714 C70.576,70.3428571 72.7017143,70.0137143 74.9645714,70.0457143 C75.0857143,70.0594286 75.0834286,69.9405714 74.9554286,69.8194286 C72.5577143,67.4994286 69.6297143,65.6914286 66.416,64.5417143 C65.3051429,67.68 64.2217143,70.816 63.1565714,73.9634286 C63.136,74.0228571 63.0514286,74.0594286 62.9645714,74.0434286 L58.2857143,74.9394286"
|
||||||
|
fill="#0AC855"
|
||||||
|
mask="url(#mask-2)"
|
||||||
|
></path>
|
||||||
|
<path
|
||||||
|
d="M62.9645714,74.0434286 L58.2857143,74.9394286 C58.2857143,74.9394286 58.3451429,74.512 58.528,73.3325714 C60.9417143,73.6754286 62.9645714,74.0434286 62.9645714,74.0434286"
|
||||||
|
fill="#0B4902"
|
||||||
|
></path>
|
||||||
|
<g transform="translate(0, 20.57)">
|
||||||
|
<mask id="mask-4" fill="white">
|
||||||
|
<polygon
|
||||||
|
points="0.137142857 0.016 67.4935952 0.016 67.4935952 59.2914286 0.137142857 59.2914286"
|
||||||
|
></polygon>
|
||||||
|
</mask>
|
||||||
|
<path
|
||||||
|
d="M13.0765714,38.6057143 C29.1177143,36.2605714 45.5222857,36.2354286 61.568,38.544 C65.5702857,33.8171429 67.8811429,27.7234286 67.44,20.6537143 C66.7474286,9.58171429 60.2765714,2.93257143 50.9508571,0.016 C51.1337143,0.153142857 51.328,0.278857143 51.4902857,0.434285714 C51.6297143,0.573714286 51.5085714,0.889142857 51.3097143,0.836571429 C47.8902857,-0.0845714286 42.5577143,4.45028571 39.1017143,1.51771429 C39.008,2.37485714 38.2331429,6.51428571 32.3314286,5.90171429 C32.192,5.888 32.1371429,5.73257143 32.24,5.64571429 C33.1542857,4.86857143 34.2765714,2.71542857 33.3142857,1.344 C30.3108571,3.33714286 28.7565714,3.424 23.2182857,1.024 C23.1725714,1.00342857 23.1908571,0.953142857 23.1794286,0.912 C10.2354286,5.51314286 6.912,14.6628571 7.51771429,24.3908571 C7.86971429,30.0091429 9.93142857,34.7748571 13.0765714,38.6057143"
|
||||||
|
fill="#FA3200"
|
||||||
|
mask="url(#mask-4)"
|
||||||
|
></path>
|
||||||
|
<path
|
||||||
|
d="M12.0868571,53.472 C12,53.488 11.9154286,53.4514286 11.8948571,53.392 C10.8274286,50.2445714 9.73485714,47.0971429 8.62171429,43.9611429 C5.41028571,45.1108571 2.49371429,46.9302857 0.0982857143,49.248 C-0.0297142857,49.3691429 -0.032,49.488 0.0891428571,49.4742857 C2.352,49.4422857 4.47771429,49.7714286 6.4,50.3771429 C6.44114286,50.3862857 6.46628571,50.4091429 6.46857143,50.4411429 C6.11428571,53.4948571 6.30857143,56.5257143 7.01257143,59.3668571 C7.04228571,59.4445714 7.14057143,59.4468571 7.20685714,59.376 C9.26628571,56.6422857 12.6742857,54.624 16.7657143,54.368 L12.0868571,53.472"
|
||||||
|
fill="#0AC855"
|
||||||
|
mask="url(#mask-4)"
|
||||||
|
></path>
|
||||||
|
</g>
|
||||||
|
<path
|
||||||
|
d="M62.9645714,74.0434286 C46.192,71.104 28.8571429,71.104 12.0868571,74.0434286 C12,74.0594286 11.9154286,74.0228571 11.8948571,73.9634286 C10.3428571,69.3851429 8.74285714,64.8182857 7.09257143,60.2628571 C7.06971429,60.1988571 7.14057143,60.1257143 7.248,60.1074286 C27.1885714,56.464 47.8605714,56.464 67.8034286,60.1074286 C67.9108571,60.1257143 67.9817143,60.1988571 67.9565714,60.2628571 C66.3085714,64.8182857 64.7085714,69.3851429 63.1565714,73.9634286 C63.136,74.0228571 63.0514286,74.0594286 62.9645714,74.0434286"
|
||||||
|
fill="#00912D"
|
||||||
|
></path>
|
||||||
|
<path
|
||||||
|
d="M12.0868571,74.0434286 L16.7657143,74.9394286 C16.7657143,74.9394286 16.704,74.512 16.5211429,73.3325714 C14.1074286,73.6754286 12.0868571,74.0434286 12.0868571,74.0434286"
|
||||||
|
fill="#0B4902"
|
||||||
|
></path>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 7.4 KiB |
BIN
assets/images/rotten-tomatoes.png
Normal file
|
After Width: | Height: | Size: 50 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg viewBox="0 0 560 560" xmlns="http://www.w3.org/2000/svg"><path fill="#fff" d="M370.57 474.214l23.466-237.956c14.93-4.796 29.498-11.15 40.23-20.262L404.16 446.278c-6.748 10.248-19.863 20.86-33.59 27.936zm-78.197 21.631l2.947-244.528c20.894-.599 47.933-3.43 70.97-8.346l-19.07 241.17c-22.724 7.518-35.934 9.848-54.847 11.704zm-99.694-252.874c23.038 4.916 50.077 7.747 70.971 8.346l2.948 244.528c-18.914-1.856-32.123-4.186-54.847-11.705l-19.072-241.17zm-67.974-26.975c10.732 9.112 25.3 15.466 40.23 20.262l23.464 237.956c-13.726-7.075-26.84-17.688-33.59-27.936l-30.104-230.282z"/><path fill="gold" d="M118.905 157.445c1.357 28.827 72.771 51.677 160.578 51.176 76.687-.438 140.659-18.546 156.329-42.336a22.976 22.976 0 00-14.058-7.426c.06-.7.098-1.406.095-2.122-.065-11.4-8.429-20.788-19.327-22.54.287-1.474.438-2.999.43-4.559-.072-12.696-10.426-22.928-23.124-22.856-.287.001-.568.036-.853.049a22.911 22.911 0 001.254-7.56c-.074-12.697-10.425-22.93-23.123-22.858a22.914 22.914 0 00-8.247 1.6c-3.632-6.835-10.606-11.6-18.737-12.149-1.416-11.4-11.157-20.195-22.93-20.129-7.41.042-13.963 3.6-18.136 9.065-4.233-4.605-10.3-7.494-17.047-7.456-12.698.072-22.932 10.424-22.86 23.118a22.983 22.983 0 001.115 6.946 22.918 22.918 0 00-13.07 7.459c-2.644-9.847-11.637-17.084-22.314-17.024-9.975.057-18.406 6.47-21.537 15.366-8.474 3.426-14.439 11.738-14.383 21.433.012 2.154.342 4.227.907 6.202a22.876 22.876 0 00-9.328-1.932c-10.012.058-18.47 6.516-21.574 15.465a22.83 22.83 0 00-9.788-2.149c-12.698.072-22.934 10.422-22.86 23.118a22.833 22.833 0 003.159 11.463c-.202.203-.379.426-.571.636"/><path fill="#FA320A" d="M404.161 446.278c-6.749 10.248-19.864 20.86-33.59 27.936l23.465-237.956c14.93-4.796 29.498-11.15 40.23-20.262L404.16 446.278zM347.22 484.14c-22.723 7.519-35.934 9.85-54.847 11.705l2.947-244.528c20.894-.599 47.933-3.43 70.973-8.346L347.22 484.14zm-135.47 0l-19.07-241.17c23.037 4.917 50.076 7.748 70.97 8.347l2.948 244.528c-18.914-1.856-32.123-4.186-54.847-11.705zm-56.94-37.862l-30.105-230.282c10.732 9.112 25.3 15.466 40.23 20.262l23.464 237.956c-13.726-7.075-26.84-17.688-33.588-27.936zm247.668-321.143c.298 1.453.465 2.955.473 4.498a23.018 23.018 0 01-.43 4.56c10.9 1.749 19.263 11.137 19.328 22.54a23.59 23.59 0 01-.095 2.12 22.976 22.976 0 0114.058 7.425c-15.669 23.792-79.642 41.9-156.327 42.34-87.807.502-159.221-22.346-160.58-51.175.192-.208.37-.433.57-.634-1.355-2.311-2.29-4.887-2.773-7.62-8.408 7.979-13.495 14.412-12.6 23.78.085 1.251 37.196 266.911 37.196 266.911 4.282 42.075 65.391 75.703 138.187 76.12 72.796-.417 133.907-34.045 138.187-76.12 0 0 37.11-265.66 37.197-266.912 1.777-18.736-20.15-35.745-52.39-47.833z"/></svg>
|
|
||||||
|
Before Width: | Height: | Size: 2.6 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg viewBox="0 0 560 560" xmlns="http://www.w3.org/2000/svg"><g transform="translate(33 140)"><path d="m43.802 267.32l237.94 23.482c4.7937 14.937 11.149 29.517 20.259 40.256l-230.27-30.125c-10.248-6.7519-20.861-19.877-27.936-33.612zm222.88-75.298c0.60053 20.906 3.4316 47.964 8.3462 71.017l-241.15-19.083c-7.518-22.739-9.8466-35.959-11.704-54.885l244.51 2.951zm8.3462-102.71c-4.9146 23.053-7.7456 50.111-8.3462 71.017l-244.51 2.951c1.8576-18.926 4.1862-32.146 11.704-54.885l241.15-19.083zm26.973-68.019c-9.1095 10.74-15.465 25.318-20.259 40.257l-237.94 23.48c7.0751-13.735 17.689-26.859 27.936-33.612l230.27-30.125z" fill="#fff"/><path d="m303.57 264.67c3.155-7.8209 14.337-12.586 22.367-12.028 8.5825 0.59581 17.699 9.6258 19.292 18.507 0.29589-0.32244 0.60578-0.62735 0.92093-0.92701 2.7558-2.6356 6.2084-4.3845 9.9867-4.8664-0.57777-2.562-0.71609-5.3045-0.3204-8.1188 1.3954-9.901 9.3336-17.326 18.422-17.252 5.8652 0.047314 11.011 3.0509 14.364 7.6649 0.29939-0.37501 0.6303-0.71848 0.95245-1.069 3.8343-19.991 6.2644-42.578 6.8177-66.547 1.8996-82.42-18.993-149.75-46.663-150.39-27.672-0.63962-51.644 65.656-53.544 148.08 0 0-1.4654 30.062 7.4042 86.951" fill="#00641E"/><path d="m490.91 354.8c1.6353-2.732 2.5492-6.0072 2.4862-9.4874 0.5305-11.245-7.1819-21.439-17.913-20.31 0.3099-1.2862 0.51299-2.6233 0.59178-4.0024 0.64255-11.264-7.1188-20.972-17.337-21.682-0.2241-0.014019-0.44471-0.021029-0.66706-0.028039 1.0627-2.795 1.5897-5.916 1.4024-9.2228-0.52875-9.3717-6.9858-17.268-15.386-18.822-3.0342-0.56076-5.9773-0.28213-8.6718 0.65188-2.5457-6.277-7.8909-10.933-14.393-11.916-0.51474-10.192-7.8699-18.58-17.34-19.238-5.9545-0.41356-11.426 2.3237-15.085 6.9061-3.3528-4.614-8.4985-7.6158-14.364-7.6632-9.0885-0.075352-17.027 7.3495-18.422 17.252-0.39569 2.8126-0.25737 5.555 0.3204 8.1188-3.7783 0.48015-7.2309 2.2308-9.9867 4.8646-0.31515 0.29966-0.62504 0.60457-0.92093 0.92701-1.5932-8.8811-10.71-17.909-19.292-18.507-8.0293-0.55726-19.357 4.3249-22.367 12.028 1.3201 13.434 9.71 50.053 40.055 82.903l0.26963 0.019276c2.9256 2.6496 6.7459 4.1093 10.818 3.7466 2.5247-0.22606 4.8498-1.1303 6.8527-2.5252l0.48848 0.033295c2.67 1.8558 5.8793 2.8266 9.2706 2.5252 1.2956-0.11566 2.5317-0.42758 3.7065-0.87269 2.9064 6.0142 9.3879 9.901 16.622 9.2631 5.6026-0.49417 10.365-3.5959 13.164-7.9611l0.90692 0.063086c2.7961 2.774 6.513 4.3897 10.522 4.2688 3.3143 5.0188 9.4019 8.1065 16.12 7.516 2.5299-0.22255 4.8918-0.95505 6.998-2.0643 3.5139 4.3266 9.2811 6.8991 15.609 6.3419 6.2557-0.5485 11.54-4.02 14.414-8.8197 2.8241 2.2693 6.3625 3.4872 10.12 3.1525 3.6452-0.32594 6.8842-2.0485 9.3179-4.6543l0.40619 0.028038c0.55326-0.80259 1.026-1.6245 1.4654-2.4533 0.010505-0.015772 0.019259-0.033296 0.028014-0.049067 0.059527-0.1104 0.13306-0.21905 0.18909-0.3312" fill="#FFD700"/><path d="m281.75 61.547l-237.94 23.48c7.0751-13.735 17.689-26.859 27.936-33.612l230.27-30.125c-9.1095 10.74-15.465 25.318-20.259 40.257zm20.259 269.51l-230.27-30.125c-10.248-6.7519-20.861-19.877-27.936-33.611l237.94 23.48c4.7937 14.937 11.149 29.517 20.259 40.256zm-268.13-87.102c-7.518-22.739-9.8466-35.957-11.704-54.885l244.51 2.951c0.60053 20.906 3.4316 47.964 8.3462 71.017l-241.15-19.083zm0-135.56l241.15-19.083c-4.9146 23.053-7.7456 50.111-8.3462 71.019l-244.51 2.9493c1.8576-18.926 4.1862-32.146 11.704-54.885zm344.72-82.679c-15.255-17.778-26.206-26.124-35.587-25.04-1.7491 0.22255-266.89 37.222-266.89 37.222-42.074 4.2828-75.7 65.432-76.117 138.28 0.4167 72.843 34.043 133.99 76.117 138.28 0 0 265.64 37.135 266.89 37.221 2.2183-0.014019 4.4086-0.31192 6.5673-0.86568-2.101-0.6256-4.0391-1.7208-5.6867-3.2139l-0.26963-0.019276c-30.345-32.848-38.735-69.47-40.055-82.903 0.003501-0.010514 0.010505-0.019276 0.014006-0.02979-0.003501 0.010514-0.010505 0.019276-0.014006 0.02979-8.8697-56.889-7.4042-86.951-7.4042-86.951 1.8996-82.42 25.872-148.72 53.544-148.08 27.67 0.63962 48.562 67.973 46.663 150.39-0.55326 23.969-2.9834 46.556-6.8177 66.547 3.9393-4.3512 9.2233-6.1842 14.133-5.8372 0.90342 0.064838 1.7823 0.2173 2.6437 0.41531 16.804-92.03-0.81238-181.89-27.729-215.44z" fill="#04A53C"/></g></svg>
|
|
||||||
|
Before Width: | Height: | Size: 4.0 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg viewBox="0 0 560 560" xmlns="http://www.w3.org/2000/svg"><path d="m478.29 296.98c-3.99-63.966-36.52-111.82-85.468-138.58 0.278 1.56-1.109 3.508-2.688 2.818-32.016-14.006-86.328 31.32-124.28 7.584 0.285 8.519-1.378 50.072-59.914 52.483-1.382 0.056-2.142-1.355-1.268-2.354 7.828-8.929 15.732-31.535 4.367-43.586-24.338 21.81-38.472 30.017-85.138 19.186-29.878 31.241-46.809 74-43.485 127.26 6.78 108.74 108.63 170.89 211.19 164.49 102.56-6.395 193.47-80.572 186.68-189.31" fill="#FA320A"/><path d="M291.375 132.293c21.075-5.023 81.693-.49 101.114 25.274 1.166 1.545-.475 4.468-2.355 3.648-32.016-14.006-86.328 31.32-124.282 7.584.285 8.519-1.378 50.072-59.914 52.483-1.382.056-2.142-1.355-1.268-2.354 7.828-8.929 15.73-31.535 4.367-43.586-26.512 23.758-40.884 31.392-98.426 15.838-1.883-.508-1.241-3.535.762-4.298 10.876-4.157 35.515-22.361 58.824-30.385 4.438-1.526 8.862-2.71 13.18-3.4-25.665-2.293-37.235-5.862-53.559-3.4-1.789.27-3.004-1.813-1.895-3.241 21.995-28.332 62.513-36.888 87.512-21.837-15.41-19.094-27.48-34.321-27.48-34.321l28.601-16.246s11.817 26.4 20.414 45.614c21.275-31.435 60.86-34.336 77.585-12.033.992 1.326-.045 3.21-1.702 3.171-13.612-.331-21.107 12.05-21.675 21.466l.197.023" fill="#00912D"/></svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.2 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg viewBox="0 0 560 560" xmlns="http://www.w3.org/2000/svg"><path d="M445.185 444.684c-79.369 4.167-95.587-86.652-126.726-86.006-13.268.279-23.726 14.151-19.133 30.32 2.525 8.888 9.53 21.923 13.944 30.011 15.57 28.544-7.447 60.845-34.383 63.577-44.76 4.54-63.433-21.426-62.278-48.007 1.3-29.84 26.6-60.331.65-73.305-27.194-13.597-49.301 39.572-75.325 51.439-23.553 10.741-56.248 2.413-67.872-23.741-8.164-18.379-6.68-53.768 29.67-67.27 22.706-8.433 73.305 11.029 75.9-13.623 2.992-28.416-53.155-30.812-70.06-37.626-29.912-12.055-47.567-37.85-33.734-65.522 10.378-20.757 40.915-29.203 64.223-20.11 27.922 10.892 32.404 39.853 46.71 51.897 12.324 10.38 29.19 11.68 40.22 4.543 8.135-5.265 10.843-16.828 7.774-27.39-4.07-14.023-14.875-22.773-25.415-31.346-18.758-15.249-45.24-28.36-29.222-69.983 13.13-34.11 51.642-35.34 51.642-35.34 15.3-1.72 29.002 2.9 40.167 12.875 14.927 13.335 17.834 31.16 15.336 50.176-2.283 17.358-8.426 32.56-11.63 49.759-3.717 19.966 6.954 40.086 27.249 40.869 26.694 1.031 34.698-19.486 37.964-32.492 4.782-19.028 11.058-36.694 28.718-47.82 25.346-15.97 60.552-12.47 76.886 18.222 12.92 24.284 8.772 57.715-11.047 75.97-8.892 8.188-19.584 11.075-31.148 11.156-16.585.117-33.162-.29-48.556 7.471-10.48 5.281-15.047 13.888-15.045 25.423 0 11.242 5.853 18.585 15.336 23.363 17.86 9.003 37.577 10.843 56.871 14.222 27.98 4.9 52.581 14.755 68.375 40.72.142.228.28.458.415.69 18.139 30.741-.831 75.005-36.476 76.878" fill="#0AC855"/></svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.4 KiB |
@@ -1 +0,0 @@
|
|||||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 190.24 81.52"><defs><linearGradient id="a" y1="40.76" x2="190.24" y2="40.76" gradientUnits="userSpaceOnUse"><stop offset="0" stop-color="#90cea1"/><stop offset=".56" stop-color="#3cbec9"/><stop offset="1" stop-color="#00b3e5"/></linearGradient></defs><path d="M105.67 36.06h66.9a17.67 17.67 0 0017.67-17.66A17.67 17.67 0 00172.57.73h-66.9A17.67 17.67 0 0088 18.4a17.67 17.67 0 0017.67 17.66zm-88 45h76.9a17.67 17.67 0 0017.67-17.66 17.67 17.67 0 00-17.67-17.67h-76.9A17.67 17.67 0 000 63.4a17.67 17.67 0 0017.67 17.66zm-7.26-45.64h7.8V6.92h10.1V0h-28v6.9h10.1zm28.1 0h7.8V8.25h.1l9 27.15h6l9.3-27.15h.1V35.4h7.8V0H66.76l-8.2 23.1h-.1L50.31 0h-11.8zm113.92 20.25a15.07 15.07 0 00-4.52-5.52 18.57 18.57 0 00-6.68-3.08 33.54 33.54 0 00-8.07-1h-11.7v35.4h12.75a24.58 24.58 0 007.55-1.15 19.34 19.34 0 006.35-3.32 16.27 16.27 0 004.37-5.5 16.91 16.91 0 001.63-7.58 18.5 18.5 0 00-1.68-8.25zM145 68.6a8.8 8.8 0 01-2.64 3.4 10.7 10.7 0 01-4 1.82 21.57 21.57 0 01-5 .55h-4.05v-21h4.6a17 17 0 014.67.63 11.66 11.66 0 013.88 1.87A9.14 9.14 0 01145 59a9.87 9.87 0 011 4.52 11.89 11.89 0 01-1 5.08zm44.63-.13a8 8 0 00-1.58-2.62 8.38 8.38 0 00-2.42-1.85 10.31 10.31 0 00-3.17-1v-.1a9.22 9.22 0 004.42-2.82 7.43 7.43 0 001.68-5 8.42 8.42 0 00-1.15-4.65 8.09 8.09 0 00-3-2.72 12.56 12.56 0 00-4.18-1.3 32.84 32.84 0 00-4.62-.33h-13.2v35.4h14.5a22.41 22.41 0 004.72-.5 13.53 13.53 0 004.28-1.65 9.42 9.42 0 003.1-3 8.52 8.52 0 001.2-4.68 9.39 9.39 0 00-.55-3.18zm-19.42-15.75h5.3a10 10 0 011.85.18 6.18 6.18 0 011.7.57 3.39 3.39 0 011.22 1.13 3.22 3.22 0 01.48 1.82 3.63 3.63 0 01-.43 1.8 3.4 3.4 0 01-1.12 1.2 4.92 4.92 0 01-1.58.65 7.51 7.51 0 01-1.77.2h-5.65zm11.72 20a3.9 3.9 0 01-1.22 1.3 4.64 4.64 0 01-1.68.7 8.18 8.18 0 01-1.82.2h-7v-8h5.9a15.35 15.35 0 012 .15 8.47 8.47 0 012.05.55 4 4 0 011.57 1.18 3.11 3.11 0 01.63 2 3.71 3.71 0 01-.43 1.92z" fill="url(#a)"/></svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.9 KiB |
@@ -1,7 +1,10 @@
|
|||||||
diff --git a/node_modules/react-native-bottom-tabs/ios/BottomAccessoryProvider.swift b/node_modules/react-native-bottom-tabs/ios/BottomAccessoryProvider.swift
|
diff --git a/node_modules/react-native-bottom-tabs/.bun-tag-b32ab1c60a5dfcf7 b/.bun-tag-b32ab1c60a5dfcf7
|
||||||
|
new file mode 100644
|
||||||
|
index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391
|
||||||
|
diff --git a/ios/BottomAccessoryProvider.swift b/ios/BottomAccessoryProvider.swift
|
||||||
index 539efee7156599e1fc795e11bf411b7dfaf12ec7..b2af39a2e6b014e9b1ae0a51b21115c19280df69 100644
|
index 539efee7156599e1fc795e11bf411b7dfaf12ec7..b2af39a2e6b014e9b1ae0a51b21115c19280df69 100644
|
||||||
--- a/node_modules/react-native-bottom-tabs/ios/BottomAccessoryProvider.swift
|
--- a/ios/BottomAccessoryProvider.swift
|
||||||
+++ b/node_modules/react-native-bottom-tabs/ios/BottomAccessoryProvider.swift
|
+++ b/ios/BottomAccessoryProvider.swift
|
||||||
@@ -8,7 +8,7 @@ import SwiftUI
|
@@ -8,7 +8,7 @@ import SwiftUI
|
||||||
self.delegate = delegate
|
self.delegate = delegate
|
||||||
}
|
}
|
||||||
@@ -11,10 +14,10 @@ index 539efee7156599e1fc795e11bf411b7dfaf12ec7..b2af39a2e6b014e9b1ae0a51b21115c1
|
|||||||
@available(iOS 26.0, *)
|
@available(iOS 26.0, *)
|
||||||
public func emitPlacementChanged(_ placement: TabViewBottomAccessoryPlacement?) {
|
public func emitPlacementChanged(_ placement: TabViewBottomAccessoryPlacement?) {
|
||||||
var placementValue = "none"
|
var placementValue = "none"
|
||||||
diff --git a/node_modules/react-native-bottom-tabs/ios/TabView/NewTabView.swift b/node_modules/react-native-bottom-tabs/ios/TabView/NewTabView.swift
|
diff --git a/ios/TabView/NewTabView.swift b/ios/TabView/NewTabView.swift
|
||||||
index 22c52cdf25ad0f7398d89197cb431ca8dc8e0f99..81411376e68803de8bd83515d42565cfa95daf2b 100644
|
index 22c52cdf25ad0f7398d89197cb431ca8dc8e0f99..81411376e68803de8bd83515d42565cfa95daf2b 100644
|
||||||
--- a/node_modules/react-native-bottom-tabs/ios/TabView/NewTabView.swift
|
--- a/ios/TabView/NewTabView.swift
|
||||||
+++ b/node_modules/react-native-bottom-tabs/ios/TabView/NewTabView.swift
|
+++ b/ios/TabView/NewTabView.swift
|
||||||
@@ -78,11 +78,11 @@ struct ConditionalBottomAccessoryModifier: ViewModifier {
|
@@ -78,11 +78,11 @@ struct ConditionalBottomAccessoryModifier: ViewModifier {
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,10 +56,10 @@ index 22c52cdf25ad0f7398d89197cb431ca8dc8e0f99..81411376e68803de8bd83515d42565cf
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
+
|
+
|
||||||
diff --git a/node_modules/react-native-bottom-tabs/ios/TabViewImpl.swift b/node_modules/react-native-bottom-tabs/ios/TabViewImpl.swift
|
diff --git a/ios/TabViewImpl.swift b/ios/TabViewImpl.swift
|
||||||
index 72938be90540ea3a483d7db9a80fb74c04d31272..277278ffdd9268a96cb09869eb1d0c0d5e6ad300 100644
|
index 72938be90540ea3a483d7db9a80fb74c04d31272..277278ffdd9268a96cb09869eb1d0c0d5e6ad300 100644
|
||||||
--- a/node_modules/react-native-bottom-tabs/ios/TabViewImpl.swift
|
--- a/ios/TabViewImpl.swift
|
||||||
+++ b/node_modules/react-native-bottom-tabs/ios/TabViewImpl.swift
|
+++ b/ios/TabViewImpl.swift
|
||||||
@@ -281,7 +281,7 @@ extension View {
|
@@ -281,7 +281,7 @@ extension View {
|
||||||
|
|
||||||
@ViewBuilder
|
@ViewBuilder
|
||||||
@@ -66,10 +69,10 @@ index 72938be90540ea3a483d7db9a80fb74c04d31272..277278ffdd9268a96cb09869eb1d0c0d
|
|||||||
if #available(iOS 26.0, macOS 26.0, *) {
|
if #available(iOS 26.0, macOS 26.0, *) {
|
||||||
if let behavior {
|
if let behavior {
|
||||||
self.tabBarMinimizeBehavior(behavior.convert())
|
self.tabBarMinimizeBehavior(behavior.convert())
|
||||||
diff --git a/node_modules/react-native-bottom-tabs/ios/TabViewProps.swift b/node_modules/react-native-bottom-tabs/ios/TabViewProps.swift
|
diff --git a/ios/TabViewProps.swift b/ios/TabViewProps.swift
|
||||||
index 9cfb29a983b34d3f84fc7a678d19ef4ff30e0325..6a5854483e66200b71722bbac12e100742222bd3 100644
|
index 9cfb29a983b34d3f84fc7a678d19ef4ff30e0325..6a5854483e66200b71722bbac12e100742222bd3 100644
|
||||||
--- a/node_modules/react-native-bottom-tabs/ios/TabViewProps.swift
|
--- a/ios/TabViewProps.swift
|
||||||
+++ b/node_modules/react-native-bottom-tabs/ios/TabViewProps.swift
|
+++ b/ios/TabViewProps.swift
|
||||||
@@ -6,7 +6,7 @@ internal enum MinimizeBehavior: String {
|
@@ -6,7 +6,7 @@ internal enum MinimizeBehavior: String {
|
||||||
case onScrollUp
|
case onScrollUp
|
||||||
case onScrollDown
|
case onScrollDown
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
diff --git a/node_modules/react-native-ios-utilities/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift b/node_modules/react-native-ios-utilities/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift
|
diff --git a/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift b/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift
|
||||||
index 09be306d5aa39337c5114c2ad6ba7513218e0751..24ff8ee2c36fef8632a7e012514fd04db9bf89fd 100644
|
index 09be306d5aa39337c5114c2ad6ba7513218e0751..24ff8ee2c36fef8632a7e012514fd04db9bf89fd 100644
|
||||||
--- a/node_modules/react-native-ios-utilities/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift
|
--- a/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift
|
||||||
+++ b/node_modules/react-native-ios-utilities/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift
|
+++ b/ios/Sources/Extensions+Helpers/RCTView+Helpers.swift
|
||||||
@@ -25,15 +25,14 @@ public extension RCTView {
|
@@ -25,15 +25,14 @@ public extension RCTView {
|
||||||
return rootView.recursivelyFindSubview(whereType: targetType);
|
return rootView.recursivelyFindSubview(whereType: targetType);
|
||||||
};
|
};
|
||||||
@@ -1,7 +1,10 @@
|
|||||||
diff --git a/node_modules/react-native-udp/react-native-udp.podspec b/node_modules/react-native-udp/react-native-udp.podspec
|
diff --git a/node_modules/react-native-udp/.bun-tag-ea7df8754aa4db91 b/.bun-tag-ea7df8754aa4db91
|
||||||
|
new file mode 100644
|
||||||
|
index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391
|
||||||
|
diff --git a/react-native-udp.podspec b/react-native-udp.podspec
|
||||||
index 7450cc7d0862aadfb47d796929c801a3dc423a57..fa3e42c0152ef2d87536b8c2e484f64d525e35ec 100644
|
index 7450cc7d0862aadfb47d796929c801a3dc423a57..fa3e42c0152ef2d87536b8c2e484f64d525e35ec 100644
|
||||||
--- a/node_modules/react-native-udp/react-native-udp.podspec
|
--- a/react-native-udp.podspec
|
||||||
+++ b/node_modules/react-native-udp/react-native-udp.podspec
|
+++ b/react-native-udp.podspec
|
||||||
@@ -9,7 +9,8 @@ Pod::Spec.new do |s|
|
@@ -9,7 +9,8 @@ Pod::Spec.new do |s|
|
||||||
s.homepage = package_json["homepage"]
|
s.homepage = package_json["homepage"]
|
||||||
s.license = package_json["license"]
|
s.license = package_json["license"]
|
||||||
43
bun.lock
@@ -11,10 +11,9 @@
|
|||||||
"@expo/react-native-action-sheet": "^4.1.1",
|
"@expo/react-native-action-sheet": "^4.1.1",
|
||||||
"@expo/ui": "~56.0.14",
|
"@expo/ui": "~56.0.14",
|
||||||
"@expo/vector-icons": "^15.0.3",
|
"@expo/vector-icons": "^15.0.3",
|
||||||
"@gorhom/bottom-sheet": "5.2.14",
|
"@gorhom/bottom-sheet": "5.2.8",
|
||||||
"@jellyfin/sdk": "^0.13.0",
|
"@jellyfin/sdk": "^0.13.0",
|
||||||
"@react-native-community/netinfo": "^12.0.0",
|
"@react-native-community/netinfo": "^12.0.0",
|
||||||
"@react-navigation/material-top-tabs": "7.4.28",
|
|
||||||
"@react-navigation/native": "^7.2.5",
|
"@react-navigation/native": "^7.2.5",
|
||||||
"@shopify/flash-list": "2.0.2",
|
"@shopify/flash-list": "2.0.2",
|
||||||
"@tanstack/query-sync-storage-persister": "^5.100.14",
|
"@tanstack/query-sync-storage-persister": "^5.100.14",
|
||||||
@@ -84,7 +83,6 @@
|
|||||||
"react-native-safe-area-context": "~5.7.0",
|
"react-native-safe-area-context": "~5.7.0",
|
||||||
"react-native-screens": "4.25.2",
|
"react-native-screens": "4.25.2",
|
||||||
"react-native-svg": "15.15.4",
|
"react-native-svg": "15.15.4",
|
||||||
"react-native-tab-view": "4.3.0",
|
|
||||||
"react-native-text-ticker": "^1.15.0",
|
"react-native-text-ticker": "^1.15.0",
|
||||||
"react-native-track-player": "github:lovegaoshi/react-native-track-player#APM",
|
"react-native-track-player": "github:lovegaoshi/react-native-track-player#APM",
|
||||||
"react-native-udp": "^4.1.7",
|
"react-native-udp": "^4.1.7",
|
||||||
@@ -108,7 +106,7 @@
|
|||||||
"@types/react": "~19.2.10",
|
"@types/react": "~19.2.10",
|
||||||
"@types/react-test-renderer": "19.1.0",
|
"@types/react-test-renderer": "19.1.0",
|
||||||
"cross-env": "10.1.0",
|
"cross-env": "10.1.0",
|
||||||
"expo-doctor": "1.19.7",
|
"expo-doctor": "1.19.8",
|
||||||
"husky": "9.1.7",
|
"husky": "9.1.7",
|
||||||
"lint-staged": "17.0.5",
|
"lint-staged": "17.0.5",
|
||||||
"react-test-renderer": "19.2.3",
|
"react-test-renderer": "19.2.3",
|
||||||
@@ -116,6 +114,11 @@
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"patchedDependencies": {
|
||||||
|
"react-native-ios-utilities@5.2.0": "bun-patches/react-native-ios-utilities@5.2.0.patch",
|
||||||
|
"react-native-udp@4.1.7": "bun-patches/react-native-udp@4.1.7.patch",
|
||||||
|
"react-native-bottom-tabs@1.2.0": "bun-patches/react-native-bottom-tabs@1.2.0.patch",
|
||||||
|
},
|
||||||
"packages": {
|
"packages": {
|
||||||
"@adobe/css-tools": ["@adobe/css-tools@4.5.0", "", {}, "sha512-6OzddxPio9UiWTCemp4N8cYLV2ZN1ncRnV1cVGtve7dhPOtRkleRyx32GQCYSwDYgaHU3USMm84tNsvKzRCa1Q=="],
|
"@adobe/css-tools": ["@adobe/css-tools@4.5.0", "", {}, "sha512-6OzddxPio9UiWTCemp4N8cYLV2ZN1ncRnV1cVGtve7dhPOtRkleRyx32GQCYSwDYgaHU3USMm84tNsvKzRCa1Q=="],
|
||||||
|
|
||||||
@@ -365,7 +368,7 @@
|
|||||||
|
|
||||||
"@expo/xcpretty": ["@expo/xcpretty@4.4.4", "", { "dependencies": { "@babel/code-frame": "^7.20.0", "chalk": "^4.1.0", "js-yaml": "^4.1.0" }, "bin": { "excpretty": "build/cli.js" } }, "sha512-4aQzz9vgxcNXFfo/iyNgDDYfsU5XGKKxWxZopw0cVotHiW+U8IJbIxMaxsINs6bHhtkG3StKNPcOrn3eBuxKPw=="],
|
"@expo/xcpretty": ["@expo/xcpretty@4.4.4", "", { "dependencies": { "@babel/code-frame": "^7.20.0", "chalk": "^4.1.0", "js-yaml": "^4.1.0" }, "bin": { "excpretty": "build/cli.js" } }, "sha512-4aQzz9vgxcNXFfo/iyNgDDYfsU5XGKKxWxZopw0cVotHiW+U8IJbIxMaxsINs6bHhtkG3StKNPcOrn3eBuxKPw=="],
|
||||||
|
|
||||||
"@gorhom/bottom-sheet": ["@gorhom/bottom-sheet@5.2.14", "", { "dependencies": { "@gorhom/portal": "1.0.14", "invariant": "^2.2.4" }, "peerDependencies": { "@types/react": "*", "@types/react-native": "*", "react": "*", "react-native": "*", "react-native-gesture-handler": ">=2.16.1", "react-native-reanimated": ">=3.16.0 || >=4.0.0-" }, "optionalPeers": ["@types/react", "@types/react-native"] }, "sha512-uLQFlDjp9z+jrOFcMSEldPqL5JdaXL3vXOh+juhwoNvXgTsEorJLjHTugXu+YccAG/0KJnShzKCrb71MHBsvJg=="],
|
"@gorhom/bottom-sheet": ["@gorhom/bottom-sheet@5.2.8", "", { "dependencies": { "@gorhom/portal": "1.0.14", "invariant": "^2.2.4" }, "peerDependencies": { "@types/react": "*", "@types/react-native": "*", "react": "*", "react-native": "*", "react-native-gesture-handler": ">=2.16.1", "react-native-reanimated": ">=3.16.0 || >=4.0.0-" }, "optionalPeers": ["@types/react", "@types/react-native"] }, "sha512-+N27SMpbBxXZQ/IA2nlEV6RGxL/qSFHKfdFKcygvW+HqPG5jVNb1OqehLQsGfBP+Up42i0gW5ppI+DhpB7UCzA=="],
|
||||||
|
|
||||||
"@gorhom/portal": ["@gorhom/portal@1.0.14", "", { "dependencies": { "nanoid": "^3.3.1" }, "peerDependencies": { "react": "*", "react-native": "*" } }, "sha512-MXyL4xvCjmgaORr/rtryDNFy3kU4qUbKlwtQqqsygd0xX3mhKjOLn6mQK8wfu0RkoE0pBE0nAasRoHua+/QZ7A=="],
|
"@gorhom/portal": ["@gorhom/portal@1.0.14", "", { "dependencies": { "nanoid": "^3.3.1" }, "peerDependencies": { "react": "*", "react-native": "*" } }, "sha512-MXyL4xvCjmgaORr/rtryDNFy3kU4qUbKlwtQqqsygd0xX3mhKjOLn6mQK8wfu0RkoE0pBE0nAasRoHua+/QZ7A=="],
|
||||||
|
|
||||||
@@ -539,10 +542,6 @@
|
|||||||
|
|
||||||
"@react-navigation/core": ["@react-navigation/core@7.17.5", "", { "dependencies": { "@react-navigation/routers": "^7.5.5", "escape-string-regexp": "^4.0.0", "fast-deep-equal": "^3.1.3", "nanoid": "^3.3.11", "query-string": "^7.1.3", "react-is": "^19.1.0", "use-latest-callback": "^0.2.4", "use-sync-external-store": "^1.5.0" }, "peerDependencies": { "react": ">= 18.2.0" } }, "sha512-6fDCwDTWC7DJn0SDb9DJGRlipaygHIc+2elpZBJI6Crl/2Pu+Z1d6W4jMJ2gZO6iHKf+Pe5sUiQ/uwepGprZtg=="],
|
"@react-navigation/core": ["@react-navigation/core@7.17.5", "", { "dependencies": { "@react-navigation/routers": "^7.5.5", "escape-string-regexp": "^4.0.0", "fast-deep-equal": "^3.1.3", "nanoid": "^3.3.11", "query-string": "^7.1.3", "react-is": "^19.1.0", "use-latest-callback": "^0.2.4", "use-sync-external-store": "^1.5.0" }, "peerDependencies": { "react": ">= 18.2.0" } }, "sha512-6fDCwDTWC7DJn0SDb9DJGRlipaygHIc+2elpZBJI6Crl/2Pu+Z1d6W4jMJ2gZO6iHKf+Pe5sUiQ/uwepGprZtg=="],
|
||||||
|
|
||||||
"@react-navigation/elements": ["@react-navigation/elements@2.9.19", "", { "dependencies": { "color": "^4.2.3", "use-latest-callback": "^0.2.4", "use-sync-external-store": "^1.5.0" }, "peerDependencies": { "@react-native-masked-view/masked-view": ">= 0.2.0", "@react-navigation/native": "^7.2.5", "react": ">= 18.2.0", "react-native": "*", "react-native-safe-area-context": ">= 4.0.0" }, "optionalPeers": ["@react-native-masked-view/masked-view"] }, "sha512-gBUvCZuUkOGw1KpLQEZIkByUz8RYPwXeoA6mZFJy9K1mxd8GdqHDMFCIoB0lfPz9rgrHj99RvtdlGZ/ZzkZv2A=="],
|
|
||||||
|
|
||||||
"@react-navigation/material-top-tabs": ["@react-navigation/material-top-tabs@7.4.28", "", { "dependencies": { "@react-navigation/elements": "^2.9.19", "color": "^4.2.3", "react-native-tab-view": "^4.3.0" }, "peerDependencies": { "@react-navigation/native": "^7.2.5", "react": ">= 18.2.0", "react-native": "*", "react-native-pager-view": ">= 6.0.0", "react-native-safe-area-context": ">= 4.0.0" } }, "sha512-WZHJSGV2PQOD2Vr9LF8apGvcsbDKukzF3Fhh8xVNIesqaSi9TPProv4dRw6YkenUkjvFVZYkOjvwAJOToePVpA=="],
|
|
||||||
|
|
||||||
"@react-navigation/native": ["@react-navigation/native@7.2.5", "", { "dependencies": { "@react-navigation/core": "^7.17.5", "escape-string-regexp": "^4.0.0", "fast-deep-equal": "^3.1.3", "nanoid": "^3.3.11", "use-latest-callback": "^0.2.4" }, "peerDependencies": { "react": ">= 18.2.0", "react-native": "*" } }, "sha512-01AAUQiiHQAfTabq+ZyU1/ZWq+AbB/J3v0CB0UTJSON6M6cuadWNsbChzrZUdqQvHrXvg96U5i2PQLJzK3+zpg=="],
|
"@react-navigation/native": ["@react-navigation/native@7.2.5", "", { "dependencies": { "@react-navigation/core": "^7.17.5", "escape-string-regexp": "^4.0.0", "fast-deep-equal": "^3.1.3", "nanoid": "^3.3.11", "use-latest-callback": "^0.2.4" }, "peerDependencies": { "react": ">= 18.2.0", "react-native": "*" } }, "sha512-01AAUQiiHQAfTabq+ZyU1/ZWq+AbB/J3v0CB0UTJSON6M6cuadWNsbChzrZUdqQvHrXvg96U5i2PQLJzK3+zpg=="],
|
||||||
|
|
||||||
"@react-navigation/routers": ["@react-navigation/routers@7.5.5", "", { "dependencies": { "nanoid": "^3.3.11" } }, "sha512-9/hhMte12Kgu+pMnLfA4EWJ0OQmIEAMVMX06FPH2yGkEQSQ3JhhCN/GkcRikzQhtEi97VYYQA15umptBUShcOQ=="],
|
"@react-navigation/routers": ["@react-navigation/routers@7.5.5", "", { "dependencies": { "nanoid": "^3.3.11" } }, "sha512-9/hhMte12Kgu+pMnLfA4EWJ0OQmIEAMVMX06FPH2yGkEQSQ3JhhCN/GkcRikzQhtEi97VYYQA15umptBUShcOQ=="],
|
||||||
@@ -969,7 +968,7 @@
|
|||||||
|
|
||||||
"expo-device": ["expo-device@56.0.4", "", { "dependencies": { "ua-parser-js": "^0.7.33" }, "peerDependencies": { "expo": "*" } }, "sha512-ucVcGPkvBrl2QHuy7XcYex2Y6BETvJ6TREutZrwLGUDnlvbpKS8KfQoNZOpvkyo5Nmm9RrasYQ0CrXmBHho2mg=="],
|
"expo-device": ["expo-device@56.0.4", "", { "dependencies": { "ua-parser-js": "^0.7.33" }, "peerDependencies": { "expo": "*" } }, "sha512-ucVcGPkvBrl2QHuy7XcYex2Y6BETvJ6TREutZrwLGUDnlvbpKS8KfQoNZOpvkyo5Nmm9RrasYQ0CrXmBHho2mg=="],
|
||||||
|
|
||||||
"expo-doctor": ["expo-doctor@1.19.7", "", { "bin": { "expo-doctor": "bin/expo-doctor.js" } }, "sha512-pzn7QtCifRlvGIQz8k7kszeYFaI5Yn81WTHlk/20tmd3jwnXxPjlcdyhFSkuRtO2v4a9gA/6aUWVBOosfffj9w=="],
|
"expo-doctor": ["expo-doctor@1.19.8", "", { "bin": { "expo-doctor": "bin/expo-doctor.js" } }, "sha512-ZHpQM+BfJe1DNaA+/ObtLYazC2x78tIV3kkfoSGR46Tj1EvzOFh1p9gFHsILI9TOyXPEcgxCkFw9AVvf8C4c1g=="],
|
||||||
|
|
||||||
"expo-file-system": ["expo-file-system@56.0.7", "", { "peerDependencies": { "expo": "*", "react-native": "*" } }, "sha512-dcKzo8ShPloM7jgfnMcJStgQebhP8owVjCkNI/aX6NMFV1CYB8bxKGMdnzJ3mXk5nfaiW+F/lSKr2UIJ02WAUA=="],
|
"expo-file-system": ["expo-file-system@56.0.7", "", { "peerDependencies": { "expo": "*", "react-native": "*" } }, "sha512-dcKzo8ShPloM7jgfnMcJStgQebhP8owVjCkNI/aX6NMFV1CYB8bxKGMdnzJ3mXk5nfaiW+F/lSKr2UIJ02WAUA=="],
|
||||||
|
|
||||||
@@ -1595,11 +1594,9 @@
|
|||||||
|
|
||||||
"react-native-svg": ["react-native-svg@15.15.4", "", { "dependencies": { "css-select": "^5.1.0", "css-tree": "^1.1.3", "warn-once": "0.1.1" }, "peerDependencies": { "react": "*", "react-native": "*" } }, "sha512-boT/vIRgj6zZKBpfTPJJiYWMbZE9duBMOwPK6kCSTgxsS947IFMOq9OgIFkpWZTB7t229H24pDRkh3W9ZK/J1A=="],
|
"react-native-svg": ["react-native-svg@15.15.4", "", { "dependencies": { "css-select": "^5.1.0", "css-tree": "^1.1.3", "warn-once": "0.1.1" }, "peerDependencies": { "react": "*", "react-native": "*" } }, "sha512-boT/vIRgj6zZKBpfTPJJiYWMbZE9duBMOwPK6kCSTgxsS947IFMOq9OgIFkpWZTB7t229H24pDRkh3W9ZK/J1A=="],
|
||||||
|
|
||||||
"react-native-tab-view": ["react-native-tab-view@4.3.0", "", { "dependencies": { "use-latest-callback": "^0.2.4" }, "peerDependencies": { "react": ">= 18.2.0", "react-native": "*", "react-native-pager-view": ">= 6.0.0" } }, "sha512-qPMF75uz/7+MuVG2g+YETdGMzlWZnhC6iI4h/7EBbwIBwNBIBi2z4OA6KhY3IOOBwGHXEIz5IyA6doDqifYBHg=="],
|
|
||||||
|
|
||||||
"react-native-text-ticker": ["react-native-text-ticker@1.15.0", "", {}, "sha512-d/uK+PIOhsYMy1r8h825iq/nADiHsabz3WMbRJSnkpQYn+K9aykUAXRRhu8ZbTAzk4CgnUWajJEFxS5ZDygsdg=="],
|
"react-native-text-ticker": ["react-native-text-ticker@1.15.0", "", {}, "sha512-d/uK+PIOhsYMy1r8h825iq/nADiHsabz3WMbRJSnkpQYn+K9aykUAXRRhu8ZbTAzk4CgnUWajJEFxS5ZDygsdg=="],
|
||||||
|
|
||||||
"react-native-track-player": ["react-native-track-player@github:lovegaoshi/react-native-track-player#33a3ecd", { "peerDependencies": { "react": "*", "react-native": "*", "react-native-windows": "*", "shaka-player": "^4.7.9" }, "optionalPeers": ["react-native-windows", "shaka-player"] }, "lovegaoshi-react-native-track-player-33a3ecd"],
|
"react-native-track-player": ["react-native-track-player@github:lovegaoshi/react-native-track-player#33a3ecd", { "peerDependencies": { "react": "*", "react-native": "*", "react-native-windows": "*", "shaka-player": "^4.7.9" }, "optionalPeers": ["react-native-windows", "shaka-player"] }, "lovegaoshi-react-native-track-player-33a3ecd", "sha512-vfkld2jUj7EPkAjIc/Vbx4Q4MtOOLmYtCYCE2dWJsyLnPqgj1f0xVzBxbeVP7dfT+eSh4KIXfdxESXaHgrXIlw=="],
|
||||||
|
|
||||||
"react-native-udp": ["react-native-udp@4.1.7", "", { "dependencies": { "buffer": "^5.6.0", "events": "^3.1.0" } }, "sha512-NUE3zewu61NCdSsLlj+l0ad6qojcVEZPT4hVG/x6DU9U4iCzwtfZSASh9vm7teAcVzLkdD+cO3411LHshAi/wA=="],
|
"react-native-udp": ["react-native-udp@4.1.7", "", { "dependencies": { "buffer": "^5.6.0", "events": "^3.1.0" } }, "sha512-NUE3zewu61NCdSsLlj+l0ad6qojcVEZPT4hVG/x6DU9U4iCzwtfZSASh9vm7teAcVzLkdD+cO3411LHshAi/wA=="],
|
||||||
|
|
||||||
@@ -2009,10 +2006,6 @@
|
|||||||
|
|
||||||
"@react-native/metro-babel-transformer/@babel/core": ["@babel/core@7.28.6", "", { "dependencies": { "@babel/code-frame": "^7.28.6", "@babel/generator": "^7.28.6", "@babel/helper-compilation-targets": "^7.28.6", "@babel/helper-module-transforms": "^7.28.6", "@babel/helpers": "^7.28.6", "@babel/parser": "^7.28.6", "@babel/template": "^7.28.6", "@babel/traverse": "^7.28.6", "@babel/types": "^7.28.6", "@jridgewell/remapping": "^2.3.5", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", "json5": "^2.2.3", "semver": "^6.3.1" } }, "sha512-H3mcG6ZDLTlYfaSNi0iOKkigqMFvkTKlGUYlD8GW7nNOYRrevuA46iTypPyv+06V3fEmvvazfntkBU34L0azAw=="],
|
"@react-native/metro-babel-transformer/@babel/core": ["@babel/core@7.28.6", "", { "dependencies": { "@babel/code-frame": "^7.28.6", "@babel/generator": "^7.28.6", "@babel/helper-compilation-targets": "^7.28.6", "@babel/helper-module-transforms": "^7.28.6", "@babel/helpers": "^7.28.6", "@babel/parser": "^7.28.6", "@babel/template": "^7.28.6", "@babel/traverse": "^7.28.6", "@babel/types": "^7.28.6", "@jridgewell/remapping": "^2.3.5", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", "json5": "^2.2.3", "semver": "^6.3.1" } }, "sha512-H3mcG6ZDLTlYfaSNi0iOKkigqMFvkTKlGUYlD8GW7nNOYRrevuA46iTypPyv+06V3fEmvvazfntkBU34L0azAw=="],
|
||||||
|
|
||||||
"@react-navigation/elements/color": ["color@4.2.3", "", { "dependencies": { "color-convert": "^2.0.1", "color-string": "^1.9.0" } }, "sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A=="],
|
|
||||||
|
|
||||||
"@react-navigation/material-top-tabs/color": ["color@4.2.3", "", { "dependencies": { "color-convert": "^2.0.1", "color-string": "^1.9.0" } }, "sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A=="],
|
|
||||||
|
|
||||||
"@testing-library/dom/aria-query": ["aria-query@5.3.0", "", { "dependencies": { "dequal": "^2.0.3" } }, "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A=="],
|
"@testing-library/dom/aria-query": ["aria-query@5.3.0", "", { "dependencies": { "dequal": "^2.0.3" } }, "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A=="],
|
||||||
|
|
||||||
"@testing-library/dom/dom-accessibility-api": ["dom-accessibility-api@0.5.16", "", {}, "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg=="],
|
"@testing-library/dom/dom-accessibility-api": ["dom-accessibility-api@0.5.16", "", {}, "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg=="],
|
||||||
@@ -2231,14 +2224,6 @@
|
|||||||
|
|
||||||
"@react-native-community/cli-server-api/open/is-wsl": ["is-wsl@1.1.0", "", {}, "sha512-gfygJYZ2gLTDlmbWMI0CE2MwnFzSN/2SZfkMlItC4K/JBlsWVDB0bO6XhqcY13YXE7iMcAJnzTCJjPiTeJJ0Mw=="],
|
"@react-native-community/cli-server-api/open/is-wsl": ["is-wsl@1.1.0", "", {}, "sha512-gfygJYZ2gLTDlmbWMI0CE2MwnFzSN/2SZfkMlItC4K/JBlsWVDB0bO6XhqcY13YXE7iMcAJnzTCJjPiTeJJ0Mw=="],
|
||||||
|
|
||||||
"@react-navigation/elements/color/color-convert": ["color-convert@2.0.1", "", { "dependencies": { "color-name": "~1.1.4" } }, "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ=="],
|
|
||||||
|
|
||||||
"@react-navigation/elements/color/color-string": ["color-string@1.9.1", "", { "dependencies": { "color-name": "^1.0.0", "simple-swizzle": "^0.2.2" } }, "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg=="],
|
|
||||||
|
|
||||||
"@react-navigation/material-top-tabs/color/color-convert": ["color-convert@2.0.1", "", { "dependencies": { "color-name": "~1.1.4" } }, "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ=="],
|
|
||||||
|
|
||||||
"@react-navigation/material-top-tabs/color/color-string": ["color-string@1.9.1", "", { "dependencies": { "color-name": "^1.0.0", "simple-swizzle": "^0.2.2" } }, "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg=="],
|
|
||||||
|
|
||||||
"@testing-library/dom/pretty-format/react-is": ["react-is@17.0.2", "", {}, "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w=="],
|
"@testing-library/dom/pretty-format/react-is": ["react-is@17.0.2", "", {}, "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w=="],
|
||||||
|
|
||||||
"ansi-fragments/slice-ansi/ansi-styles": ["ansi-styles@3.2.1", "", { "dependencies": { "color-convert": "^1.9.0" } }, "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA=="],
|
"ansi-fragments/slice-ansi/ansi-styles": ["ansi-styles@3.2.1", "", { "dependencies": { "color-convert": "^1.9.0" } }, "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA=="],
|
||||||
@@ -2351,14 +2336,6 @@
|
|||||||
|
|
||||||
"@expo/package-manager/ora/strip-ansi/ansi-regex": ["ansi-regex@4.1.1", "", {}, "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g=="],
|
"@expo/package-manager/ora/strip-ansi/ansi-regex": ["ansi-regex@4.1.1", "", {}, "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g=="],
|
||||||
|
|
||||||
"@react-navigation/elements/color/color-convert/color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
|
|
||||||
|
|
||||||
"@react-navigation/elements/color/color-string/color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
|
|
||||||
|
|
||||||
"@react-navigation/material-top-tabs/color/color-convert/color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
|
|
||||||
|
|
||||||
"@react-navigation/material-top-tabs/color/color-string/color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
|
|
||||||
|
|
||||||
"ansi-fragments/slice-ansi/ansi-styles/color-convert": ["color-convert@1.9.3", "", { "dependencies": { "color-name": "1.1.3" } }, "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg=="],
|
"ansi-fragments/slice-ansi/ansi-styles/color-convert": ["color-convert@1.9.3", "", { "dependencies": { "color-name": "1.1.3" } }, "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg=="],
|
||||||
|
|
||||||
"chalk/ansi-styles/color-convert/color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
|
"chalk/ansi-styles/color-convert/color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="],
|
||||||
|
|||||||
@@ -89,14 +89,14 @@ export const IntroSheet = forwardRef<IntroSheetRef>((_, ref) => {
|
|||||||
</Text>
|
</Text>
|
||||||
<View className='flex flex-row items-center mt-4'>
|
<View className='flex flex-row items-center mt-4'>
|
||||||
<Image
|
<Image
|
||||||
source={require("@/assets/icons/seerr-logo.svg")}
|
source={require("@/assets/icons/jellyseerr-logo.svg")}
|
||||||
style={{
|
style={{
|
||||||
width: 50,
|
width: 50,
|
||||||
height: 50,
|
height: 50,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View className='shrink ml-2'>
|
<View className='shrink ml-2'>
|
||||||
<Text className='font-bold mb-1'>Seerr</Text>
|
<Text className='font-bold mb-1'>Jellyseerr</Text>
|
||||||
<Text className='shrink text-xs'>
|
<Text className='shrink text-xs'>
|
||||||
{t("home.intro.jellyseerr_feature_description")}
|
{t("home.intro.jellyseerr_feature_description")}
|
||||||
</Text>
|
</Text>
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
import { Ionicons } from "@expo/vector-icons";
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import { BottomSheetScrollView } from "@gorhom/bottom-sheet";
|
import { BottomSheetScrollView } from "@gorhom/bottom-sheet";
|
||||||
import React, { useEffect } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { Platform, StyleSheet, TouchableOpacity, View } from "react-native";
|
import {
|
||||||
|
type LayoutChangeEvent,
|
||||||
|
Platform,
|
||||||
|
StyleSheet,
|
||||||
|
TouchableOpacity,
|
||||||
|
View,
|
||||||
|
} from "react-native";
|
||||||
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { useGlobalModal } from "@/providers/GlobalModalProvider";
|
import { useGlobalModal } from "@/providers/GlobalModalProvider";
|
||||||
@@ -211,6 +217,24 @@ const PlatformDropdownComponent = ({
|
|||||||
}: PlatformDropdownProps) => {
|
}: PlatformDropdownProps) => {
|
||||||
const { showModal, hideModal, isVisible } = useGlobalModal();
|
const { showModal, hideModal, isVisible } = useGlobalModal();
|
||||||
|
|
||||||
|
// @expo/ui's <Host> (SDK 55) fills its available space by default, and
|
||||||
|
// `matchContents` doesn't help here: it reports the native Menu's size via
|
||||||
|
// setStyleSize and overrides any explicit size. Instead we measure the
|
||||||
|
// trigger's intrinsic size in plain RN (off-layout) and pin it on the Host.
|
||||||
|
const [triggerSize, setTriggerSize] = useState<{
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
} | null>(null);
|
||||||
|
|
||||||
|
const handleMeasureTrigger = (e: LayoutChangeEvent) => {
|
||||||
|
const { width, height } = e.nativeEvent.layout;
|
||||||
|
setTriggerSize((prev) =>
|
||||||
|
prev && prev.width === width && prev.height === height
|
||||||
|
? prev
|
||||||
|
: { width, height },
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
// Handle controlled open state for Android
|
// Handle controlled open state for Android
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (Platform.OS === "android" && controlledOpen === true) {
|
if (Platform.OS === "android" && controlledOpen === true) {
|
||||||
@@ -241,11 +265,25 @@ const PlatformDropdownComponent = ({
|
|||||||
}, [isVisible, controlledOpen, controlledOnOpenChange]);
|
}, [isVisible, controlledOpen, controlledOnOpenChange]);
|
||||||
|
|
||||||
if (Platform.OS === "ios" && !Platform.isTV) {
|
if (Platform.OS === "ios" && !Platform.isTV) {
|
||||||
// @expo/ui's <Host> can't size to content, so an in-flow invisible copy of
|
// Pin the wrapper to the measured trigger size. @expo/ui's <Host> (SDK 55)
|
||||||
// the trigger sizes the wrapper while the Host overlays the real Menu.
|
// fills its parent and reports its own size via setStyleSize, so it can't
|
||||||
|
// size itself to content. If the wrapper has no size, the Host's `flex: 1`
|
||||||
|
// height depends on the parent while the parent depends on the Host — a
|
||||||
|
// circular dependency that collapses to 0 for any selector nested more than
|
||||||
|
// one level deep (so only the first, shallowest dropdown stays visible).
|
||||||
|
// Giving the wrapper the measured size breaks the cycle; the Host then
|
||||||
|
// fills a concrete box.
|
||||||
return (
|
return (
|
||||||
<View>
|
<View style={triggerSize ?? { opacity: 0 }}>
|
||||||
<View pointerEvents='none' aria-hidden style={{ opacity: 0 }}>
|
{/* Hidden measurer: lays the trigger out off-flow to capture its
|
||||||
|
intrinsic size. Absolutely positioned WITHOUT right/bottom so it
|
||||||
|
sizes to the trigger's content rather than to its parent. */}
|
||||||
|
<View
|
||||||
|
style={{ position: "absolute", top: 0, left: 0, opacity: 0 }}
|
||||||
|
pointerEvents='none'
|
||||||
|
aria-hidden
|
||||||
|
onLayout={handleMeasureTrigger}
|
||||||
|
>
|
||||||
{trigger}
|
{trigger}
|
||||||
</View>
|
</View>
|
||||||
<Host style={[StyleSheet.absoluteFill, expoUIConfig?.hostStyle as any]}>
|
<Host style={[StyleSheet.absoluteFill, expoUIConfig?.hostStyle as any]}>
|
||||||
|
|||||||
@@ -40,8 +40,8 @@ export const Ratings: React.FC<Props> = ({ item, ...props }) => {
|
|||||||
<Image
|
<Image
|
||||||
source={
|
source={
|
||||||
item.CriticRating < 60
|
item.CriticRating < 60
|
||||||
? require("@/assets/images/rt_rotten.svg")
|
? require("@/assets/images/rotten-tomatoes.png")
|
||||||
: require("@/assets/images/rt_fresh.svg")
|
: require("@/assets/images/not-rotten-tomatoes.svg")
|
||||||
}
|
}
|
||||||
style={{
|
style={{
|
||||||
width: 14,
|
width: 14,
|
||||||
@@ -89,8 +89,8 @@ export const JellyserrRatings: React.FC<{
|
|||||||
className='mr-1'
|
className='mr-1'
|
||||||
source={
|
source={
|
||||||
data?.criticsRating === "Rotten"
|
data?.criticsRating === "Rotten"
|
||||||
? require("@/assets/images/rt_rotten.svg")
|
? require("@/utils/jellyseerr/src/assets/rt_rotten.svg")
|
||||||
: require("@/assets/images/rt_fresh.svg")
|
: require("@/utils/jellyseerr/src/assets/rt_fresh.svg")
|
||||||
}
|
}
|
||||||
style={{
|
style={{
|
||||||
width: 14,
|
width: 14,
|
||||||
@@ -109,8 +109,8 @@ export const JellyserrRatings: React.FC<{
|
|||||||
className='mr-1'
|
className='mr-1'
|
||||||
source={
|
source={
|
||||||
data?.audienceRating === "Spilled"
|
data?.audienceRating === "Spilled"
|
||||||
? require("@/assets/images/rt_aud_rotten.svg")
|
? require("@/utils/jellyseerr/src/assets/rt_aud_rotten.svg")
|
||||||
: require("@/assets/images/rt_aud_fresh.svg")
|
: require("@/utils/jellyseerr/src/assets/rt_aud_fresh.svg")
|
||||||
}
|
}
|
||||||
style={{
|
style={{
|
||||||
width: 14,
|
width: 14,
|
||||||
@@ -127,7 +127,7 @@ export const JellyserrRatings: React.FC<{
|
|||||||
iconLeft={
|
iconLeft={
|
||||||
<Image
|
<Image
|
||||||
className='mr-1'
|
className='mr-1'
|
||||||
source={require("@/assets/images/tmdb_logo.svg")}
|
source={require("@/utils/jellyseerr/src/assets/tmdb_logo.svg")}
|
||||||
style={{
|
style={{
|
||||||
width: 14,
|
width: 14,
|
||||||
height: 14,
|
height: 14,
|
||||||
|
|||||||
@@ -11,7 +11,6 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { FlatList, Modal, Pressable, StyleSheet, View } from "react-native";
|
import { FlatList, Modal, Pressable, StyleSheet, View } from "react-native";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { Colors } from "@/constants/Colors";
|
import { Colors } from "@/constants/Colors";
|
||||||
import { useControlsSafeAreaInsets } from "@/hooks/useControlsSafeAreaInsets";
|
|
||||||
import {
|
import {
|
||||||
type ChapterEntry,
|
type ChapterEntry,
|
||||||
chapterStartsMs,
|
chapterStartsMs,
|
||||||
@@ -39,7 +38,6 @@ function ChapterListComponent({
|
|||||||
onClose,
|
onClose,
|
||||||
}: ChapterListProps) {
|
}: ChapterListProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const safeArea = useControlsSafeAreaInsets();
|
|
||||||
const listRef = useRef<FlatList<ChapterEntry>>(null);
|
const listRef = useRef<FlatList<ChapterEntry>>(null);
|
||||||
|
|
||||||
const entries = useMemo(() => sortedChapters(chapters), [chapters]);
|
const entries = useMemo(() => sortedChapters(chapters), [chapters]);
|
||||||
@@ -76,22 +74,9 @@ function ChapterListComponent({
|
|||||||
transparent
|
transparent
|
||||||
animationType='slide'
|
animationType='slide'
|
||||||
onRequestClose={onClose}
|
onRequestClose={onClose}
|
||||||
// iOS defaults <Modal> to portrait-only; without this it rotates the app
|
|
||||||
// back to portrait when opened from the landscape player. Android ignores it.
|
|
||||||
supportedOrientations={["portrait", "landscape"]}
|
|
||||||
>
|
>
|
||||||
<Pressable onPress={onClose} style={styles.backdrop}>
|
<Pressable onPress={onClose} style={styles.backdrop}>
|
||||||
<Pressable
|
<Pressable onPress={(e) => e.stopPropagation()} style={styles.sheet}>
|
||||||
onPress={(e) => e.stopPropagation()}
|
|
||||||
style={[
|
|
||||||
styles.sheet,
|
|
||||||
{
|
|
||||||
marginLeft: safeArea.left,
|
|
||||||
marginRight: safeArea.right,
|
|
||||||
paddingBottom: safeArea.bottom,
|
|
||||||
},
|
|
||||||
]}
|
|
||||||
>
|
|
||||||
<View style={styles.header}>
|
<View style={styles.header}>
|
||||||
<Text style={styles.title}>{t("chapters.title")}</Text>
|
<Text style={styles.title}>{t("chapters.title")}</Text>
|
||||||
<Pressable
|
<Pressable
|
||||||
@@ -172,12 +157,14 @@ const styles = StyleSheet.create({
|
|||||||
backdrop: {
|
backdrop: {
|
||||||
flex: 1,
|
flex: 1,
|
||||||
justifyContent: "flex-end",
|
justifyContent: "flex-end",
|
||||||
|
backgroundColor: "rgba(0,0,0,0.6)",
|
||||||
},
|
},
|
||||||
sheet: {
|
sheet: {
|
||||||
backgroundColor: Colors.background,
|
backgroundColor: Colors.background,
|
||||||
borderTopLeftRadius: 16,
|
borderTopLeftRadius: 16,
|
||||||
borderTopRightRadius: 16,
|
borderTopRightRadius: 16,
|
||||||
maxHeight: "70%",
|
maxHeight: "70%",
|
||||||
|
paddingBottom: 24,
|
||||||
},
|
},
|
||||||
header: {
|
header: {
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
|
|||||||
@@ -37,12 +37,11 @@ export const ProgressBar: React.FC<ProgressBarProps> = ({ item }) => {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<View
|
<View
|
||||||
style={
|
style={{
|
||||||
Platform.isTV
|
width: `${progress}%`,
|
||||||
? { width: `${progress}%`, backgroundColor: "#ffffff" }
|
backgroundColor: Platform.isTV ? "#ffffff" : undefined,
|
||||||
: { width: `${progress}%` }
|
}}
|
||||||
}
|
className={`absolute bottom-0 left-0 h-1 w-full ${Platform.isTV ? "" : "bg-purple-600"}`}
|
||||||
className={`absolute bottom-0 left-0 h-1 ${Platform.isTV ? "" : "bg-purple-600"}`}
|
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import { useActionSheet } from "@expo/react-native-action-sheet";
|
|||||||
import type { BaseItemDto } from "@jellyfin/sdk/lib/generated-client/models";
|
import type { BaseItemDto } from "@jellyfin/sdk/lib/generated-client/models";
|
||||||
import { useSegments } from "expo-router";
|
import { useSegments } from "expo-router";
|
||||||
import { type PropsWithChildren, useCallback } from "react";
|
import { type PropsWithChildren, useCallback } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import {
|
import {
|
||||||
Platform,
|
Platform,
|
||||||
TouchableOpacity,
|
TouchableOpacity,
|
||||||
@@ -150,7 +149,6 @@ export const TouchableItemRouter: React.FC<PropsWithChildren<Props>> = ({
|
|||||||
children,
|
children,
|
||||||
...props
|
...props
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
|
||||||
const segments = useSegments();
|
const segments = useSegments();
|
||||||
const { showActionSheetWithOptions } = useActionSheet();
|
const { showActionSheetWithOptions } = useActionSheet();
|
||||||
const markAsPlayedStatus = useMarkAsPlayed([item]);
|
const markAsPlayedStatus = useMarkAsPlayed([item]);
|
||||||
@@ -184,13 +182,11 @@ export const TouchableItemRouter: React.FC<PropsWithChildren<Props>> = ({
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
const options: string[] = [
|
const options: string[] = [
|
||||||
t("common.mark_as_played"),
|
"Mark as Played",
|
||||||
t("common.mark_as_not_played"),
|
"Mark as Not Played",
|
||||||
isFavorite
|
isFavorite ? "Unmark as Favorite" : "Mark as Favorite",
|
||||||
? t("music.track_options.remove_from_favorites")
|
...(isOffline ? ["Delete Download"] : []),
|
||||||
: t("music.track_options.add_to_favorites"),
|
"Cancel",
|
||||||
...(isOffline ? [t("home.downloads.delete_download")] : []),
|
|
||||||
t("common.cancel"),
|
|
||||||
];
|
];
|
||||||
const cancelButtonIndex = options.length - 1;
|
const cancelButtonIndex = options.length - 1;
|
||||||
const destructiveButtonIndex = isOffline
|
const destructiveButtonIndex = isOffline
|
||||||
@@ -223,7 +219,6 @@ export const TouchableItemRouter: React.FC<PropsWithChildren<Props>> = ({
|
|||||||
isOffline,
|
isOffline,
|
||||||
deleteFile,
|
deleteFile,
|
||||||
item.Id,
|
item.Id,
|
||||||
t,
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (
|
if (
|
||||||
|
|||||||
@@ -133,6 +133,7 @@ const HomeMobile = () => {
|
|||||||
onPress={() => {
|
onPress={() => {
|
||||||
router.push("/(auth)/downloads");
|
router.push("/(auth)/downloads");
|
||||||
}}
|
}}
|
||||||
|
className='ml-1.5'
|
||||||
style={{ marginRight: Platform.OS === "android" ? 16 : 0 }}
|
style={{ marginRight: Platform.OS === "android" ? 16 : 0 }}
|
||||||
>
|
>
|
||||||
<Feather
|
<Feather
|
||||||
|
|||||||
@@ -37,20 +37,7 @@ export const ItemPeopleSections: React.FC<Props> = ({ item, ...props }) => {
|
|||||||
return { ...item, People: people } as BaseItemDto;
|
return { ...item, People: people } as BaseItemDto;
|
||||||
}, [item, people]);
|
}, [item, people]);
|
||||||
|
|
||||||
// Jellyfin can list the same person several times (e.g. an actor also
|
const topPeople = useMemo(() => people.slice(0, 3), [people]);
|
||||||
// credited as writer). Dedupe by Id so the same actor section isn't rendered
|
|
||||||
// twice and we still surface 3 distinct people.
|
|
||||||
const topPeople = useMemo(() => {
|
|
||||||
const seen = new Set<string>();
|
|
||||||
const unique: BaseItemPerson[] = [];
|
|
||||||
for (const person of people) {
|
|
||||||
if (!person.Id || seen.has(person.Id)) continue;
|
|
||||||
seen.add(person.Id);
|
|
||||||
unique.push(person);
|
|
||||||
if (unique.length >= 3) break;
|
|
||||||
}
|
|
||||||
return unique;
|
|
||||||
}, [people]);
|
|
||||||
|
|
||||||
const renderActorSection = useCallback(
|
const renderActorSection = useCallback(
|
||||||
(person: BaseItemPerson, idx: number, total: number) => {
|
(person: BaseItemPerson, idx: number, total: number) => {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { t } from "i18next";
|
import { t } from "i18next";
|
||||||
import React, { useCallback, useState } from "react";
|
import React, { useCallback, useState } from "react";
|
||||||
import { Platform, ScrollView, View } from "react-native";
|
import { ScrollView, View } from "react-native";
|
||||||
import { Button } from "@/components/Button";
|
import { Button } from "@/components/Button";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { useScaledTVTypography } from "@/constants/TVTypography";
|
import { useScaledTVTypography } from "@/constants/TVTypography";
|
||||||
@@ -107,7 +107,7 @@ export const TVAddServerForm: React.FC<TVAddServerFormProps> = ({
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
{/* Pair with Phone */}
|
{/* Pair with Phone */}
|
||||||
{Platform.OS !== "ios" && onStartPairing && (
|
{onStartPairing && (
|
||||||
<View>
|
<View>
|
||||||
<Button
|
<Button
|
||||||
onPress={onStartPairing}
|
onPress={onStartPairing}
|
||||||
|
|||||||
@@ -401,6 +401,10 @@ export const TVJellyseerrSearchResults: React.FC<
|
|||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const hasMovies = movieResults && movieResults.length > 0;
|
||||||
|
const hasTv = tvResults && tvResults.length > 0;
|
||||||
|
const hasPersons = personResults && personResults.length > 0;
|
||||||
|
|
||||||
if (loading) {
|
if (loading) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -427,26 +431,22 @@ export const TVJellyseerrSearchResults: React.FC<
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<View>
|
<View>
|
||||||
{/* No section requests `hasTVPreferredFocus`: the native search field
|
|
||||||
keeps focus while typing, otherwise the first result would re-grab
|
|
||||||
focus on every keystroke as results re-render. The user navigates
|
|
||||||
down to the grid manually. */}
|
|
||||||
<TVJellyseerrMovieSection
|
<TVJellyseerrMovieSection
|
||||||
title={t("search.request_movies")}
|
title={t("search.request_movies")}
|
||||||
items={movieResults}
|
items={movieResults}
|
||||||
isFirstSection={false}
|
isFirstSection={hasMovies}
|
||||||
onItemPress={onMoviePress}
|
onItemPress={onMoviePress}
|
||||||
/>
|
/>
|
||||||
<TVJellyseerrTvSection
|
<TVJellyseerrTvSection
|
||||||
title={t("search.request_series")}
|
title={t("search.request_series")}
|
||||||
items={tvResults}
|
items={tvResults}
|
||||||
isFirstSection={false}
|
isFirstSection={!hasMovies && hasTv}
|
||||||
onItemPress={onTvPress}
|
onItemPress={onTvPress}
|
||||||
/>
|
/>
|
||||||
<TVJellyseerrPersonSection
|
<TVJellyseerrPersonSection
|
||||||
title={t("search.actors")}
|
title={t("search.actors")}
|
||||||
items={personResults}
|
items={personResults}
|
||||||
isFirstSection={false}
|
isFirstSection={!hasMovies && !hasTv && hasPersons}
|
||||||
onItemPress={onPersonPress}
|
onItemPress={onPersonPress}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import type { BaseItemDto } from "@jellyfin/sdk/lib/generated-client/models";
|
|||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { useMemo } from "react";
|
import { useMemo } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Platform, ScrollView, TextInput, View } from "react-native";
|
import { ScrollView, View } from "react-native";
|
||||||
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { TVDiscover } from "@/components/jellyseerr/discover/TVDiscover";
|
import { TVDiscover } from "@/components/jellyseerr/discover/TVDiscover";
|
||||||
@@ -231,48 +231,23 @@ export const TVSearchPage: React.FC<TVSearchPageProps> = ({
|
|||||||
paddingTop: insets.top + TOP_PADDING,
|
paddingTop: insets.top + TOP_PADDING,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{/* Search bar: native tvOS SwiftUI `.searchable` on Apple TV, standard
|
{/* Native tvOS search field (SwiftUI `.searchable`, our `tv-search`
|
||||||
TextInput fallback on Android TV (the native module is Apple-only). */}
|
module). It renders the native search bar + grid keyboard and
|
||||||
{Platform.OS === "ios" ? (
|
forwards typed text into the existing query pipeline via setSearch;
|
||||||
<View
|
our own results grid renders below. */}
|
||||||
style={{
|
<View
|
||||||
marginBottom: 24,
|
style={{
|
||||||
height: SEARCH_AREA_HEIGHT,
|
marginBottom: 24,
|
||||||
}}
|
marginHorizontal: HORIZONTAL_PADDING,
|
||||||
>
|
height: SEARCH_AREA_HEIGHT,
|
||||||
{/* No horizontal margin here: the native tvOS search bar centers
|
}}
|
||||||
itself and renders a trailing "Hold to Dictate" hint. */}
|
>
|
||||||
<TvSearchView
|
<TvSearchView
|
||||||
style={{ width: "100%", height: "100%" }}
|
style={{ width: "100%", height: "100%" }}
|
||||||
placeholder={t("search.search")}
|
placeholder={t("search.search")}
|
||||||
onChangeText={(e) => setSearch(e.nativeEvent.text)}
|
onChangeText={(e) => setSearch(e.nativeEvent.text)}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
) : (
|
|
||||||
<View
|
|
||||||
style={{
|
|
||||||
marginHorizontal: HORIZONTAL_PADDING,
|
|
||||||
marginBottom: 24,
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<TextInput
|
|
||||||
style={{
|
|
||||||
height: 56,
|
|
||||||
width: "100%",
|
|
||||||
backgroundColor: "#262626",
|
|
||||||
borderRadius: 12,
|
|
||||||
paddingHorizontal: 20,
|
|
||||||
fontSize: 28,
|
|
||||||
color: "#fff",
|
|
||||||
}}
|
|
||||||
placeholder={t("search.search")}
|
|
||||||
placeholderTextColor='rgba(255,255,255,0.4)'
|
|
||||||
onChangeText={setSearch}
|
|
||||||
defaultValue=''
|
|
||||||
autoFocus={false}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<ScrollView
|
<ScrollView
|
||||||
@@ -305,17 +280,13 @@ export const TVSearchPage: React.FC<TVSearchPageProps> = ({
|
|||||||
{/* Library Search Results */}
|
{/* Library Search Results */}
|
||||||
{isLibraryMode && !loading && (
|
{isLibraryMode && !loading && (
|
||||||
<View style={{ gap: SECTION_GAP }}>
|
<View style={{ gap: SECTION_GAP }}>
|
||||||
{sections.map((section) => (
|
{sections.map((section, index) => (
|
||||||
<TVSearchSection
|
<TVSearchSection
|
||||||
key={section.key}
|
key={section.key}
|
||||||
title={section.title}
|
title={section.title}
|
||||||
items={section.items!}
|
items={section.items!}
|
||||||
orientation={section.orientation || "vertical"}
|
orientation={section.orientation || "vertical"}
|
||||||
// Never auto-focus a result. The native search field owns focus
|
isFirstSection={index === 0}
|
||||||
// while typing; `hasTVPreferredFocus` here would re-grab focus on
|
|
||||||
// every keystroke as results re-render. User navigates down to the
|
|
||||||
// grid manually.
|
|
||||||
isFirstSection={false}
|
|
||||||
onItemPress={onItemPress}
|
onItemPress={onItemPress}
|
||||||
onItemLongPress={onItemLongPress}
|
onItemLongPress={onItemLongPress}
|
||||||
imageUrlGetter={
|
imageUrlGetter={
|
||||||
|
|||||||
@@ -297,12 +297,12 @@ export const TVSearchSection: React.FC<TVSearchSectionProps> = ({
|
|||||||
removeClippedSubviews={false}
|
removeClippedSubviews={false}
|
||||||
getItemLayout={getItemLayout}
|
getItemLayout={getItemLayout}
|
||||||
style={{ overflow: "visible" }}
|
style={{ overflow: "visible" }}
|
||||||
// Edge padding via contentContainerStyle, NOT contentInset+contentOffset.
|
contentInset={{
|
||||||
// contentOffset only applies on initial mount; since this FlatList is
|
left: edgePadding,
|
||||||
// reused across searches (stable key), a second search left the inset
|
right: edgePadding,
|
||||||
// without the offset and the grid snapped flush to the left edge.
|
}}
|
||||||
|
contentOffset={{ x: -edgePadding, y: 0 }}
|
||||||
contentContainerStyle={{
|
contentContainerStyle={{
|
||||||
paddingHorizontal: edgePadding,
|
|
||||||
paddingVertical: SCALE_PADDING,
|
paddingVertical: SCALE_PADDING,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -31,12 +31,8 @@ export const SeasonEpisodesCarousel: React.FC<Props> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
const [api] = useAtom(apiAtom);
|
const [api] = useAtom(apiAtom);
|
||||||
const [user] = useAtom(userAtom);
|
const [user] = useAtom(userAtom);
|
||||||
const router = useRouter();
|
|
||||||
const isOffline = useOfflineMode();
|
const isOffline = useOfflineMode();
|
||||||
// Read the live (cached) downloads DB inside the query rather than the
|
const router = useRouter();
|
||||||
// provider's downloadedItems snapshot, so refetches after
|
|
||||||
// updateDownloadedItem() reflect the latest state instead of a stale
|
|
||||||
// refreshKey-gated snapshot. getAllDownloadedItems() is cached, so this stays cheap.
|
|
||||||
const { getDownloadedItems } = useDownload();
|
const { getDownloadedItems } = useDownload();
|
||||||
|
|
||||||
const scrollRef = useRef<HorizontalScrollRef>(null);
|
const scrollRef = useRef<HorizontalScrollRef>(null);
|
||||||
|
|||||||
@@ -229,7 +229,7 @@ export const LibraryOptionsSheet: React.FC<Props> = ({
|
|||||||
/>
|
/>
|
||||||
</OptionGroup>
|
</OptionGroup>
|
||||||
|
|
||||||
<OptionGroup title={t("library.options.options_title")}>
|
<OptionGroup title='Options'>
|
||||||
<ToggleItem
|
<ToggleItem
|
||||||
label={t("library.options.show_titles")}
|
label={t("library.options.show_titles")}
|
||||||
value={settings.showTitles}
|
value={settings.showTitles}
|
||||||
|
|||||||
@@ -196,10 +196,7 @@ export const OtherSettings: React.FC = () => {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<ListItem
|
<ListItem title={t("home.settings.other.max_auto_play_episode_count")}>
|
||||||
title={t("home.settings.other.max_auto_play_episode_count")}
|
|
||||||
disabled={pluginSettings?.maxAutoPlayEpisodeCount?.locked}
|
|
||||||
>
|
|
||||||
<PlatformDropdown
|
<PlatformDropdown
|
||||||
groups={autoPlayEpisodeOptions}
|
groups={autoPlayEpisodeOptions}
|
||||||
trigger={
|
trigger={
|
||||||
|
|||||||
@@ -229,10 +229,7 @@ export const PlaybackControlsSettings: React.FC = () => {
|
|||||||
|
|
||||||
<ListItem
|
<ListItem
|
||||||
title={t("home.settings.other.max_auto_play_episode_count")}
|
title={t("home.settings.other.max_auto_play_episode_count")}
|
||||||
disabled={
|
disabled={!settings.autoPlayNextEpisode}
|
||||||
!settings.autoPlayNextEpisode ||
|
|
||||||
pluginSettings?.maxAutoPlayEpisodeCount?.locked
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<PlatformDropdown
|
<PlatformDropdown
|
||||||
groups={autoPlayEpisodeOptions}
|
groups={autoPlayEpisodeOptions}
|
||||||
|
|||||||
@@ -105,14 +105,14 @@ const AudioSlider: React.FC<AudioSliderProps> = ({ setVisibility }) => {
|
|||||||
maximumValue={max}
|
maximumValue={max}
|
||||||
thumbWidth={0}
|
thumbWidth={0}
|
||||||
onValueChange={handleValueChange}
|
onValueChange={handleValueChange}
|
||||||
renderBubble={() => null}
|
|
||||||
renderThumb={() => null}
|
|
||||||
containerStyle={{
|
containerStyle={{
|
||||||
borderRadius: 50,
|
borderRadius: 50,
|
||||||
}}
|
}}
|
||||||
theme={{
|
theme={{
|
||||||
minimumTrackTintColor: "#FDFDFD",
|
minimumTrackTintColor: "#FDFDFD",
|
||||||
maximumTrackTintColor: "#5A5A5A",
|
maximumTrackTintColor: "#5A5A5A",
|
||||||
|
bubbleBackgroundColor: "transparent", // Hide the value bubble
|
||||||
|
bubbleTextColor: "transparent", // Hide the value text
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<Ionicons
|
<Ionicons
|
||||||
|
|||||||
@@ -8,10 +8,10 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { Pressable, View } from "react-native";
|
import { Pressable, View } from "react-native";
|
||||||
import { Slider } from "react-native-awesome-slider";
|
import { Slider } from "react-native-awesome-slider";
|
||||||
import { type SharedValue } from "react-native-reanimated";
|
import { type SharedValue } from "react-native-reanimated";
|
||||||
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { ChapterList } from "@/components/chapters/ChapterList";
|
import { ChapterList } from "@/components/chapters/ChapterList";
|
||||||
import { ChapterTicks } from "@/components/chapters/ChapterTicks";
|
import { ChapterTicks } from "@/components/chapters/ChapterTicks";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { useControlsSafeAreaInsets } from "@/hooks/useControlsSafeAreaInsets";
|
|
||||||
import { useSettings } from "@/utils/atoms/settings";
|
import { useSettings } from "@/utils/atoms/settings";
|
||||||
import { chapterMarkers, chapterNameAt } from "@/utils/chapters";
|
import { chapterMarkers, chapterNameAt } from "@/utils/chapters";
|
||||||
import NextEpisodeCountDownButton from "./NextEpisodeCountDownButton";
|
import NextEpisodeCountDownButton from "./NextEpisodeCountDownButton";
|
||||||
@@ -75,6 +75,9 @@ interface BottomControlsProps {
|
|||||||
minutes: number;
|
minutes: number;
|
||||||
seconds: number;
|
seconds: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Chapter props
|
||||||
|
chapterPositions?: number[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export const BottomControls: FC<BottomControlsProps> = ({
|
export const BottomControls: FC<BottomControlsProps> = ({
|
||||||
@@ -108,10 +111,11 @@ export const BottomControls: FC<BottomControlsProps> = ({
|
|||||||
trickPlayUrl,
|
trickPlayUrl,
|
||||||
trickplayInfo,
|
trickplayInfo,
|
||||||
time,
|
time,
|
||||||
|
chapterPositions = [],
|
||||||
}) => {
|
}) => {
|
||||||
const { settings } = useSettings();
|
const { settings } = useSettings();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const insets = useControlsSafeAreaInsets();
|
const insets = useSafeAreaInsets();
|
||||||
const [chapterListVisible, setChapterListVisible] = useState(false);
|
const [chapterListVisible, setChapterListVisible] = useState(false);
|
||||||
|
|
||||||
// Only expose chapter UI when there are at least two real markers.
|
// Only expose chapter UI when there are at least two real markers.
|
||||||
@@ -142,9 +146,13 @@ export const BottomControls: FC<BottomControlsProps> = ({
|
|||||||
style={[
|
style={[
|
||||||
{
|
{
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
right: insets.right,
|
right:
|
||||||
left: insets.left,
|
(settings?.safeAreaInControlsEnabled ?? true) ? insets.right : 0,
|
||||||
bottom: Math.max(insets.bottom - 17, 0),
|
left: (settings?.safeAreaInControlsEnabled ?? true) ? insets.left : 0,
|
||||||
|
bottom:
|
||||||
|
(settings?.safeAreaInControlsEnabled ?? true)
|
||||||
|
? Math.max(insets.bottom - 17, 0)
|
||||||
|
: 0,
|
||||||
},
|
},
|
||||||
]}
|
]}
|
||||||
className={"flex flex-col px-2"}
|
className={"flex flex-col px-2"}
|
||||||
@@ -180,6 +188,17 @@ export const BottomControls: FC<BottomControlsProps> = ({
|
|||||||
) : null}
|
) : null}
|
||||||
</View>
|
</View>
|
||||||
<View className='flex flex-row items-center space-x-2 shrink-0'>
|
<View className='flex flex-row items-center space-x-2 shrink-0'>
|
||||||
|
{hasChapters && (
|
||||||
|
<Pressable
|
||||||
|
onPress={() => setChapterListVisible(true)}
|
||||||
|
hitSlop={10}
|
||||||
|
className='justify-center mr-4'
|
||||||
|
accessibilityRole='button'
|
||||||
|
accessibilityLabel={t("chapters.open")}
|
||||||
|
>
|
||||||
|
<Ionicons name='bookmarks' size={24} color='white' />
|
||||||
|
</Pressable>
|
||||||
|
)}
|
||||||
<SkipButton
|
<SkipButton
|
||||||
showButton={showSkipButton}
|
showButton={showSkipButton}
|
||||||
onPress={skipIntro}
|
onPress={skipIntro}
|
||||||
@@ -211,17 +230,6 @@ export const BottomControls: FC<BottomControlsProps> = ({
|
|||||||
onPress={handleNextEpisodeManual}
|
onPress={handleNextEpisodeManual}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{hasChapters && (
|
|
||||||
<Pressable
|
|
||||||
onPress={() => setChapterListVisible(true)}
|
|
||||||
hitSlop={10}
|
|
||||||
className='justify-center ml-4'
|
|
||||||
accessibilityRole='button'
|
|
||||||
accessibilityLabel={t("chapters.open")}
|
|
||||||
>
|
|
||||||
<Ionicons name='bookmarks' size={24} color='white' />
|
|
||||||
</Pressable>
|
|
||||||
)}
|
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View
|
<View
|
||||||
|
|||||||
@@ -88,14 +88,14 @@ const BrightnessSlider = () => {
|
|||||||
maximumValue={max}
|
maximumValue={max}
|
||||||
thumbWidth={0}
|
thumbWidth={0}
|
||||||
onValueChange={handleValueChange}
|
onValueChange={handleValueChange}
|
||||||
renderBubble={() => null}
|
|
||||||
renderThumb={() => null}
|
|
||||||
containerStyle={{
|
containerStyle={{
|
||||||
borderRadius: 50,
|
borderRadius: 50,
|
||||||
}}
|
}}
|
||||||
theme={{
|
theme={{
|
||||||
minimumTrackTintColor: "#FDFDFD",
|
minimumTrackTintColor: "#FDFDFD",
|
||||||
maximumTrackTintColor: "#5A5A5A",
|
maximumTrackTintColor: "#5A5A5A",
|
||||||
|
bubbleBackgroundColor: "transparent", // Hide the value bubble
|
||||||
|
bubbleTextColor: "transparent", // Hide the value text
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<Ionicons
|
<Ionicons
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import { Ionicons } from "@expo/vector-icons";
|
import { Ionicons } from "@expo/vector-icons";
|
||||||
import type { FC } from "react";
|
import type { FC } from "react";
|
||||||
import { Platform, TouchableOpacity, View } from "react-native";
|
import { Platform, TouchableOpacity, View } from "react-native";
|
||||||
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { Loader } from "@/components/Loader";
|
import { Loader } from "@/components/Loader";
|
||||||
import { useControlsSafeAreaInsets } from "@/hooks/useControlsSafeAreaInsets";
|
|
||||||
import { useSettings } from "@/utils/atoms/settings";
|
import { useSettings } from "@/utils/atoms/settings";
|
||||||
import AudioSlider from "./AudioSlider";
|
import AudioSlider from "./AudioSlider";
|
||||||
import BrightnessSlider from "./BrightnessSlider";
|
import BrightnessSlider from "./BrightnessSlider";
|
||||||
@@ -42,15 +42,15 @@ export const CenterControls: FC<CenterControlsProps> = ({
|
|||||||
goToNextChapter,
|
goToNextChapter,
|
||||||
}) => {
|
}) => {
|
||||||
const { settings } = useSettings();
|
const { settings } = useSettings();
|
||||||
const insets = useControlsSafeAreaInsets();
|
const insets = useSafeAreaInsets();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View
|
<View
|
||||||
style={{
|
style={{
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
top: "50%",
|
top: "50%",
|
||||||
left: insets.left,
|
left: (settings?.safeAreaInControlsEnabled ?? true) ? insets.left : 0,
|
||||||
right: insets.right,
|
right: (settings?.safeAreaInControlsEnabled ?? true) ? insets.right : 0,
|
||||||
flexDirection: "row",
|
flexDirection: "row",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
|
|||||||
@@ -219,6 +219,7 @@ export const Controls: FC<Props> = ({
|
|||||||
hasNextChapter,
|
hasNextChapter,
|
||||||
goToPreviousChapter,
|
goToPreviousChapter,
|
||||||
goToNextChapter,
|
goToNextChapter,
|
||||||
|
chapterPositions,
|
||||||
} = useChapterNavigation({
|
} = useChapterNavigation({
|
||||||
chapters: item.Chapters,
|
chapters: item.Chapters,
|
||||||
progress,
|
progress,
|
||||||
@@ -365,9 +366,7 @@ export const Controls: FC<Props> = ({
|
|||||||
{ applyLanguagePreferences: true },
|
{ applyLanguagePreferences: true },
|
||||||
);
|
);
|
||||||
|
|
||||||
// Use setParams instead of replace to avoid unmounting/remounting the player,
|
const queryParams = new URLSearchParams({
|
||||||
// which would create a new MPV native view and crash with "mp_initialize already initialized".
|
|
||||||
router.setParams({
|
|
||||||
...(offline && { offline: "true" }),
|
...(offline && { offline: "true" }),
|
||||||
itemId: item.Id ?? "",
|
itemId: item.Id ?? "",
|
||||||
audioIndex: defaultAudioIndex?.toString() ?? "",
|
audioIndex: defaultAudioIndex?.toString() ?? "",
|
||||||
@@ -376,17 +375,11 @@ export const Controls: FC<Props> = ({
|
|||||||
bitrateValue: bitrateValue?.toString(),
|
bitrateValue: bitrateValue?.toString(),
|
||||||
playbackPosition:
|
playbackPosition:
|
||||||
item.UserData?.PlaybackPositionTicks?.toString() ?? "",
|
item.UserData?.PlaybackPositionTicks?.toString() ?? "",
|
||||||
});
|
}).toString();
|
||||||
|
|
||||||
|
router.replace(`player/direct-player?${queryParams}` as any);
|
||||||
},
|
},
|
||||||
[
|
[settings, subtitleIndex, audioIndex, mediaSource, bitrateValue, router],
|
||||||
settings,
|
|
||||||
subtitleIndex,
|
|
||||||
audioIndex,
|
|
||||||
mediaSource,
|
|
||||||
bitrateValue,
|
|
||||||
router,
|
|
||||||
offline,
|
|
||||||
],
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const goToPreviousItem = useCallback(() => {
|
const goToPreviousItem = useCallback(() => {
|
||||||
@@ -592,6 +585,7 @@ export const Controls: FC<Props> = ({
|
|||||||
trickPlayUrl={trickPlayUrl}
|
trickPlayUrl={trickPlayUrl}
|
||||||
trickplayInfo={trickplayInfo}
|
trickplayInfo={trickplayInfo}
|
||||||
time={isSliding || showRemoteBubble ? time : remoteTime}
|
time={isSliding || showRemoteBubble ? time : remoteTime}
|
||||||
|
chapterPositions={chapterPositions}
|
||||||
/>
|
/>
|
||||||
</Animated.View>
|
</Animated.View>
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -1254,7 +1254,7 @@ export const Controls: FC<Props> = ({
|
|||||||
<Text
|
<Text
|
||||||
style={[styles.endsAtText, { fontSize: typography.callout }]}
|
style={[styles.endsAtText, { fontSize: typography.callout }]}
|
||||||
>
|
>
|
||||||
{t("player.ends_at", { time: getFinishTime() })}
|
{t("player.ends_at")} {getFinishTime()}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
@@ -1448,7 +1448,7 @@ export const Controls: FC<Props> = ({
|
|||||||
<Text
|
<Text
|
||||||
style={[styles.endsAtText, { fontSize: typography.callout }]}
|
style={[styles.endsAtText, { fontSize: typography.callout }]}
|
||||||
>
|
>
|
||||||
{t("player.ends_at", { time: getFinishTime() })}
|
{t("player.ends_at")} {getFinishTime()}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { useQuery, useQueryClient } from "@tanstack/react-query";
|
|||||||
import { atom, useAtom } from "jotai";
|
import { atom, useAtom } from "jotai";
|
||||||
import { useEffect, useMemo, useRef } from "react";
|
import { useEffect, useMemo, useRef } from "react";
|
||||||
import { TouchableOpacity, View } from "react-native";
|
import { TouchableOpacity, View } from "react-native";
|
||||||
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import ContinueWatchingPoster from "@/components/ContinueWatchingPoster";
|
import ContinueWatchingPoster from "@/components/ContinueWatchingPoster";
|
||||||
import {
|
import {
|
||||||
HorizontalScroll,
|
HorizontalScroll,
|
||||||
@@ -16,10 +17,10 @@ import {
|
|||||||
SeasonDropdown,
|
SeasonDropdown,
|
||||||
type SeasonIndexState,
|
type SeasonIndexState,
|
||||||
} from "@/components/series/SeasonDropdown";
|
} from "@/components/series/SeasonDropdown";
|
||||||
import { useControlsSafeAreaInsets } from "@/hooks/useControlsSafeAreaInsets";
|
|
||||||
import { useDownload } from "@/providers/DownloadProvider";
|
import { useDownload } from "@/providers/DownloadProvider";
|
||||||
import { apiAtom, userAtom } from "@/providers/JellyfinProvider";
|
import { apiAtom, userAtom } from "@/providers/JellyfinProvider";
|
||||||
import { useOfflineMode } from "@/providers/OfflineModeProvider";
|
import { useOfflineMode } from "@/providers/OfflineModeProvider";
|
||||||
|
import { useSettings } from "@/utils/atoms/settings";
|
||||||
import {
|
import {
|
||||||
getDownloadedEpisodesForSeason,
|
getDownloadedEpisodesForSeason,
|
||||||
getDownloadedSeasonNumbers,
|
getDownloadedSeasonNumbers,
|
||||||
@@ -45,7 +46,8 @@ export const EpisodeList: React.FC<Props> = ({ item, close, goToItem }) => {
|
|||||||
scrollViewRef.current?.scrollToIndex(index, 100);
|
scrollViewRef.current?.scrollToIndex(index, 100);
|
||||||
};
|
};
|
||||||
const isOffline = useOfflineMode();
|
const isOffline = useOfflineMode();
|
||||||
const insets = useControlsSafeAreaInsets();
|
const { settings } = useSettings();
|
||||||
|
const insets = useSafeAreaInsets();
|
||||||
|
|
||||||
// Set the initial season index
|
// Set the initial season index
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -57,11 +59,6 @@ export const EpisodeList: React.FC<Props> = ({ item, close, goToItem }) => {
|
|||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
// Read the live (cached) downloads DB inside the query rather than the
|
|
||||||
// provider's downloadedItems snapshot. The snapshot only refreshes on the
|
|
||||||
// provider refreshKey, so after updateDownloadedItem() invalidates
|
|
||||||
// ["episodes"]/["seasons"] (e.g. progress/played writes) the refetch would
|
|
||||||
// return stale data. getAllDownloadedItems() is cached, so this stays cheap.
|
|
||||||
const { getDownloadedItems } = useDownload();
|
const { getDownloadedItems } = useDownload();
|
||||||
|
|
||||||
const seasonIndex = seasonIndexState[item.ParentId ?? ""];
|
const seasonIndex = seasonIndexState[item.ParentId ?? ""];
|
||||||
@@ -185,9 +182,12 @@ export const EpisodeList: React.FC<Props> = ({ item, close, goToItem }) => {
|
|||||||
backgroundColor: "black",
|
backgroundColor: "black",
|
||||||
height: "100%",
|
height: "100%",
|
||||||
width: "100%",
|
width: "100%",
|
||||||
paddingTop: insets.top,
|
paddingTop:
|
||||||
paddingLeft: insets.left,
|
(settings?.safeAreaInControlsEnabled ?? true) ? insets.top : 0,
|
||||||
paddingRight: insets.right,
|
paddingLeft:
|
||||||
|
(settings?.safeAreaInControlsEnabled ?? true) ? insets.left : 0,
|
||||||
|
paddingRight:
|
||||||
|
(settings?.safeAreaInControlsEnabled ?? true) ? insets.right : 0,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<View
|
<View
|
||||||
|
|||||||
@@ -5,11 +5,12 @@ import type {
|
|||||||
} from "@jellyfin/sdk/lib/generated-client";
|
} from "@jellyfin/sdk/lib/generated-client";
|
||||||
import { type FC, useCallback, useState } from "react";
|
import { type FC, useCallback, useState } from "react";
|
||||||
import { Platform, TouchableOpacity, View } from "react-native";
|
import { Platform, TouchableOpacity, View } from "react-native";
|
||||||
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import useRouter from "@/hooks/useAppRouter";
|
import useRouter from "@/hooks/useAppRouter";
|
||||||
import { useControlsSafeAreaInsets } from "@/hooks/useControlsSafeAreaInsets";
|
|
||||||
import { useHaptic } from "@/hooks/useHaptic";
|
import { useHaptic } from "@/hooks/useHaptic";
|
||||||
import { useOrientation } from "@/hooks/useOrientation";
|
import { useOrientation } from "@/hooks/useOrientation";
|
||||||
import { OrientationLock } from "@/packages/expo-screen-orientation";
|
import { OrientationLock } from "@/packages/expo-screen-orientation";
|
||||||
|
import { useSettings } from "@/utils/atoms/settings";
|
||||||
import { HEADER_LAYOUT, ICON_SIZES } from "./constants";
|
import { HEADER_LAYOUT, ICON_SIZES } from "./constants";
|
||||||
import DropdownView from "./dropdown/DropdownView";
|
import DropdownView from "./dropdown/DropdownView";
|
||||||
import { PlaybackSpeedScope } from "./utils/playback-speed-settings";
|
import { PlaybackSpeedScope } from "./utils/playback-speed-settings";
|
||||||
@@ -57,8 +58,9 @@ export const HeaderControls: FC<HeaderControlsProps> = ({
|
|||||||
showTechnicalInfo = false,
|
showTechnicalInfo = false,
|
||||||
onToggleTechnicalInfo,
|
onToggleTechnicalInfo,
|
||||||
}) => {
|
}) => {
|
||||||
|
const { settings } = useSettings();
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const insets = useControlsSafeAreaInsets();
|
const insets = useSafeAreaInsets();
|
||||||
const lightHapticFeedback = useHaptic("light");
|
const lightHapticFeedback = useHaptic("light");
|
||||||
const { orientation, lockOrientation } = useOrientation();
|
const { orientation, lockOrientation } = useOrientation();
|
||||||
const [isTogglingOrientation, setIsTogglingOrientation] = useState(false);
|
const [isTogglingOrientation, setIsTogglingOrientation] = useState(false);
|
||||||
@@ -97,9 +99,10 @@ export const HeaderControls: FC<HeaderControlsProps> = ({
|
|||||||
style={[
|
style={[
|
||||||
{
|
{
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
top: insets.top,
|
top: (settings?.safeAreaInControlsEnabled ?? true) ? insets.top : 0,
|
||||||
left: insets.left,
|
left: (settings?.safeAreaInControlsEnabled ?? true) ? insets.left : 0,
|
||||||
right: insets.right,
|
right:
|
||||||
|
(settings?.safeAreaInControlsEnabled ?? true) ? insets.right : 0,
|
||||||
padding: HEADER_LAYOUT.CONTAINER_PADDING,
|
padding: HEADER_LAYOUT.CONTAINER_PADDING,
|
||||||
},
|
},
|
||||||
]}
|
]}
|
||||||
|
|||||||
@@ -16,8 +16,8 @@ import Animated, {
|
|||||||
} from "react-native-reanimated";
|
} from "react-native-reanimated";
|
||||||
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { useScaledTVTypography } from "@/constants/TVTypography";
|
import { useScaledTVTypography } from "@/constants/TVTypography";
|
||||||
import { useControlsSafeAreaInsets } from "@/hooks/useControlsSafeAreaInsets";
|
|
||||||
import type { TechnicalInfo } from "@/modules/mpv-player";
|
import type { TechnicalInfo } from "@/modules/mpv-player";
|
||||||
|
import { useSettings } from "@/utils/atoms/settings";
|
||||||
import { HEADER_LAYOUT } from "./constants";
|
import { HEADER_LAYOUT } from "./constants";
|
||||||
|
|
||||||
type PlayMethod = "DirectPlay" | "DirectStream" | "Transcode";
|
type PlayMethod = "DirectPlay" | "DirectStream" | "Transcode";
|
||||||
@@ -184,8 +184,8 @@ export const TechnicalInfoOverlay: FC<TechnicalInfoOverlayProps> = memo(
|
|||||||
currentAudioIndex,
|
currentAudioIndex,
|
||||||
}) => {
|
}) => {
|
||||||
const typography = useScaledTVTypography();
|
const typography = useScaledTVTypography();
|
||||||
|
const { settings } = useSettings();
|
||||||
const insets = useSafeAreaInsets();
|
const insets = useSafeAreaInsets();
|
||||||
const safeInsets = useControlsSafeAreaInsets();
|
|
||||||
const [info, setInfo] = useState<TechnicalInfo | null>(null);
|
const [info, setInfo] = useState<TechnicalInfo | null>(null);
|
||||||
|
|
||||||
const opacity = useSharedValue(0);
|
const opacity = useSharedValue(0);
|
||||||
@@ -268,8 +268,14 @@ export const TechnicalInfoOverlay: FC<TechnicalInfoOverlayProps> = memo(
|
|||||||
left: Math.max(insets.left, 48) + 20,
|
left: Math.max(insets.left, 48) + 20,
|
||||||
}
|
}
|
||||||
: {
|
: {
|
||||||
top: safeInsets.top + HEADER_LAYOUT.CONTAINER_PADDING + 4,
|
top:
|
||||||
left: safeInsets.left + HEADER_LAYOUT.CONTAINER_PADDING + 20,
|
(settings?.safeAreaInControlsEnabled ?? true)
|
||||||
|
? insets.top + HEADER_LAYOUT.CONTAINER_PADDING + 4
|
||||||
|
: HEADER_LAYOUT.CONTAINER_PADDING + 4,
|
||||||
|
left:
|
||||||
|
(settings?.safeAreaInControlsEnabled ?? true)
|
||||||
|
? insets.left + HEADER_LAYOUT.CONTAINER_PADDING + 20
|
||||||
|
: HEADER_LAYOUT.CONTAINER_PADDING + 20,
|
||||||
};
|
};
|
||||||
|
|
||||||
const textStyle = Platform.isTV
|
const textStyle = Platform.isTV
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import type { FC } from "react";
|
import type { FC } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { View } from "react-native";
|
import { View } from "react-native";
|
||||||
import { Text } from "@/components/common/Text";
|
import { Text } from "@/components/common/Text";
|
||||||
import { formatTimeString } from "@/utils/time";
|
import { formatTimeString } from "@/utils/time";
|
||||||
@@ -17,8 +16,6 @@ export const TimeDisplay: FC<TimeDisplayProps> = ({
|
|||||||
currentTime,
|
currentTime,
|
||||||
remainingTime,
|
remainingTime,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
|
||||||
|
|
||||||
const getFinishTime = () => {
|
const getFinishTime = () => {
|
||||||
const now = new Date();
|
const now = new Date();
|
||||||
// remainingTime is in ms
|
// remainingTime is in ms
|
||||||
@@ -40,7 +37,7 @@ export const TimeDisplay: FC<TimeDisplayProps> = ({
|
|||||||
-{formatTimeString(remainingTime, "ms")}
|
-{formatTimeString(remainingTime, "ms")}
|
||||||
</Text>
|
</Text>
|
||||||
<Text className='text-[10px] text-neutral-500 opacity-70'>
|
<Text className='text-[10px] text-neutral-500 opacity-70'>
|
||||||
{t("player.ends_at", { time: getFinishTime() })}
|
ends at {getFinishTime()}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
49
eas.json
@@ -1,7 +1,6 @@
|
|||||||
{
|
{
|
||||||
"cli": {
|
"cli": {
|
||||||
"version": ">= 16.0.0",
|
"version": ">= 9.1.0"
|
||||||
"appVersionSource": "remote"
|
|
||||||
},
|
},
|
||||||
"build": {
|
"build": {
|
||||||
"development": {
|
"development": {
|
||||||
@@ -52,70 +51,44 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"production": {
|
"production": {
|
||||||
"bun": "1.3.5",
|
|
||||||
"environment": "production",
|
"environment": "production",
|
||||||
"autoIncrement": true,
|
"channel": "0.54.0",
|
||||||
"android": {
|
"android": {
|
||||||
"image": "latest",
|
"image": "latest"
|
||||||
"config": "android-production.yml"
|
|
||||||
},
|
|
||||||
"ios": {
|
|
||||||
"config": "ios-production.yml"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"production-apk": {
|
"production-apk": {
|
||||||
"bun": "1.3.5",
|
|
||||||
"environment": "production",
|
"environment": "production",
|
||||||
"autoIncrement": true,
|
"channel": "0.54.0",
|
||||||
"android": {
|
"android": {
|
||||||
"buildType": "apk",
|
"buildType": "apk",
|
||||||
"image": "latest",
|
"image": "latest"
|
||||||
"config": "android-production-apk.yml"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"production-apk-tv": {
|
"production-apk-tv": {
|
||||||
"bun": "1.3.5",
|
|
||||||
"environment": "production",
|
"environment": "production",
|
||||||
"autoIncrement": true,
|
"channel": "0.54.0",
|
||||||
"android": {
|
"android": {
|
||||||
"buildType": "apk",
|
"buildType": "apk",
|
||||||
"image": "latest",
|
"image": "latest"
|
||||||
"config": "android-production-tv.yml"
|
|
||||||
},
|
},
|
||||||
"env": {
|
"env": {
|
||||||
"EXPO_TV": "1"
|
"EXPO_TV": "1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"production_tv": {
|
"production_tv": {
|
||||||
"bun": "1.3.5",
|
|
||||||
"environment": "production",
|
"environment": "production",
|
||||||
"autoIncrement": true,
|
"channel": "0.54.0",
|
||||||
"env": {
|
"env": {
|
||||||
"EXPO_TV": "1"
|
"EXPO_TV": "1"
|
||||||
},
|
},
|
||||||
"ios": {
|
"ios": {
|
||||||
"credentialsSource": "local",
|
"credentialsSource": "local"
|
||||||
"config": "ios-production.yml"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"submit": {
|
"submit": {
|
||||||
"production": {
|
"production": {},
|
||||||
"ios": {
|
"production_tv": {}
|
||||||
"appleTeamId": "MWD5K362T8",
|
|
||||||
"ascAppId": "6593660679"
|
|
||||||
},
|
|
||||||
"android": {
|
|
||||||
"serviceAccountKeyPath": "./google-service-account.json",
|
|
||||||
"track": "internal",
|
|
||||||
"releaseStatus": "completed"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"production_tv": {
|
|
||||||
"ios": {
|
|
||||||
"appleTeamId": "MWD5K362T8",
|
|
||||||
"ascAppId": "6593660679"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +0,0 @@
|
|||||||
import {
|
|
||||||
type EdgeInsets,
|
|
||||||
useSafeAreaInsets,
|
|
||||||
} from "react-native-safe-area-context";
|
|
||||||
import { useSettings } from "@/utils/atoms/settings";
|
|
||||||
|
|
||||||
const ZERO_INSETS: EdgeInsets = { top: 0, right: 0, bottom: 0, left: 0 };
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns safe-area insets to apply to in-player controls, honoring the
|
|
||||||
* `safeAreaInControlsEnabled` user setting. When the setting is disabled,
|
|
||||||
* returns zero insets so controls can sit flush against the screen edges.
|
|
||||||
*/
|
|
||||||
export const useControlsSafeAreaInsets = (): EdgeInsets => {
|
|
||||||
const { settings } = useSettings();
|
|
||||||
const insets = useSafeAreaInsets();
|
|
||||||
return settings.safeAreaInControlsEnabled ? insets : ZERO_INSETS;
|
|
||||||
};
|
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
import { File, Paths } from "expo-file-system";
|
|
||||||
import { useCallback } from "react";
|
import { useCallback } from "react";
|
||||||
import { storage } from "@/utils/mmkv";
|
import { storage } from "@/utils/mmkv";
|
||||||
|
|
||||||
@@ -13,28 +12,36 @@ const useImageStorage = () => {
|
|||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
/**
|
|
||||||
* expo-file-system instead of fetch+Blob+FileReader: the latter silently
|
|
||||||
* resolves to an empty payload under RN's New Architecture.
|
|
||||||
*/
|
|
||||||
const image2Base64 = useCallback(async (url?: string | null) => {
|
const image2Base64 = useCallback(async (url?: string | null) => {
|
||||||
if (!url) return null;
|
if (!url) return null;
|
||||||
|
|
||||||
const tmpFile = new File(
|
let blob: Blob;
|
||||||
Paths.cache,
|
|
||||||
`img-${Date.now()}-${Math.random().toString(36).slice(2)}.jpg`,
|
|
||||||
);
|
|
||||||
try {
|
try {
|
||||||
const downloaded = await File.downloadFileAsync(url, tmpFile, {
|
// Fetch the data from the URL
|
||||||
idempotent: true,
|
const response = await fetch(url);
|
||||||
});
|
blob = await response.blob();
|
||||||
return await downloaded.base64();
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.warn("Error fetching image:", error);
|
console.warn("Error fetching image:", error);
|
||||||
return null;
|
return null;
|
||||||
} finally {
|
|
||||||
if (tmpFile.exists) tmpFile.delete();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create a FileReader instance
|
||||||
|
const reader = new FileReader();
|
||||||
|
|
||||||
|
// Convert blob to base64
|
||||||
|
return new Promise<string>((resolve, reject) => {
|
||||||
|
reader.onloadend = () => {
|
||||||
|
if (typeof reader.result === "string") {
|
||||||
|
// Extract the base64 string (remove the data URL prefix)
|
||||||
|
const base64 = reader.result.split(",")[1];
|
||||||
|
resolve(base64);
|
||||||
|
} else {
|
||||||
|
reject(new Error("Failed to convert image to base64"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
reader.onerror = reject;
|
||||||
|
reader.readAsDataURL(blob);
|
||||||
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const saveImage = useCallback(
|
const saveImage = useCallback(
|
||||||
|
|||||||
@@ -109,35 +109,30 @@ export const usePlaybackManager = ({
|
|||||||
staleTime: 0,
|
staleTime: 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
|
||||||
* Derive prev/next from the current item's real position in the adjacent
|
|
||||||
* list rather than from the array length. `getEpisodes({ adjacentTo })` does
|
|
||||||
* not guarantee a fixed [prev, current, next] shape — at the first/last
|
|
||||||
* episode it can still return the current item as the first/last entry — so
|
|
||||||
* length-based indexing wrongly surfaces the current episode as "previous".
|
|
||||||
*/
|
|
||||||
const currentIndex = useMemo(
|
|
||||||
() => adjacentItems?.findIndex((e) => e.Id === item?.Id) ?? -1,
|
|
||||||
[adjacentItems, item],
|
|
||||||
);
|
|
||||||
|
|
||||||
/** A neighbour is only navigable if it has an actual media file (not a
|
|
||||||
* "Virtual"/missing episode placeholder, e.g. an absent Special). */
|
|
||||||
const isNavigable = (episode?: BaseItemDto | null): episode is BaseItemDto =>
|
|
||||||
!!episode && episode.Id !== item?.Id && episode.LocationType !== "Virtual";
|
|
||||||
|
|
||||||
const previousItem = useMemo(() => {
|
const previousItem = useMemo(() => {
|
||||||
if (!adjacentItems || currentIndex <= 0) return null;
|
if (!adjacentItems || adjacentItems.length <= 1) {
|
||||||
const candidate = adjacentItems[currentIndex - 1];
|
return null;
|
||||||
return isNavigable(candidate) ? candidate : null;
|
}
|
||||||
}, [adjacentItems, currentIndex, item]);
|
|
||||||
|
if (adjacentItems.length === 2) {
|
||||||
|
return adjacentItems[0].Id === item?.Id ? null : adjacentItems[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
return adjacentItems[0];
|
||||||
|
}, [adjacentItems, item]);
|
||||||
|
|
||||||
/** The next item in the series */
|
/** The next item in the series */
|
||||||
const nextItem = useMemo(() => {
|
const nextItem = useMemo(() => {
|
||||||
if (!adjacentItems || currentIndex < 0) return null;
|
if (!adjacentItems || adjacentItems.length <= 1) {
|
||||||
const candidate = adjacentItems[currentIndex + 1];
|
return null;
|
||||||
return isNavigable(candidate) ? candidate : null;
|
}
|
||||||
}, [adjacentItems, currentIndex, item]);
|
|
||||||
|
if (adjacentItems.length === 2) {
|
||||||
|
return adjacentItems[1].Id === item?.Id ? null : adjacentItems[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return adjacentItems[2];
|
||||||
|
}, [adjacentItems, item]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reports playback progress.
|
* Reports playback progress.
|
||||||
|
|||||||
@@ -1,20 +1,46 @@
|
|||||||
apply plugin: 'expo-module-gradle-plugin'
|
plugins {
|
||||||
|
id 'com.android.library'
|
||||||
|
id 'kotlin-android'
|
||||||
|
}
|
||||||
|
|
||||||
group = 'expo.modules.backgrounddownloader'
|
group = 'expo.modules.backgrounddownloader'
|
||||||
version = '1.0.0'
|
version = '1.0.0'
|
||||||
|
|
||||||
expoModule {
|
def expoModulesCorePlugin = new File(project(":expo-modules-core").projectDir.absolutePath, "ExpoModulesCorePlugin.gradle")
|
||||||
canBePublished false
|
def kotlinVersion = findProperty('android.kotlinVersion') ?: '1.9.25'
|
||||||
}
|
|
||||||
|
apply from: expoModulesCorePlugin
|
||||||
|
|
||||||
|
applyKotlinExpoModulesCorePlugin()
|
||||||
|
useDefaultAndroidSdkVersions()
|
||||||
|
useCoreDependencies()
|
||||||
|
useExpoPublishing()
|
||||||
|
|
||||||
android {
|
android {
|
||||||
namespace "expo.modules.backgrounddownloader"
|
namespace "expo.modules.backgrounddownloader"
|
||||||
defaultConfig {
|
|
||||||
versionCode 1
|
compileOptions {
|
||||||
versionName "1.0.0"
|
sourceCompatibility JavaVersion.VERSION_17
|
||||||
|
targetCompatibility JavaVersion.VERSION_17
|
||||||
|
}
|
||||||
|
|
||||||
|
kotlinOptions {
|
||||||
|
jvmTarget = "17"
|
||||||
|
}
|
||||||
|
|
||||||
|
lintOptions {
|
||||||
|
abortOnError false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
|
implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion"
|
||||||
implementation "com.squareup.okhttp3:okhttp:4.12.0"
|
implementation "com.squareup.okhttp3:okhttp:4.12.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tasks.withType(org.jetbrains.kotlin.gradle.tasks.KotlinCompile).configureEach {
|
||||||
|
kotlinOptions {
|
||||||
|
jvmTarget = "17"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -715,7 +715,9 @@ class MPVLayerRenderer(private val context: Context) : MPVLib.EventObserver {
|
|||||||
// dropped), so we (re)apply here for embedded and external alike.
|
// dropped), so we (re)apply here for embedded and external alike.
|
||||||
// This is what makes a carried-over subtitle show up on the next
|
// This is what makes a carried-over subtitle show up on the next
|
||||||
// episode without a manual re-selection.
|
// episode without a manual re-selection.
|
||||||
initialAudioId?.let { if (it > 0) setAudioTrack(it) }
|
if (initialAudioId != null && initialAudioId > 0) {
|
||||||
|
setAudioTrack(initialAudioId)
|
||||||
|
}
|
||||||
initialSubtitleId?.let { setSubtitleTrack(it) } ?: disableSubtitles()
|
initialSubtitleId?.let { setSubtitleTrack(it) } ?: disableSubtitles()
|
||||||
|
|
||||||
if (!isReadyToSeek) {
|
if (!isReadyToSeek) {
|
||||||
|
|||||||
@@ -213,7 +213,7 @@ public class MpvPlayerModule: Module {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Defines events that the view can send to JavaScript
|
// Defines events that the view can send to JavaScript
|
||||||
Events("onLoad", "onPlaybackStateChange", "onProgress", "onError", "onTracksReady", "onPictureInPictureChange")
|
Events("onLoad", "onPlaybackStateChange", "onProgress", "onError", "onTracksReady")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -61,7 +61,6 @@ class MpvPlayerView: ExpoView {
|
|||||||
let onProgress = EventDispatcher()
|
let onProgress = EventDispatcher()
|
||||||
let onError = EventDispatcher()
|
let onError = EventDispatcher()
|
||||||
let onTracksReady = EventDispatcher()
|
let onTracksReady = EventDispatcher()
|
||||||
let onPictureInPictureChange = EventDispatcher()
|
|
||||||
|
|
||||||
private var currentURL: URL?
|
private var currentURL: URL?
|
||||||
private var cachedPosition: Double = 0
|
private var cachedPosition: Double = 0
|
||||||
@@ -82,6 +81,7 @@ class MpvPlayerView: ExpoView {
|
|||||||
private func setupView() {
|
private func setupView() {
|
||||||
clipsToBounds = true
|
clipsToBounds = true
|
||||||
backgroundColor = .black
|
backgroundColor = .black
|
||||||
|
configureAudioSession()
|
||||||
|
|
||||||
videoContainer = UIView()
|
videoContainer = UIView()
|
||||||
videoContainer.translatesAutoresizingMaskIntoConstraints = false
|
videoContainer.translatesAutoresizingMaskIntoConstraints = false
|
||||||
@@ -141,26 +141,21 @@ class MpvPlayerView: ExpoView {
|
|||||||
CATransaction.commit()
|
CATransaction.commit()
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Audio Session & Notifications
|
|
||||||
|
|
||||||
private func configureAudioSession() {
|
private func configureAudioSession() {
|
||||||
let session = AVAudioSession.sharedInstance()
|
let audioSession = AVAudioSession.sharedInstance()
|
||||||
do {
|
do {
|
||||||
try session.setCategory(.playback, mode: .moviePlayback, policy: .longFormAudio, options: [])
|
try audioSession.setCategory(
|
||||||
try session.setActive(true)
|
.playback,
|
||||||
|
mode: .moviePlayback,
|
||||||
|
policy: .longFormAudio,
|
||||||
|
options: []
|
||||||
|
)
|
||||||
|
try audioSession.setActive(true)
|
||||||
} catch {
|
} catch {
|
||||||
print("Failed to configure audio session: \(error)")
|
print("Failed to configure audio session: \(error)")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// MARK: - Audio Session & Notifications
|
||||||
/// Deactivate the session AND reset the category — `setActive(false)` alone
|
|
||||||
/// leaves `.playback`/`.longFormAudio` on the shared singleton, so any later
|
|
||||||
/// reactivation (foreground, route change, other modules) re-steals audio.
|
|
||||||
private func tearDownAudioSession() {
|
|
||||||
let session = AVAudioSession.sharedInstance()
|
|
||||||
try? session.setActive(false, options: .notifyOthersOnDeactivation)
|
|
||||||
try? session.setCategory(.ambient, mode: .default, options: [.mixWithOthers])
|
|
||||||
}
|
|
||||||
|
|
||||||
private func setupNotifications() {
|
private func setupNotifications() {
|
||||||
// Handle audio session interruptions (e.g., incoming calls, other apps playing audio)
|
// Handle audio session interruptions (e.g., incoming calls, other apps playing audio)
|
||||||
@@ -275,7 +270,6 @@ class MpvPlayerView: ExpoView {
|
|||||||
|
|
||||||
func play() {
|
func play() {
|
||||||
intendedPlayState = true
|
intendedPlayState = true
|
||||||
configureAudioSession()
|
|
||||||
setupRemoteCommands()
|
setupRemoteCommands()
|
||||||
renderer?.play()
|
renderer?.play()
|
||||||
pipController?.setPlaybackRate(1.0)
|
pipController?.setPlaybackRate(1.0)
|
||||||
@@ -446,7 +440,6 @@ class MpvPlayerView: ExpoView {
|
|||||||
renderer?.stop()
|
renderer?.stop()
|
||||||
displayLayer.removeFromSuperlayer()
|
displayLayer.removeFromSuperlayer()
|
||||||
clearNowPlayingInfo()
|
clearNowPlayingInfo()
|
||||||
tearDownAudioSession()
|
|
||||||
NotificationCenter.default.removeObserver(self)
|
NotificationCenter.default.removeObserver(self)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -526,7 +519,9 @@ extension MpvPlayerView: MPVLayerRendererDelegate {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func renderer(_: MPVLayerRenderer, didSelectAudioOutput audioOutput: String) {
|
func renderer(_: MPVLayerRenderer, didSelectAudioOutput audioOutput: String) {
|
||||||
print("[MPV] Audio output ready (\(audioOutput)), syncing Now Playing")
|
// Audio output is now active - this is the right time to activate audio session and set Now Playing
|
||||||
|
print("[MPV] Audio output ready (\(audioOutput)), activating audio session and syncing Now Playing")
|
||||||
|
nowPlayingManager.activateAudioSession()
|
||||||
syncNowPlaying(isPlaying: !isPaused())
|
syncNowPlaying(isPlaying: !isPaused())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -638,9 +633,6 @@ extension MpvPlayerView: PiPControllerDelegate {
|
|||||||
print("PiP did start: \(didStartPictureInPicture)")
|
print("PiP did start: \(didStartPictureInPicture)")
|
||||||
// Ensure current time is synced when PiP starts
|
// Ensure current time is synced when PiP starts
|
||||||
pipController?.setCurrentTimeFromSeconds(cachedPosition, duration: cachedDuration)
|
pipController?.setCurrentTimeFromSeconds(cachedPosition, duration: cachedDuration)
|
||||||
// Notify JS of the actual PiP active state. `didStartPictureInPicture`
|
|
||||||
// is `false` when AVKit reports a failure to start, so reflect that.
|
|
||||||
onPictureInPictureChange(["isActive": didStartPictureInPicture])
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func pipController(_ controller: PiPController, willStopPictureInPicture: Bool) {
|
func pipController(_ controller: PiPController, willStopPictureInPicture: Bool) {
|
||||||
@@ -659,9 +651,6 @@ extension MpvPlayerView: PiPControllerDelegate {
|
|||||||
if _isZoomedToFill {
|
if _isZoomedToFill {
|
||||||
displayLayer.videoGravity = .resizeAspectFill
|
displayLayer.videoGravity = .resizeAspectFill
|
||||||
}
|
}
|
||||||
// Notify JS that PiP has fully stopped so the controls overlay can
|
|
||||||
// be re-mounted when the user returns to full screen.
|
|
||||||
onPictureInPictureChange(["isActive": false])
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func pipController(_ controller: PiPController, restoreUserInterfaceForPictureInPictureStop completionHandler: @escaping (Bool) -> Void) {
|
func pipController(_ controller: PiPController, restoreUserInterfaceForPictureInPictureStop completionHandler: @escaping (Bool) -> Void) {
|
||||||
|
|||||||
@@ -1,13 +1,11 @@
|
|||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { MpvPlayerViewProps } from "./MpvPlayer.types";
|
import { MpvPlayerViewProps } from "./MpvPlayer.types";
|
||||||
|
|
||||||
export default function MpvPlayerView(props: MpvPlayerViewProps) {
|
export default function MpvPlayerView(props: MpvPlayerViewProps) {
|
||||||
const url = props.source?.url ?? "";
|
const url = props.source?.url ?? "";
|
||||||
const { t } = useTranslation();
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<iframe
|
<iframe
|
||||||
title={t("player.mpv_player_title")}
|
title='MPV Player'
|
||||||
style={{ flex: 1 }}
|
style={{ flex: 1 }}
|
||||||
src={url}
|
src={url}
|
||||||
onLoad={() => props.onLoad?.({ nativeEvent: { url } })}
|
onLoad={() => props.onLoad?.({ nativeEvent: { url } })}
|
||||||
|
|||||||
@@ -1,13 +1,10 @@
|
|||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
<uses-permission android:name="com.android.providers.tv.permission.WRITE_EPG_DATA" />
|
|
||||||
<application>
|
<application>
|
||||||
<receiver android:name=".TvRecommendationsReceiver" android:exported="true">
|
<receiver
|
||||||
|
android:name=".TvRecommendationsReceiver"
|
||||||
|
android:exported="true">
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
<action android:name="android.media.tv.action.INITIALIZE_PROGRAMS" />
|
<action android:name="android.media.tv.action.INITIALIZE_PROGRAMS" />
|
||||||
<category android:name="android.intent.category.DEFAULT" />
|
|
||||||
</intent-filter>
|
|
||||||
<intent-filter>
|
|
||||||
<action android:name="android.media.tv.action.PREVIEW_PROGRAM_BROWSABLE_DISABLED" />
|
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
</receiver>
|
</receiver>
|
||||||
</application>
|
</application>
|
||||||
|
|||||||
@@ -61,61 +61,31 @@ internal object TvRecommendationsPublisher {
|
|||||||
|
|
||||||
fun clear(context: Context): Boolean {
|
fun clear(context: Context): Boolean {
|
||||||
val prefs = preferences(context)
|
val prefs = preferences(context)
|
||||||
|
val channelId = prefs.getLong(KEY_CHANNEL_ID, -1L)
|
||||||
|
val programIds = prefs.getString(KEY_PROGRAM_IDS, null)?.let(::JSONObject)
|
||||||
val contentResolver = context.contentResolver
|
val contentResolver = context.contentResolver
|
||||||
|
|
||||||
// KEY_PROGRAM_IDS is now { channelId: "{ providerId: programId }" }
|
if (programIds != null) {
|
||||||
val allProgramIds = prefs.getString(KEY_PROGRAM_IDS, null)?.let(::JSONObject)
|
|
||||||
if (allProgramIds != null) {
|
|
||||||
var deletedPrograms = 0
|
var deletedPrograms = 0
|
||||||
val channelKeys = allProgramIds.keys()
|
val keys = programIds.keys()
|
||||||
while (channelKeys.hasNext()) {
|
while (keys.hasNext()) {
|
||||||
val channelIdStr = channelKeys.next()
|
val key = keys.next()
|
||||||
val programIdsJson = allProgramIds.optString(channelIdStr)
|
val programId = programIds.optLong(key, -1L)
|
||||||
if (programIdsJson.isBlank()) continue
|
if (programId > 0L) {
|
||||||
|
contentResolver.delete(
|
||||||
try {
|
TvContractCompat.buildPreviewProgramUri(programId),
|
||||||
val programIds = JSONObject(programIdsJson)
|
null,
|
||||||
val keys = programIds.keys()
|
null
|
||||||
while (keys.hasNext()) {
|
)
|
||||||
val providerId = keys.next()
|
deletedPrograms += 1
|
||||||
val programId = programIds.optLong(providerId, -1L)
|
|
||||||
if (programId > 0L) {
|
|
||||||
deletePreviewProgram(contentResolver, programId)
|
|
||||||
deletedPrograms += 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.w(TAG, "clear(): failed to parse programIds for channel $channelIdStr", e)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Notify the channel
|
|
||||||
val channelId = channelIdStr.toLongOrNull() ?: -1L
|
|
||||||
if (channelId > 0L) {
|
|
||||||
try {
|
|
||||||
contentResolver.notifyChange(TvContractCompat.buildChannelUri(channelId), null)
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "clear(): lost provider permission, cannot notify channel $channelId", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove per-channel pref
|
|
||||||
prefs.edit().remove("programIds_$channelIdStr").apply()
|
|
||||||
}
|
}
|
||||||
Log.d(TAG, "clear(): deleted $deletedPrograms preview program(s)")
|
Log.d(TAG, "clear(): deleted $deletedPrograms preview program(s)")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Also handle legacy format (flat { providerId: programId }) for migration
|
if (channelId > 0L) {
|
||||||
val legacyProgramIds = prefs.getString("legacy_" + KEY_PROGRAM_IDS, null)?.let(::JSONObject)
|
contentResolver.notifyChange(TvContractCompat.buildChannelUri(channelId), null)
|
||||||
if (legacyProgramIds != null) {
|
Log.d(TAG, "clear(): notified channel $channelId")
|
||||||
val keys = legacyProgramIds.keys()
|
|
||||||
while (keys.hasNext()) {
|
|
||||||
val key = keys.next()
|
|
||||||
val programId = legacyProgramIds.optLong(key, -1L)
|
|
||||||
if (programId > 0L) {
|
|
||||||
deletePreviewProgram(contentResolver, programId)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
prefs.edit().remove("legacy_" + KEY_PROGRAM_IDS).apply()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
prefs.edit()
|
prefs.edit()
|
||||||
@@ -126,262 +96,126 @@ internal object TvRecommendationsPublisher {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Delete a single preview program from the TvProvider.
|
|
||||||
* Called by clear(), synchronize() (stale programs), and TvRecommendationsReceiver (user removed).
|
|
||||||
*/
|
|
||||||
fun deletePreviewProgram(context: Context, programId: Long) {
|
|
||||||
try {
|
|
||||||
context.contentResolver.delete(
|
|
||||||
TvContractCompat.buildPreviewProgramUri(programId),
|
|
||||||
null,
|
|
||||||
null
|
|
||||||
)
|
|
||||||
Log.d(TAG, "deletePreviewProgram(): deleted programId=$programId")
|
|
||||||
|
|
||||||
// Also remove from stored programIds prefs
|
|
||||||
removeProgramFromPrefs(context, programId)
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "deletePreviewProgram(): lost provider permission for programId=$programId", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun deletePreviewProgram(contentResolver: android.content.ContentResolver, programId: Long) {
|
|
||||||
try {
|
|
||||||
contentResolver.delete(
|
|
||||||
TvContractCompat.buildPreviewProgramUri(programId),
|
|
||||||
null,
|
|
||||||
null
|
|
||||||
)
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "deletePreviewProgram(): lost provider permission for programId=$programId", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun removeProgramFromPrefs(context: Context, programId: Long) {
|
|
||||||
val prefs = preferences(context)
|
|
||||||
val programIdsJson = prefs.getString(KEY_PROGRAM_IDS, null) ?: return
|
|
||||||
try {
|
|
||||||
val programIds = JSONObject(programIdsJson)
|
|
||||||
val keys = programIds.keys()
|
|
||||||
while (keys.hasNext()) {
|
|
||||||
val key = keys.next()
|
|
||||||
if (programIds.optLong(key, -1L) == programId) {
|
|
||||||
programIds.remove(key)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
prefs.edit().putString(KEY_PROGRAM_IDS, programIds.toString()).apply()
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.w(TAG, "removeProgramFromPrefs(): failed to update prefs for programId=$programId", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun synchronize(context: Context, payload: JSONObject): Boolean {
|
private fun synchronize(context: Context, payload: JSONObject): Boolean {
|
||||||
val sections = payload.optJSONArray("sections") ?: JSONArray()
|
val sections = payload.optJSONArray("sections") ?: JSONArray()
|
||||||
if (sections.length() == 0) {
|
val firstSection = if (sections.length() > 0) sections.optJSONObject(0) else null
|
||||||
Log.w(TAG, "synchronize(): no sections in payload")
|
val sectionTitle = firstSection?.optString("title")?.takeIf { it.isNotBlank() } ?: DEFAULT_CHANNEL_NAME
|
||||||
return false
|
val items = firstSection?.optJSONArray("items") ?: JSONArray()
|
||||||
}
|
|
||||||
|
|
||||||
val prefs = preferences(context)
|
|
||||||
val allNextProgramIds = JSONObject()
|
|
||||||
var totalActive = 0
|
|
||||||
var totalDeleted = 0
|
|
||||||
|
|
||||||
for (sectionIndex in 0 until sections.length()) {
|
|
||||||
val section = sections.optJSONObject(sectionIndex) ?: continue
|
|
||||||
val sectionTitle = section.optString("title")?.takeIf { it.isNotBlank() } ?: DEFAULT_CHANNEL_NAME
|
|
||||||
val items = section.optJSONArray("items") ?: JSONArray()
|
|
||||||
|
|
||||||
Log.d(
|
|
||||||
TAG,
|
|
||||||
"synchronize(): section \"$sectionTitle\" ($sectionIndex/${sections.length()}) with ${items.length()} item(s)"
|
|
||||||
)
|
|
||||||
|
|
||||||
val channelId = getOrCreateChannel(context, sectionTitle)
|
|
||||||
if (channelId <= 0L) {
|
|
||||||
Log.w(TAG, "synchronize(): failed to get or create channel for \"$sectionTitle\"")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// Per Android docs: check channel.isBrowsable() and request if needed.
|
|
||||||
if (!isChannelBrowsable(context, channelId)) {
|
|
||||||
Log.d(TAG, "synchronize(): channel $channelId not browsable, requesting browsable")
|
|
||||||
TvContractCompat.requestChannelBrowsable(context, channelId)
|
|
||||||
}
|
|
||||||
|
|
||||||
val prefKey = "programIds_$channelId"
|
|
||||||
val previousProgramIds = prefs.getString(prefKey, null)
|
|
||||||
?.let(::JSONObject)
|
|
||||||
?: JSONObject()
|
|
||||||
val nextProgramIds = JSONObject()
|
|
||||||
val activeProviderIds = mutableSetOf<String>()
|
|
||||||
|
|
||||||
for (index in 0 until items.length()) {
|
|
||||||
val item = items.optJSONObject(index) ?: continue
|
|
||||||
val providerId = item.optString("id")
|
|
||||||
if (providerId.isBlank()) continue
|
|
||||||
|
|
||||||
val programId = upsertPreviewProgram(
|
|
||||||
context = context,
|
|
||||||
channelId = channelId,
|
|
||||||
item = item,
|
|
||||||
previousProgramId = previousProgramIds.optLong(providerId, -1L),
|
|
||||||
weight = index
|
|
||||||
)
|
|
||||||
|
|
||||||
if (programId > 0L) {
|
|
||||||
activeProviderIds += providerId
|
|
||||||
nextProgramIds.put(providerId, programId)
|
|
||||||
Log.d(TAG, "synchronize(): upserted program for item=$providerId programId=$programId")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var deletedPrograms = 0
|
|
||||||
val previousKeys = previousProgramIds.keys()
|
|
||||||
while (previousKeys.hasNext()) {
|
|
||||||
val providerId = previousKeys.next()
|
|
||||||
if (activeProviderIds.contains(providerId)) continue
|
|
||||||
|
|
||||||
val programId = previousProgramIds.optLong(providerId, -1L)
|
|
||||||
if (programId > 0L) {
|
|
||||||
deletePreviewProgram(context, programId)
|
|
||||||
deletedPrograms += 1
|
|
||||||
Log.d(TAG, "synchronize(): deleted stale programId=$programId for item=$providerId")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
allNextProgramIds.put(channelId.toString(), nextProgramIds.toString())
|
|
||||||
prefs.edit().putString(prefKey, nextProgramIds.toString()).apply()
|
|
||||||
totalActive += activeProviderIds.size
|
|
||||||
totalDeleted += deletedPrograms
|
|
||||||
|
|
||||||
logProviderState(context, channelId)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Store all channel program IDs for clear() to use
|
|
||||||
prefs.edit().putString(KEY_PROGRAM_IDS, allNextProgramIds.toString()).apply()
|
|
||||||
|
|
||||||
Log.d(
|
Log.d(
|
||||||
TAG,
|
TAG,
|
||||||
"synchronize(): completed across ${sections.length()} section(s), $totalActive active program(s), deleted $totalDeleted stale program(s)"
|
"synchronize(): using section \"$sectionTitle\" with ${items.length()} item(s)"
|
||||||
|
)
|
||||||
|
|
||||||
|
val channelId = getOrCreateChannel(context, sectionTitle)
|
||||||
|
if (channelId <= 0L) {
|
||||||
|
Log.w(TAG, "synchronize(): failed to get or create preview channel")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
Log.d(TAG, "synchronize(): publishing into channelId=$channelId")
|
||||||
|
|
||||||
|
val previousProgramIds = preferences(context)
|
||||||
|
.getString(KEY_PROGRAM_IDS, null)
|
||||||
|
?.let(::JSONObject)
|
||||||
|
?: JSONObject()
|
||||||
|
val nextProgramIds = JSONObject()
|
||||||
|
val activeProviderIds = mutableSetOf<String>()
|
||||||
|
|
||||||
|
for (index in 0 until items.length()) {
|
||||||
|
val item = items.optJSONObject(index) ?: continue
|
||||||
|
val providerId = item.optString("id")
|
||||||
|
if (providerId.isBlank()) continue
|
||||||
|
|
||||||
|
val programId = upsertPreviewProgram(
|
||||||
|
context = context,
|
||||||
|
channelId = channelId,
|
||||||
|
item = item,
|
||||||
|
previousProgramId = previousProgramIds.optLong(providerId, -1L),
|
||||||
|
weight = index
|
||||||
|
)
|
||||||
|
|
||||||
|
if (programId > 0L) {
|
||||||
|
activeProviderIds += providerId
|
||||||
|
nextProgramIds.put(providerId, programId)
|
||||||
|
Log.d(TAG, "synchronize(): upserted program for item=$providerId programId=$programId")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var deletedPrograms = 0
|
||||||
|
val previousKeys = previousProgramIds.keys()
|
||||||
|
while (previousKeys.hasNext()) {
|
||||||
|
val providerId = previousKeys.next()
|
||||||
|
if (activeProviderIds.contains(providerId)) continue
|
||||||
|
|
||||||
|
val programId = previousProgramIds.optLong(providerId, -1L)
|
||||||
|
if (programId > 0L) {
|
||||||
|
context.contentResolver.delete(
|
||||||
|
TvContractCompat.buildPreviewProgramUri(programId),
|
||||||
|
null,
|
||||||
|
null
|
||||||
|
)
|
||||||
|
deletedPrograms += 1
|
||||||
|
Log.d(TAG, "synchronize(): deleted stale programId=$programId for item=$providerId")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
preferences(context)
|
||||||
|
.edit()
|
||||||
|
.putLong(KEY_CHANNEL_ID, channelId)
|
||||||
|
.putString(KEY_PROGRAM_IDS, nextProgramIds.toString())
|
||||||
|
.apply()
|
||||||
|
|
||||||
|
logProviderState(context, channelId)
|
||||||
|
|
||||||
|
Log.d(
|
||||||
|
TAG,
|
||||||
|
"synchronize(): completed with ${activeProviderIds.size} active program(s), deleted $deletedPrograms stale program(s)"
|
||||||
)
|
)
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Query provider to check if a channel is browsable.
|
|
||||||
* Per Android docs: "check channel.isBrowsable() before updating programs."
|
|
||||||
*/
|
|
||||||
private fun isChannelBrowsable(context: Context, channelId: Long): Boolean {
|
|
||||||
return try {
|
|
||||||
context.contentResolver.query(
|
|
||||||
TvContractCompat.buildChannelUri(channelId),
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
null
|
|
||||||
)?.use { cursor ->
|
|
||||||
if (cursor.moveToFirst()) {
|
|
||||||
val browsableIndex = cursor.getColumnIndex(TvContractCompat.Channels.COLUMN_BROWSABLE)
|
|
||||||
if (browsableIndex >= 0) cursor.getInt(browsableIndex) == 1 else true
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
} ?: false
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "isChannelBrowsable(): lost provider permission for channelId=$channelId", e)
|
|
||||||
true // Assume browsable if we can't check, to avoid blocking updates
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Query provider to verify a channel actually exists.
|
|
||||||
* Prevents the channel-delete-recreate bug: if update() returns 0 rows
|
|
||||||
* we must first check whether the channel was deleted by the system
|
|
||||||
* or if the update simply failed for another reason.
|
|
||||||
*/
|
|
||||||
private fun channelExistsInProvider(context: Context, channelId: Long): Boolean {
|
|
||||||
return try {
|
|
||||||
context.contentResolver.query(
|
|
||||||
TvContractCompat.buildChannelUri(channelId),
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
null
|
|
||||||
)?.use { cursor ->
|
|
||||||
cursor.moveToFirst()
|
|
||||||
} ?: false
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "channelExistsInProvider(): lost provider permission for channelId=$channelId", e)
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getOrCreateChannel(context: Context, displayName: String): Long {
|
private fun getOrCreateChannel(context: Context, displayName: String): Long {
|
||||||
val prefs = preferences(context)
|
val prefs = preferences(context)
|
||||||
val existingChannelId = prefs.getLong(KEY_CHANNEL_ID, -1L)
|
val existingChannelId = prefs.getLong(KEY_CHANNEL_ID, -1L)
|
||||||
val contentResolver = context.contentResolver
|
val contentResolver = context.contentResolver
|
||||||
|
|
||||||
if (existingChannelId > 0L) {
|
if (existingChannelId > 0L) {
|
||||||
// Query provider first to verify channel actually exists (prevents recreate bug)
|
val updated = Channel.Builder()
|
||||||
val exists = channelExistsInProvider(context, existingChannelId)
|
.setType(TvContractCompat.Channels.TYPE_PREVIEW)
|
||||||
|
.setDisplayName(displayName)
|
||||||
|
.setAppLinkIntentUri(buildIntentUri(context, "streamyfin://"))
|
||||||
|
.build()
|
||||||
|
|
||||||
if (exists) {
|
val updatedRows = contentResolver.update(
|
||||||
// Channel exists — update it in place, never recreate
|
TvContractCompat.buildChannelUri(existingChannelId),
|
||||||
val updated = Channel.Builder()
|
updated.toContentValues(),
|
||||||
.setType(TvContractCompat.Channels.TYPE_PREVIEW)
|
null,
|
||||||
.setDisplayName(displayName)
|
null
|
||||||
.setAppLinkIntentUri(buildIntentUri(context, "streamyfin://"))
|
)
|
||||||
.build()
|
|
||||||
|
|
||||||
try {
|
if (updatedRows > 0) {
|
||||||
val updatedRows = contentResolver.update(
|
TvContractCompat.requestChannelBrowsable(context, existingChannelId)
|
||||||
TvContractCompat.buildChannelUri(existingChannelId),
|
storeChannelLogo(context, existingChannelId)
|
||||||
updated.toContentValues(),
|
Log.d(TAG, "getOrCreateChannel(): updated existing channelId=$existingChannelId and requested browsable")
|
||||||
null,
|
return existingChannelId
|
||||||
null
|
|
||||||
)
|
|
||||||
|
|
||||||
if (updatedRows > 0) {
|
|
||||||
TvContractCompat.requestChannelBrowsable(context, existingChannelId)
|
|
||||||
storeChannelLogo(context, existingChannelId)
|
|
||||||
Log.d(TAG, "getOrCreateChannel(): updated existing channelId=$existingChannelId and requested browsable")
|
|
||||||
return existingChannelId
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update returned 0 rows but channel exists — log and return existing ID, don't recreate
|
|
||||||
Log.e(TAG, "getOrCreateChannel(): update returned 0 for existing channelId=$existingChannelId but channel exists — not recreating")
|
|
||||||
return existingChannelId
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "getOrCreateChannel(): lost provider permission updating channelId=$existingChannelId", e)
|
|
||||||
return existingChannelId
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Channel truly doesn't exist in provider — recreate
|
Log.w(TAG, "getOrCreateChannel(): stored channelId=$existingChannelId was stale, recreating")
|
||||||
Log.w(TAG, "getOrCreateChannel(): channelId=$existingChannelId not in provider, recreating")
|
|
||||||
prefs.edit().remove(KEY_CHANNEL_ID).apply()
|
prefs.edit().remove(KEY_CHANNEL_ID).apply()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create a new channel
|
|
||||||
val channel = Channel.Builder()
|
val channel = Channel.Builder()
|
||||||
.setType(TvContractCompat.Channels.TYPE_PREVIEW)
|
.setType(TvContractCompat.Channels.TYPE_PREVIEW)
|
||||||
.setDisplayName(displayName)
|
.setDisplayName(displayName)
|
||||||
.setAppLinkIntentUri(buildIntentUri(context, "streamyfin://"))
|
.setAppLinkIntentUri(buildIntentUri(context, "streamyfin://"))
|
||||||
.build()
|
.build()
|
||||||
|
|
||||||
val channelUri = try {
|
val channelUri = contentResolver.insert(
|
||||||
contentResolver.insert(
|
TvContractCompat.Channels.CONTENT_URI,
|
||||||
TvContractCompat.Channels.CONTENT_URI,
|
channel.toContentValues()
|
||||||
channel.toContentValues()
|
) ?: return -1L
|
||||||
)
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.e(TAG, "getOrCreateChannel(): lost provider permission, cannot create channel", e)
|
|
||||||
null
|
|
||||||
} ?: return -1L
|
|
||||||
|
|
||||||
val channelId = ContentUris.parseId(channelUri)
|
val channelId = ContentUris.parseId(channelUri)
|
||||||
TvContractCompat.requestChannelBrowsable(context, channelId)
|
TvContractCompat.requestChannelBrowsable(context, channelId)
|
||||||
@@ -415,62 +249,42 @@ internal object TvRecommendationsPublisher {
|
|||||||
builder.setDescription(it)
|
builder.setDescription(it)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Per Android docs: use unique URIs for all images to avoid stale cache
|
|
||||||
imageUrl.takeIf { it.isNotBlank() }?.let {
|
imageUrl.takeIf { it.isNotBlank() }?.let {
|
||||||
val uniqueImageUrl = appendCacheBuster(it)
|
val imageUri = Uri.parse(it)
|
||||||
val imageUri = Uri.parse(uniqueImageUrl)
|
|
||||||
builder.setPosterArtUri(imageUri)
|
builder.setPosterArtUri(imageUri)
|
||||||
builder.setThumbnailUri(imageUri)
|
builder.setThumbnailUri(imageUri)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
val contentValues = builder.build().toContentValues()
|
val contentValues = builder.build().toContentValues()
|
||||||
val contentResolver = context.contentResolver
|
val contentResolver = context.contentResolver
|
||||||
|
|
||||||
if (previousProgramId > 0L) {
|
if (previousProgramId > 0L) {
|
||||||
try {
|
val updatedRows = contentResolver.update(
|
||||||
val updatedRows = contentResolver.update(
|
TvContractCompat.buildPreviewProgramUri(previousProgramId),
|
||||||
TvContractCompat.buildPreviewProgramUri(previousProgramId),
|
contentValues,
|
||||||
contentValues,
|
null,
|
||||||
null,
|
null
|
||||||
null
|
)
|
||||||
)
|
|
||||||
|
|
||||||
if (updatedRows > 0) {
|
if (updatedRows > 0) {
|
||||||
Log.d(TAG, "upsertPreviewProgram(): updated existing programId=$previousProgramId")
|
Log.d(TAG, "upsertPreviewProgram(): updated existing programId=$previousProgramId")
|
||||||
return previousProgramId
|
return previousProgramId
|
||||||
}
|
|
||||||
|
|
||||||
Log.w(TAG, "upsertPreviewProgram(): existing programId=$previousProgramId was stale, inserting new row")
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "upsertPreviewProgram(): lost provider permission for programId=$previousProgramId", e)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Log.w(TAG, "upsertPreviewProgram(): existing programId=$previousProgramId was stale, inserting new row")
|
||||||
}
|
}
|
||||||
|
|
||||||
val insertedUri = try {
|
val insertedUri = contentResolver.insert(
|
||||||
contentResolver.insert(
|
TvContractCompat.PreviewPrograms.CONTENT_URI,
|
||||||
TvContractCompat.PreviewPrograms.CONTENT_URI,
|
contentValues
|
||||||
contentValues
|
) ?: return -1L
|
||||||
)
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "upsertPreviewProgram(): lost provider permission, cannot insert program", e)
|
|
||||||
null
|
|
||||||
} ?: return -1L
|
|
||||||
|
|
||||||
val programId = ContentUris.parseId(insertedUri)
|
val programId = ContentUris.parseId(insertedUri)
|
||||||
Log.d(TAG, "upsertPreviewProgram(): inserted new programId=$programId")
|
Log.d(TAG, "upsertPreviewProgram(): inserted new programId=$programId")
|
||||||
return programId
|
return programId
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Append a cache-busting parameter to ensure unique URIs when images change.
|
|
||||||
* Per Android docs: "Use unique Uris for all images... the old image will
|
|
||||||
* continue to appear if you don't change the Uri."
|
|
||||||
*/
|
|
||||||
private fun appendCacheBuster(imageUrl: String): String {
|
|
||||||
val separator = if (imageUrl.contains("?")) "&" else "?"
|
|
||||||
return "$imageUrl${separator}_t=${System.currentTimeMillis()}"
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun buildIntentUri(context: Context, deepLink: String): Uri {
|
private fun buildIntentUri(context: Context, deepLink: String): Uri {
|
||||||
val intent = Intent(Intent.ACTION_VIEW).apply {
|
val intent = Intent(Intent.ACTION_VIEW).apply {
|
||||||
data = Uri.parse(deepLink)
|
data = Uri.parse(deepLink)
|
||||||
@@ -492,17 +306,13 @@ internal object TvRecommendationsPublisher {
|
|||||||
|
|
||||||
private fun storeChannelLogo(context: Context, channelId: Long) {
|
private fun storeChannelLogo(context: Context, channelId: Long) {
|
||||||
val bitmap = applicationIconBitmap(context) ?: return
|
val bitmap = applicationIconBitmap(context) ?: return
|
||||||
try {
|
val outputStream = context.contentResolver.openOutputStream(
|
||||||
val outputStream = context.contentResolver.openOutputStream(
|
TvContractCompat.buildChannelLogoUri(channelId)
|
||||||
TvContractCompat.buildChannelLogoUri(channelId)
|
) ?: return
|
||||||
) ?: return
|
|
||||||
|
|
||||||
outputStream.use { stream ->
|
outputStream.use { stream ->
|
||||||
bitmap.compress(Bitmap.CompressFormat.PNG, 100, stream)
|
bitmap.compress(Bitmap.CompressFormat.PNG, 100, stream)
|
||||||
stream.flush()
|
stream.flush()
|
||||||
}
|
|
||||||
} catch (e: SecurityException) {
|
|
||||||
Log.w(TAG, "storeChannelLogo(): lost provider permission for channelId=$channelId", e)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -531,14 +341,9 @@ internal object TvRecommendationsPublisher {
|
|||||||
return bitmap
|
return bitmap
|
||||||
}
|
}
|
||||||
|
|
||||||
fun getChannelId(context: Context): Long {
|
|
||||||
return preferences(context).getLong(KEY_CHANNEL_ID, -1L)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun preferences(context: Context): SharedPreferences {
|
private fun preferences(context: Context): SharedPreferences {
|
||||||
return context.getSharedPreferences(PREFS_NAME, Context.MODE_PRIVATE)
|
return context.getSharedPreferences(PREFS_NAME, Context.MODE_PRIVATE)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun logProviderState(context: Context, channelId: Long) {
|
private fun logProviderState(context: Context, channelId: Long) {
|
||||||
val contentResolver = context.contentResolver
|
val contentResolver = context.contentResolver
|
||||||
|
|
||||||
@@ -567,8 +372,8 @@ internal object TvRecommendationsPublisher {
|
|||||||
Log.w(TAG, "logProviderState(): channelId=$channelId exists=false")
|
Log.w(TAG, "logProviderState(): channelId=$channelId exists=false")
|
||||||
}
|
}
|
||||||
} ?: Log.w(TAG, "logProviderState(): channel query returned null for channelId=$channelId")
|
} ?: Log.w(TAG, "logProviderState(): channel query returned null for channelId=$channelId")
|
||||||
} catch (error: SecurityException) {
|
} catch (error: Exception) {
|
||||||
Log.w(TAG, "logProviderState(): lost provider permission for channelId=$channelId", error)
|
Log.w(TAG, "logProviderState(): failed to query channelId=$channelId", error)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,19 +1,12 @@
|
|||||||
import { requireNativeView } from "expo";
|
import { requireNativeView } from "expo";
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import type { View } from "react-native";
|
import type { View } from "react-native";
|
||||||
import { Platform } from "react-native";
|
|
||||||
|
|
||||||
import type { TvSearchViewProps } from "./TvSearchView.types";
|
import type { TvSearchViewProps } from "./TvSearchView.types";
|
||||||
|
|
||||||
// The native TvSearchModule is Apple-only (tvOS SwiftUI `.searchable`).
|
|
||||||
// On Android the component is never rendered, but we must avoid calling
|
|
||||||
// `requireNativeView` at module-scope because it would crash on import.
|
|
||||||
const NativeView: React.ComponentType<
|
const NativeView: React.ComponentType<
|
||||||
TvSearchViewProps & React.RefAttributes<View>
|
TvSearchViewProps & React.RefAttributes<View>
|
||||||
> =
|
> = requireNativeView("TvSearchModule");
|
||||||
Platform.OS === "ios"
|
|
||||||
? requireNativeView("TvSearchModule")
|
|
||||||
: ((() => null) as any);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Forwards its ref to the underlying native view so it can be used as a
|
* Forwards its ref to the underlying native view so it can be used as a
|
||||||
|
|||||||
13
package.json
@@ -32,10 +32,9 @@
|
|||||||
"@expo/react-native-action-sheet": "^4.1.1",
|
"@expo/react-native-action-sheet": "^4.1.1",
|
||||||
"@expo/ui": "~56.0.14",
|
"@expo/ui": "~56.0.14",
|
||||||
"@expo/vector-icons": "^15.0.3",
|
"@expo/vector-icons": "^15.0.3",
|
||||||
"@gorhom/bottom-sheet": "5.2.14",
|
"@gorhom/bottom-sheet": "5.2.8",
|
||||||
"@jellyfin/sdk": "^0.13.0",
|
"@jellyfin/sdk": "^0.13.0",
|
||||||
"@react-native-community/netinfo": "^12.0.0",
|
"@react-native-community/netinfo": "^12.0.0",
|
||||||
"@react-navigation/material-top-tabs": "7.4.28",
|
|
||||||
"@react-navigation/native": "^7.2.5",
|
"@react-navigation/native": "^7.2.5",
|
||||||
"@shopify/flash-list": "2.0.2",
|
"@shopify/flash-list": "2.0.2",
|
||||||
"@tanstack/query-sync-storage-persister": "^5.100.14",
|
"@tanstack/query-sync-storage-persister": "^5.100.14",
|
||||||
@@ -105,7 +104,6 @@
|
|||||||
"react-native-safe-area-context": "~5.7.0",
|
"react-native-safe-area-context": "~5.7.0",
|
||||||
"react-native-screens": "4.25.2",
|
"react-native-screens": "4.25.2",
|
||||||
"react-native-svg": "15.15.4",
|
"react-native-svg": "15.15.4",
|
||||||
"react-native-tab-view": "4.3.0",
|
|
||||||
"react-native-text-ticker": "^1.15.0",
|
"react-native-text-ticker": "^1.15.0",
|
||||||
"react-native-track-player": "github:lovegaoshi/react-native-track-player#APM",
|
"react-native-track-player": "github:lovegaoshi/react-native-track-player#APM",
|
||||||
"react-native-udp": "^4.1.7",
|
"react-native-udp": "^4.1.7",
|
||||||
@@ -129,7 +127,7 @@
|
|||||||
"@types/react": "~19.2.10",
|
"@types/react": "~19.2.10",
|
||||||
"@types/react-test-renderer": "19.1.0",
|
"@types/react-test-renderer": "19.1.0",
|
||||||
"cross-env": "10.1.0",
|
"cross-env": "10.1.0",
|
||||||
"expo-doctor": "1.19.7",
|
"expo-doctor": "1.19.8",
|
||||||
"husky": "9.1.7",
|
"husky": "9.1.7",
|
||||||
"lint-staged": "17.0.5",
|
"lint-staged": "17.0.5",
|
||||||
"react-test-renderer": "19.2.3",
|
"react-test-renderer": "19.2.3",
|
||||||
@@ -164,5 +162,10 @@
|
|||||||
},
|
},
|
||||||
"trustedDependencies": [
|
"trustedDependencies": [
|
||||||
"unrs-resolver"
|
"unrs-resolver"
|
||||||
]
|
],
|
||||||
|
"patchedDependencies": {
|
||||||
|
"react-native-udp@4.1.7": "bun-patches/react-native-udp@4.1.7.patch",
|
||||||
|
"react-native-bottom-tabs@1.2.0": "bun-patches/react-native-bottom-tabs@1.2.0.patch",
|
||||||
|
"react-native-ios-utilities@5.2.0": "bun-patches/react-native-ios-utilities@5.2.0.patch"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,28 +39,6 @@ function buildPatch() {
|
|||||||
" end",
|
" end",
|
||||||
" end",
|
" end",
|
||||||
"",
|
"",
|
||||||
" # iOS 26 / Xcode 26: the APP target itself compiles ExpoModulesProvider.swift,",
|
|
||||||
" # which imports SwiftUI-based modules (ExpoUI, ExpoGlassEffect, GlassPoster, ExpoBlur, …).",
|
|
||||||
" # That emits a `-framework SwiftUICore` autolink into the app executable's OWN object",
|
|
||||||
" # files, so the pods-only flag above is not enough — the app's link still fails with",
|
|
||||||
" # `cannot link directly with 'SwiftUICore'`. Drop the autolink on the user app target",
|
|
||||||
" # too. Phone-only — tvOS has no SwiftUICore split and must stay untouched.",
|
|
||||||
" if ENV['EXPO_TV'] != '1'",
|
|
||||||
" installer.aggregate_targets.each do |agg|",
|
|
||||||
" next unless agg.user_project",
|
|
||||||
" agg.user_project.native_targets.each do |target|",
|
|
||||||
" target.build_configurations.each do |cfg|",
|
|
||||||
" existing = cfg.build_settings['OTHER_SWIFT_FLAGS'] || '$(inherited)'",
|
|
||||||
" existing = existing.join(' ') if existing.is_a?(Array)",
|
|
||||||
" unless existing.include?('-disable-autolink-framework -Xfrontend SwiftUICore')",
|
|
||||||
" cfg.build_settings['OTHER_SWIFT_FLAGS'] = existing + ' -Xfrontend -disable-autolink-framework -Xfrontend SwiftUICore'",
|
|
||||||
" end",
|
|
||||||
" end",
|
|
||||||
" end",
|
|
||||||
" agg.user_project.save",
|
|
||||||
" end",
|
|
||||||
" end",
|
|
||||||
"",
|
|
||||||
" # Safely patch RCTThirdPartyComponentsProvider.mm to avoid startup crash on unlinked Fabric components",
|
" # Safely patch RCTThirdPartyComponentsProvider.mm to avoid startup crash on unlinked Fabric components",
|
||||||
' filepath = "#{installer.sandbox.root}/../build/generated/ios/ReactCodegen/RCTThirdPartyComponentsProvider.mm"',
|
' filepath = "#{installer.sandbox.root}/../build/generated/ios/ReactCodegen/RCTThirdPartyComponentsProvider.mm"',
|
||||||
" if File.exist?(filepath)",
|
" if File.exist?(filepath)",
|
||||||
|
|||||||
@@ -4,16 +4,9 @@ const { withEntitlementsPlist } = require("expo/config-plugins");
|
|||||||
* Expo config plugin to add User Management entitlement for tvOS profile linking
|
* Expo config plugin to add User Management entitlement for tvOS profile linking
|
||||||
*/
|
*/
|
||||||
const withTVUserManagement = (config) => {
|
const withTVUserManagement = (config) => {
|
||||||
// Only add for tvOS builds. The entitlement is restricted by Apple and must
|
|
||||||
// be present in the provisioning profile, so injecting it into mobile builds
|
|
||||||
// breaks signing ("Entitlement ... not found and could not be included in
|
|
||||||
// profile"). The entitlement is only needed for tvOS
|
|
||||||
// TVUserManager.currentUserIdentifier.
|
|
||||||
if (process.env.EXPO_TV !== "1") {
|
|
||||||
return config;
|
|
||||||
}
|
|
||||||
|
|
||||||
return withEntitlementsPlist(config, (config) => {
|
return withEntitlementsPlist(config, (config) => {
|
||||||
|
// Only add for tvOS builds (check if building for TV)
|
||||||
|
// The entitlement is needed for TVUserManager.currentUserIdentifier to work
|
||||||
config.modResults["com.apple.developer.user-management"] = [
|
config.modResults["com.apple.developer.user-management"] = [
|
||||||
"runs-as-current-user",
|
"runs-as-current-user",
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -4,68 +4,28 @@ import type { DownloadedItem, DownloadsDatabase } from "./types";
|
|||||||
|
|
||||||
const DOWNLOADS_DATABASE_KEY = "downloads.v2.json";
|
const DOWNLOADS_DATABASE_KEY = "downloads.v2.json";
|
||||||
|
|
||||||
// Performance optimization: Cache the parsed database to avoid repeated JSON.parse calls
|
|
||||||
let cachedDb: DownloadsDatabase | null = null;
|
|
||||||
let cacheVersion = 0;
|
|
||||||
|
|
||||||
// Performance optimization: Cache the flattened items array
|
|
||||||
let cachedItems: DownloadedItem[] | null = null;
|
|
||||||
let itemsCacheVersion = -1;
|
|
||||||
|
|
||||||
// Performance optimization: Index for O(1) item lookups by ID
|
|
||||||
let itemIndex: Map<string, DownloadedItem> | null = null;
|
|
||||||
let indexCacheVersion = -1;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the downloads database from storage
|
* Get the downloads database from storage
|
||||||
* PERFORMANCE: Caches the parsed database to avoid repeated JSON.parse calls.
|
|
||||||
* NOTE: Returns the shared cached instance — do NOT mutate it directly. Go
|
|
||||||
* through addDownloadedItem/updateDownloadedItem/removeDownloadedItem so
|
|
||||||
* saveDownloadsDatabase() runs and the derived caches stay consistent.
|
|
||||||
*/
|
*/
|
||||||
export function getDownloadsDatabase(): DownloadsDatabase {
|
export function getDownloadsDatabase(): DownloadsDatabase {
|
||||||
// Return cached database if available
|
|
||||||
if (cachedDb !== null) {
|
|
||||||
return cachedDb;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse from storage and cache the result
|
|
||||||
const file = storage.getString(DOWNLOADS_DATABASE_KEY);
|
const file = storage.getString(DOWNLOADS_DATABASE_KEY);
|
||||||
if (file) {
|
if (file) {
|
||||||
cachedDb = JSON.parse(file) as DownloadsDatabase;
|
return JSON.parse(file) as DownloadsDatabase;
|
||||||
return cachedDb;
|
|
||||||
}
|
}
|
||||||
|
return { movies: {}, series: {}, other: {} };
|
||||||
const emptyDb = { movies: {}, series: {}, other: {} };
|
|
||||||
cachedDb = emptyDb;
|
|
||||||
return emptyDb;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save the downloads database to storage
|
* Save the downloads database to storage
|
||||||
* PERFORMANCE: Updates cache and invalidates derived caches
|
|
||||||
*/
|
*/
|
||||||
export function saveDownloadsDatabase(db: DownloadsDatabase): void {
|
export function saveDownloadsDatabase(db: DownloadsDatabase): void {
|
||||||
storage.set(DOWNLOADS_DATABASE_KEY, JSON.stringify(db));
|
storage.set(DOWNLOADS_DATABASE_KEY, JSON.stringify(db));
|
||||||
// Update the cache with the new database
|
|
||||||
cachedDb = db;
|
|
||||||
// Invalidate derived caches (items array and index)
|
|
||||||
cachedItems = null;
|
|
||||||
itemIndex = null;
|
|
||||||
cacheVersion++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get all downloaded items as a flat array
|
* Get all downloaded items as a flat array
|
||||||
* PERFORMANCE: Caches the flattened array to avoid rebuilding on every call
|
|
||||||
*/
|
*/
|
||||||
export function getAllDownloadedItems(): DownloadedItem[] {
|
export function getAllDownloadedItems(): DownloadedItem[] {
|
||||||
// Return cached items if available and up-to-date
|
|
||||||
if (cachedItems !== null && itemsCacheVersion === cacheVersion) {
|
|
||||||
return cachedItems;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build the items array from the database
|
|
||||||
const db = getDownloadsDatabase();
|
const db = getDownloadsDatabase();
|
||||||
const items: DownloadedItem[] = [];
|
const items: DownloadedItem[] = [];
|
||||||
|
|
||||||
@@ -87,41 +47,34 @@ export function getAllDownloadedItems(): DownloadedItem[] {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cache the result
|
|
||||||
cachedItems = items;
|
|
||||||
itemsCacheVersion = cacheVersion;
|
|
||||||
|
|
||||||
return items;
|
return items;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build or refresh the item index for O(1) lookups
|
* Get a downloaded item by its ID
|
||||||
*/
|
*/
|
||||||
function ensureItemIndex(): void {
|
export function getDownloadedItemById(id: string): DownloadedItem | undefined {
|
||||||
if (itemIndex !== null && indexCacheVersion === cacheVersion) {
|
const db = getDownloadsDatabase();
|
||||||
return; // Index is up-to-date
|
|
||||||
|
if (db.movies[id]) {
|
||||||
|
return db.movies[id];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build new index from all items
|
for (const series of Object.values(db.series)) {
|
||||||
itemIndex = new Map<string, DownloadedItem>();
|
for (const season of Object.values(series.seasons)) {
|
||||||
const items = getAllDownloadedItems();
|
for (const episode of Object.values(season.episodes)) {
|
||||||
|
if (episode.item.Id === id) {
|
||||||
for (const item of items) {
|
return episode;
|
||||||
if (item.item.Id) {
|
}
|
||||||
itemIndex.set(item.item.Id, item);
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
indexCacheVersion = cacheVersion;
|
if (db.other?.[id]) {
|
||||||
}
|
return db.other[id];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
return undefined;
|
||||||
* Get a downloaded item by its ID
|
|
||||||
* PERFORMANCE: Uses O(1) index lookup instead of O(n²) iteration
|
|
||||||
*/
|
|
||||||
export function getDownloadedItemById(id: string): DownloadedItem | undefined {
|
|
||||||
ensureItemIndex();
|
|
||||||
return itemIndex!.get(id);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -268,5 +221,4 @@ export function updateDownloadedItem(
|
|||||||
*/
|
*/
|
||||||
export function clearAllDownloadedItems(): void {
|
export function clearAllDownloadedItems(): void {
|
||||||
saveDownloadsDatabase({ movies: {}, series: {}, other: {} });
|
saveDownloadsDatabase({ movies: {}, series: {}, other: {} });
|
||||||
// saveDownloadsDatabase already invalidates caches
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ const initialApi = (() => {
|
|||||||
const id = getOrSetDeviceId();
|
const id = getOrSetDeviceId();
|
||||||
const deviceName = getDeviceNameSync();
|
const deviceName = getDeviceNameSync();
|
||||||
const jellyfinInstance = new Jellyfin({
|
const jellyfinInstance = new Jellyfin({
|
||||||
clientInfo: { name: "Streamyfin", version: "0.54.1" },
|
clientInfo: { name: "Streamyfin", version: "0.54.0" },
|
||||||
deviceInfo: {
|
deviceInfo: {
|
||||||
name: deviceName,
|
name: deviceName,
|
||||||
id,
|
id,
|
||||||
@@ -69,13 +69,6 @@ const initialApi = (() => {
|
|||||||
|
|
||||||
const initialUser = (() => {
|
const initialUser = (() => {
|
||||||
try {
|
try {
|
||||||
// Only return a stored user if we also have a token. Otherwise the
|
|
||||||
// user atom would be populated while the api atom is null (e.g. after
|
|
||||||
// a logout that left stale user JSON in storage), which causes
|
|
||||||
// useProtectedRoute to keep us inside the (auth) group instead of
|
|
||||||
// redirecting to /login.
|
|
||||||
const token = storage.getString("token");
|
|
||||||
if (!token) return null;
|
|
||||||
const userStr = storage.getString("user");
|
const userStr = storage.getString("user");
|
||||||
if (userStr) {
|
if (userStr) {
|
||||||
return JSON.parse(userStr) as UserDto;
|
return JSON.parse(userStr) as UserDto;
|
||||||
@@ -135,7 +128,7 @@ export const JellyfinProvider: React.FC<{ children: ReactNode }> = ({
|
|||||||
const id = getOrSetDeviceId();
|
const id = getOrSetDeviceId();
|
||||||
const deviceName = getDeviceNameSync();
|
const deviceName = getDeviceNameSync();
|
||||||
return new Jellyfin({
|
return new Jellyfin({
|
||||||
clientInfo: { name: "Streamyfin", version: "0.54.1" },
|
clientInfo: { name: "Streamyfin", version: "0.54.0" },
|
||||||
deviceInfo: {
|
deviceInfo: {
|
||||||
name: deviceName,
|
name: deviceName,
|
||||||
id,
|
id,
|
||||||
@@ -169,7 +162,7 @@ export const JellyfinProvider: React.FC<{ children: ReactNode }> = ({
|
|||||||
return {
|
return {
|
||||||
authorization: `MediaBrowser Client="Streamyfin", Device=${
|
authorization: `MediaBrowser Client="Streamyfin", Device=${
|
||||||
Platform.OS === "android" ? "Android" : "iOS"
|
Platform.OS === "android" ? "Android" : "iOS"
|
||||||
}, DeviceId="${deviceId}", Version="0.54.1"`,
|
}, DeviceId="${deviceId}", Version="0.54.0"`,
|
||||||
};
|
};
|
||||||
}, [deviceId]);
|
}, [deviceId]);
|
||||||
|
|
||||||
@@ -409,7 +402,6 @@ export const JellyfinProvider: React.FC<{ children: ReactNode }> = ({
|
|||||||
);
|
);
|
||||||
|
|
||||||
storage.remove("token");
|
storage.remove("token");
|
||||||
storage.remove("user");
|
|
||||||
clearTVDiscoverySafely();
|
clearTVDiscoverySafely();
|
||||||
setUser(null);
|
setUser(null);
|
||||||
setApi(null);
|
setApi(null);
|
||||||
@@ -619,54 +611,44 @@ export const JellyfinProvider: React.FC<{ children: ReactNode }> = ({
|
|||||||
setUser(storedUser);
|
setUser(storedUser);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dismiss splash screen with cached data immediately,
|
const response = await getUserApi(apiInstance).getCurrentUser();
|
||||||
// fetch fresh user data in the background
|
setUser(response.data);
|
||||||
setInitialLoaded(true);
|
|
||||||
|
|
||||||
try {
|
// Migrate current session to secure storage if not already saved
|
||||||
const response = await getUserApi(apiInstance).getCurrentUser();
|
if (storedUser?.Id && storedUser?.Name) {
|
||||||
setUser(response.data);
|
const existingCredential = await getAccountCredential(
|
||||||
|
serverUrl,
|
||||||
// Migrate current session to secure storage if not already saved
|
storedUser.Id,
|
||||||
if (storedUser?.Id && storedUser?.Name) {
|
);
|
||||||
const existingCredential = await getAccountCredential(
|
if (!existingCredential) {
|
||||||
|
await saveAccountCredential({
|
||||||
serverUrl,
|
serverUrl,
|
||||||
storedUser.Id,
|
serverName: "",
|
||||||
);
|
token,
|
||||||
if (!existingCredential) {
|
userId: storedUser.Id,
|
||||||
await saveAccountCredential({
|
username: storedUser.Name,
|
||||||
serverUrl,
|
savedAt: Date.now(),
|
||||||
serverName: "",
|
securityType: "none",
|
||||||
token,
|
primaryImageTag: response.data.PrimaryImageTag ?? undefined,
|
||||||
userId: storedUser.Id,
|
});
|
||||||
username: storedUser.Name,
|
} else if (
|
||||||
savedAt: Date.now(),
|
response.data.PrimaryImageTag !==
|
||||||
securityType: "none",
|
existingCredential.primaryImageTag
|
||||||
primaryImageTag: response.data.PrimaryImageTag ?? undefined,
|
) {
|
||||||
});
|
// Update image tag if it has changed
|
||||||
} else if (
|
addAccountToServer(serverUrl, existingCredential.serverName, {
|
||||||
response.data.PrimaryImageTag !==
|
userId: existingCredential.userId,
|
||||||
existingCredential.primaryImageTag
|
username: existingCredential.username,
|
||||||
) {
|
securityType: existingCredential.securityType,
|
||||||
// Update image tag if it has changed
|
savedAt: existingCredential.savedAt,
|
||||||
addAccountToServer(serverUrl, existingCredential.serverName, {
|
primaryImageTag: response.data.PrimaryImageTag ?? undefined,
|
||||||
userId: existingCredential.userId,
|
});
|
||||||
username: existingCredential.username,
|
|
||||||
securityType: existingCredential.securityType,
|
|
||||||
savedAt: existingCredential.savedAt,
|
|
||||||
primaryImageTag: response.data.PrimaryImageTag ?? undefined,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} catch (e) {
|
|
||||||
// Background fetch failed — app already rendered with cached data
|
|
||||||
console.warn("Background user fetch failed, using cached data:", e);
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
setInitialLoaded(true);
|
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
|
} finally {
|
||||||
setInitialLoaded(true);
|
setInitialLoaded(true);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -29,10 +29,6 @@
|
|||||||
<string>$(MARKETING_VERSION)</string>
|
<string>$(MARKETING_VERSION)</string>
|
||||||
<key>CFBundleVersion</key>
|
<key>CFBundleVersion</key>
|
||||||
<string>$(CURRENT_PROJECT_VERSION)</string>
|
<string>$(CURRENT_PROJECT_VERSION)</string>
|
||||||
<key>UIRequiredDeviceCapabilities</key>
|
|
||||||
<array>
|
|
||||||
<string>arm64</string>
|
|
||||||
</array>
|
|
||||||
<key>NSExtension</key>
|
<key>NSExtension</key>
|
||||||
<dict>
|
<dict>
|
||||||
<key>NSExtensionPointIdentifier</key>
|
<key>NSExtensionPointIdentifier</key>
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "خطأ",
|
"error_title": "خطأ",
|
||||||
"login_title": "تسجيل الدخول",
|
"login_title": "تسجيل الدخول",
|
||||||
"login_to_title": "تسجيل الدخول إلى",
|
"login_to_title": "تسجيل الدخول إلى",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "اسم المستخدم",
|
"username_placeholder": "اسم المستخدم",
|
||||||
"password_placeholder": "كلمة المرور",
|
"password_placeholder": "كلمة المرور",
|
||||||
"login_button": "تسجيل الدخول",
|
"login_button": "تسجيل الدخول",
|
||||||
@@ -33,54 +30,48 @@
|
|||||||
"connect_button": "اتصل",
|
"connect_button": "اتصل",
|
||||||
"previous_servers": "الخوادم السابقة",
|
"previous_servers": "الخوادم السابقة",
|
||||||
"clear_button": "مسح",
|
"clear_button": "مسح",
|
||||||
"swipe_to_remove": "مرر للإزالة",
|
"swipe_to_remove": "Swipe to remove",
|
||||||
"search_for_local_servers": "البحث عن الخوادم المحلية",
|
"search_for_local_servers": "البحث عن الخوادم المحلية",
|
||||||
"searching": "جاري البحث...",
|
"searching": "جاري البحث...",
|
||||||
"servers": "الخوادم",
|
"servers": "الخوادم",
|
||||||
"saved": "تم الحفظ",
|
"saved": "Saved",
|
||||||
"session_expired": "انتهت الجلسة",
|
"session_expired": "Session Expired",
|
||||||
"please_login_again": "انتهت مدة صلاحية جلستك. الرجاء تسجيل الدخول مرة أخرى.",
|
"please_login_again": "Your saved session has expired. Please log in again.",
|
||||||
"remove_saved_login": "إزالة تسجيل دخول محفوظ",
|
"remove_saved_login": "Remove Saved Login",
|
||||||
"remove_saved_login_description": "سيؤدي هذا إلى إزالة بيانات تسجيل الدخول الخاص بك المحفوظة لهذا الخادم. ستحتاج إلى إدخال اسم المستخدم وكلمة المرور مرة أخرى في المرة القادمة.",
|
"remove_saved_login_description": "This will remove your saved credentials for this server. You'll need to enter your username and password again next time.",
|
||||||
"accounts_count": "الحسابات {{count}}",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "اختر الحساب",
|
"select_account": "Select Account",
|
||||||
"add_account": "إضافة حساب",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "سيؤدي هذا إلى إزالة بيانات تسجيل الدخول لـ {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "حفظ الحساب",
|
"title": "Save Account",
|
||||||
"save_for_later": "حفظ هذا الحساب",
|
"save_for_later": "Save this account",
|
||||||
"security_option": "خيارات الأمان",
|
"security_option": "Security Option",
|
||||||
"no_protection": "بدون حماية",
|
"no_protection": "No protection",
|
||||||
"no_protection_desc": "تسجيل دخول سريع بدون مصادقة",
|
"no_protection_desc": "Quick login without authentication",
|
||||||
"pin_code": "رمز PIN",
|
"pin_code": "PIN code",
|
||||||
"pin_code_desc": "رمز PIN مكون من 4 أرقام مطلوب عند التبديل",
|
"pin_code_desc": "4-digit PIN required when switching",
|
||||||
"password": "أعد إدخال كلمة المرور",
|
"password": "Re-enter password",
|
||||||
"password_desc": "كلمة المرور مطلوبة عند التبديل",
|
"password_desc": "Password required when switching",
|
||||||
"save_button": "حفظ",
|
"save_button": "Save",
|
||||||
"cancel_button": "إلغاء"
|
"cancel_button": "Cancel"
|
||||||
},
|
},
|
||||||
"pin": {
|
"pin": {
|
||||||
"enter_pin": "أدخل رمز PIN",
|
"enter_pin": "Enter PIN",
|
||||||
"enter_pin_for": "أدخل رمز PIN لـ {{username}}",
|
"enter_pin_for": "Enter PIN for {{username}}",
|
||||||
"enter_4_digits": "ادخل 4 أرقام",
|
"enter_4_digits": "Enter 4 digits",
|
||||||
"invalid_pin": "PIN غير صالح",
|
"invalid_pin": "Invalid PIN",
|
||||||
"setup_pin": "تعيين رمز PIN",
|
"setup_pin": "Set Up PIN",
|
||||||
"confirm_pin": "تأكيد رمز PIN",
|
"confirm_pin": "Confirm PIN",
|
||||||
"pins_dont_match": "رموز PIN غير متطابقة",
|
"pins_dont_match": "PINs don't match",
|
||||||
"forgot_pin": "نسيت رمز PIN؟",
|
"forgot_pin": "Forgot PIN?",
|
||||||
"forgot_pin_desc": "سيتم إزالة بيانات تسجيل الدخول المحفوظة الخاصة بك"
|
"forgot_pin_desc": "Your saved credentials will be removed"
|
||||||
},
|
},
|
||||||
"password": {
|
"password": {
|
||||||
"enter_password": "أدخل كلمة المرور",
|
"enter_password": "Enter Password",
|
||||||
"enter_password_for": "أدخل كلمة المرور لـ {{username}}",
|
"enter_password_for": "Enter password for {{username}}",
|
||||||
"invalid_password": "كلمة المرور غير صحيحة"
|
"invalid_password": "Invalid password"
|
||||||
},
|
},
|
||||||
"home": {
|
"home": {
|
||||||
"checking_server_connection": "التحقق من اتصال الخادم...",
|
"checking_server_connection": "التحقق من اتصال الخادم...",
|
||||||
@@ -95,9 +86,8 @@
|
|||||||
"oops": "عفوًا!",
|
"oops": "عفوًا!",
|
||||||
"error_message": "حدث خطأ ما.\nيرجى تسجيل الخروج ثم الدخول مرة أخرى.",
|
"error_message": "حدث خطأ ما.\nيرجى تسجيل الخروج ثم الدخول مرة أخرى.",
|
||||||
"continue_watching": "متابعة المشاهدة",
|
"continue_watching": "متابعة المشاهدة",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "التالي",
|
"next_up": "التالي",
|
||||||
"continue_and_next_up": "تابع و التالي",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "أضيف مؤخراً في {{libraryName}}",
|
"recently_added_in": "أضيف مؤخراً في {{libraryName}}",
|
||||||
"suggested_movies": "أفلام مقترحة",
|
"suggested_movies": "أفلام مقترحة",
|
||||||
"suggested_episodes": "حلقات مقترحة",
|
"suggested_episodes": "حلقات مقترحة",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "الإعدادات",
|
"settings_title": "الإعدادات",
|
||||||
"log_out_button": "تسجيل الخروج",
|
"log_out_button": "تسجيل الخروج",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "الأقسام"
|
"title": "الأقسام"
|
||||||
},
|
},
|
||||||
@@ -136,45 +120,36 @@
|
|||||||
},
|
},
|
||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "المظهر",
|
"title": "المظهر",
|
||||||
"merge_next_up_continue_watching": "دمج تابع المشاهدة والتالي",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "إخفاء زر جلسة البث عن بُعد",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "الشبكة",
|
"title": "Network",
|
||||||
"local_network": "الشبكة المحلية",
|
"local_network": "Local Network",
|
||||||
"auto_switch_enabled": "التبديل التلقائي عند المنزل",
|
"auto_switch_enabled": "Auto-switch when at home",
|
||||||
"auto_switch_description": "التبديل تلقائياً إلى رابط URL محلي عند الاتصال بشبكة WiFi المنزلية",
|
"auto_switch_description": "Automatically switch to local URL when connected to home WiFi",
|
||||||
"local_url": "رابط محلي",
|
"local_url": "Local URL",
|
||||||
"local_url_hint": "أدخل عنوان الخادم المحلي الخاص بك (على سبيل المثال http://192.168.1.100:8096)",
|
"local_url_hint": "Enter your local server address (e.g., http://192.168.1.100:8096)",
|
||||||
"local_url_placeholder": "http://192.168.1.100:8096",
|
"local_url_placeholder": "http://192.168.1.100:8096",
|
||||||
"home_wifi_networks": "شبكات WiFi المنزل",
|
"home_wifi_networks": "Home WiFi Networks",
|
||||||
"add_current_network": "إضافة \"{{ssid}}\"",
|
"add_current_network": "Add \"{{ssid}}\"",
|
||||||
"not_connected_to_wifi": "غير متصل بشبكة WiFi",
|
"not_connected_to_wifi": "Not connected to WiFi",
|
||||||
"no_networks_configured": "لا توجد شبكات مكونة",
|
"no_networks_configured": "No networks configured",
|
||||||
"add_network_hint": "إضافة شبكة WiFi المنزلية الخاصة بك لتمكين التبديل التلقائي",
|
"add_network_hint": "Add your home WiFi network to enable auto-switching",
|
||||||
"current_wifi": "شبكة WiFi الحالية",
|
"current_wifi": "Current WiFi",
|
||||||
"using_url": "استخدام",
|
"using_url": "Using",
|
||||||
"local": "رابط محلي",
|
"local": "Local URL",
|
||||||
"remote": "الـ URL الخارجي",
|
"remote": "Remote URL",
|
||||||
"not_connected": "غير متصل",
|
"not_connected": "Not connected",
|
||||||
"current_server": "الخادم الحالي",
|
"current_server": "Current Server",
|
||||||
"remote_url": "الـ URL الخارجي",
|
"remote_url": "Remote URL",
|
||||||
"active_url": "الرابط النشط",
|
"active_url": "Active URL",
|
||||||
"not_configured": "لم يتم تكوينه",
|
"not_configured": "Not configured",
|
||||||
"network_added": "تمت إضافة الشبكة",
|
"network_added": "Network added",
|
||||||
"network_already_added": "الشبكة مضافة مسبقاً",
|
"network_already_added": "Network already added",
|
||||||
"no_wifi_connected": "غير متصل بشبكة WiFi",
|
"no_wifi_connected": "Not connected to WiFi",
|
||||||
"permission_denied": "تم رفض إذن الوصول إلى الموقع",
|
"permission_denied": "Location permission denied",
|
||||||
"permission_denied_explanation": "يتطلب التعرف على شبكة WiFi للتبديل التلقائي الحصول على إذن الوصول إلى الموقع. يرجى تفعيله من الإعدادات."
|
"permission_denied_explanation": "Location permission is required to detect WiFi network for auto-switching. Please enable it in Settings."
|
||||||
},
|
},
|
||||||
"user_info": {
|
"user_info": {
|
||||||
"user_info_title": "معلومات المستخدم",
|
"user_info_title": "معلومات المستخدم",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "مدة الترجيع",
|
"rewind_length": "مدة الترجيع",
|
||||||
"seconds_unit": "ث"
|
"seconds_unit": "ث"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "التحكم بالإيماءات",
|
"gesture_controls_title": "التحكم بالإيماءات",
|
||||||
"horizontal_swipe_skip": "السحب الأفقي للتخطي",
|
"horizontal_swipe_skip": "السحب الأفقي للتخطي",
|
||||||
@@ -223,10 +182,10 @@
|
|||||||
"left_side_brightness_description": "اسحب لأعلى/لأسفل على الجانب الأيسر لضبط السطوع",
|
"left_side_brightness_description": "اسحب لأعلى/لأسفل على الجانب الأيسر لضبط السطوع",
|
||||||
"right_side_volume": "التحكم في مستوى الصوت من الجانب الأيمن",
|
"right_side_volume": "التحكم في مستوى الصوت من الجانب الأيمن",
|
||||||
"right_side_volume_description": "اسحب لأعلى/لأسفل على الجانب الأيمن لضبط مستوى الصوت",
|
"right_side_volume_description": "اسحب لأعلى/لأسفل على الجانب الأيمن لضبط مستوى الصوت",
|
||||||
"hide_volume_slider": "إخفاء شريط مستوى الصوت",
|
"hide_volume_slider": "Hide Volume Slider",
|
||||||
"hide_volume_slider_description": "إخفاء شريط التحكم في مستوى الصوت في مشغل الفيديو",
|
"hide_volume_slider_description": "Hide the volume slider in the video player",
|
||||||
"hide_brightness_slider": "إخفاء شريط السطوع",
|
"hide_brightness_slider": "Hide Brightness Slider",
|
||||||
"hide_brightness_slider_description": "إخفاء شريط التحكم في السطوع في مشغل الفيديو"
|
"hide_brightness_slider_description": "Hide the brightness slider in the video player"
|
||||||
},
|
},
|
||||||
"audio": {
|
"audio": {
|
||||||
"audio_title": "الصوت",
|
"audio_title": "الصوت",
|
||||||
@@ -236,12 +195,12 @@
|
|||||||
"none": "لا شيء",
|
"none": "لا شيء",
|
||||||
"language": "اللغة",
|
"language": "اللغة",
|
||||||
"transcode_mode": {
|
"transcode_mode": {
|
||||||
"title": "تحويل ترميز الصوت",
|
"title": "Audio Transcoding",
|
||||||
"description": "يتحكم في كيفية التعامل مع الصوت المحيطي (7.1، TrueHD، DTS-HD)",
|
"description": "Controls how surround audio (7.1, TrueHD, DTS-HD) is handled",
|
||||||
"auto": "تلقائي",
|
"auto": "Auto",
|
||||||
"stereo": "إجبار تشغيل ستيريو",
|
"stereo": "Force Stereo",
|
||||||
"5_1": "السماح بـ 5.1",
|
"5_1": "Allow 5.1",
|
||||||
"passthrough": "تمرير الصوت"
|
"passthrough": "Passthrough"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"subtitles": {
|
"subtitles": {
|
||||||
@@ -292,45 +251,29 @@
|
|||||||
"Normal": "عادي",
|
"Normal": "عادي",
|
||||||
"Thick": "سميك"
|
"Thick": "سميك"
|
||||||
},
|
},
|
||||||
"subtitle_color": "لون الترجمة",
|
"subtitle_color": "Subtitle Color",
|
||||||
"subtitle_background_color": "لون الخلفية",
|
"subtitle_background_color": "Background Color",
|
||||||
"subtitle_font": "خط الترجمة",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "إعدادات KSPlayer",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "فك الترميز بواسطة الجهاز",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "استخدم تسريع العتاد لفك ترميز الفيديو. قم بتعطيله إذا واجهت مشكلات في التشغيل.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "إعدادات ترجمة VLC",
|
"title": "VLC Subtitle Settings",
|
||||||
"hint": "تخصيص مظهر الترجمة لمشغل VLC. تصبح التغييرات سارية المفعول عند التشغيل التالي.",
|
"hint": "Customize subtitle appearance for VLC player. Changes take effect on next playback.",
|
||||||
"text_color": "لون النص",
|
"text_color": "Text Color",
|
||||||
"background_color": "لون الخلفية",
|
"background_color": "Background Color",
|
||||||
"background_opacity": "شفافية الخلفية",
|
"background_opacity": "Background Opacity",
|
||||||
"outline_color": "لون إطار الخط",
|
"outline_color": "Outline Color",
|
||||||
"outline_opacity": "شفافية إطار الخط",
|
"outline_opacity": "Outline Opacity",
|
||||||
"outline_thickness": "سمك إطار الخط",
|
"outline_thickness": "Outline Thickness",
|
||||||
"bold": "خط عريض",
|
"bold": "Bold Text",
|
||||||
"margin": "الهامش السفلي"
|
"margin": "Bottom Margin"
|
||||||
},
|
},
|
||||||
"video_player": {
|
"video_player": {
|
||||||
"title": "مشغل الفيديو",
|
"title": "Video Player",
|
||||||
"video_player": "مشغل الفيديو",
|
"video_player": "Video Player",
|
||||||
"video_player_description": "اختر مشغل الفيديو الذي سيتم استخدامه على نظام iOS.",
|
"video_player_description": "Choose which video player to use on iOS.",
|
||||||
"ksplayer": "KSPlayer",
|
"ksplayer": "KSPlayer",
|
||||||
"vlc": "VLC"
|
"vlc": "VLC"
|
||||||
},
|
},
|
||||||
@@ -362,8 +305,8 @@
|
|||||||
"select_liraries_you_want_to_hide": "اختر المكتبات التي تريد إخفاءها من تبويب المكتبة وأقسام الصفحة الرئيسية.",
|
"select_liraries_you_want_to_hide": "اختر المكتبات التي تريد إخفاءها من تبويب المكتبة وأقسام الصفحة الرئيسية.",
|
||||||
"disable_haptic_feedback": "تعطيل ردود الفعل اللمسية",
|
"disable_haptic_feedback": "تعطيل ردود الفعل اللمسية",
|
||||||
"default_quality": "الجودة الافتراضية",
|
"default_quality": "الجودة الافتراضية",
|
||||||
"default_playback_speed": "سرعة التشغيل الافتراضية",
|
"default_playback_speed": "Default Playback Speed",
|
||||||
"auto_play_next_episode": "تشغيل الحلقة التالية تلقائياً",
|
"auto_play_next_episode": "Auto-play Next Episode",
|
||||||
"max_auto_play_episode_count": "الحد الأقصى لعدد الحلقات التي يتم تشغيلها تلقائيًا",
|
"max_auto_play_episode_count": "الحد الأقصى لعدد الحلقات التي يتم تشغيلها تلقائيًا",
|
||||||
"disabled": "معطل"
|
"disabled": "معطل"
|
||||||
},
|
},
|
||||||
@@ -371,15 +314,15 @@
|
|||||||
"downloads_title": "التنزيلات"
|
"downloads_title": "التنزيلات"
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "الموسيقى",
|
"title": "Music",
|
||||||
"playback_title": "التشغيل",
|
"playback_title": "Playback",
|
||||||
"playback_description": "ضبط كيفية تشغيل الموسيقى.",
|
"playback_description": "Configure how music is played.",
|
||||||
"prefer_downloaded": "تفضيل الأغاني التي تم تنزيلها",
|
"prefer_downloaded": "Prefer Downloaded Songs",
|
||||||
"caching_title": "التخزين المؤقت",
|
"caching_title": "Caching",
|
||||||
"caching_description": "تخزين الأغاني التالية مؤقتاً تلقائياً لضمان تشغيل أكثر سلاسة.",
|
"caching_description": "Automatically cache upcoming tracks for smoother playback.",
|
||||||
"lookahead_enabled": "تفعيل التخزين المؤقت الاستباقي",
|
"lookahead_enabled": "Enable Look-Ahead Caching",
|
||||||
"lookahead_count": "عدد الأغاني المراد تخزينها مسبقاً",
|
"lookahead_count": "Tracks to Pre-cache",
|
||||||
"max_cache_size": "الحد الأقصى لحجم التخزين المؤقت"
|
"max_cache_size": "Max Cache Size"
|
||||||
},
|
},
|
||||||
"plugins": {
|
"plugins": {
|
||||||
"plugins_title": "الإضافات",
|
"plugins_title": "الإضافات",
|
||||||
@@ -414,39 +357,39 @@
|
|||||||
"save_button": "حفظ",
|
"save_button": "حفظ",
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"saved": "تم الحفظ",
|
"saved": "تم الحفظ",
|
||||||
"refreshed": "تم تحديث الإعدادات من الخادم"
|
"refreshed": "Settings refreshed from server"
|
||||||
},
|
},
|
||||||
"refresh_from_server": "تحديث الإعدادات من الخادم"
|
"refresh_from_server": "Refresh Settings from Server"
|
||||||
},
|
},
|
||||||
"streamystats": {
|
"streamystats": {
|
||||||
"enable_streamystats": "تفعيل Streamystats",
|
"enable_streamystats": "Enable Streamystats",
|
||||||
"disable_streamystats": "تعطيل Streamystats",
|
"disable_streamystats": "Disable Streamystats",
|
||||||
"enable_search": "استخدم للبحث",
|
"enable_search": "Use for Search",
|
||||||
"url": "الرابط",
|
"url": "URL",
|
||||||
"server_url_placeholder": "http(s)://streamystats.example.com",
|
"server_url_placeholder": "http(s)://streamystats.example.com",
|
||||||
"streamystats_search_hint": "أدخل رابط خادم Streamystats الخاص بك. يجب أن يتضمن الرابط البروتوكول http أو https مع رقم المنفذ اختيارياً.",
|
"streamystats_search_hint": "Enter the URL for your Streamystats server. The URL should include http or https and optionally the port.",
|
||||||
"read_more_about_streamystats": "اقرأ المزيد عن Streamystats.",
|
"read_more_about_streamystats": "Read More About Streamystats.",
|
||||||
"save_button": "حفظ",
|
"save_button": "Save",
|
||||||
"save": "حفظ",
|
"save": "Save",
|
||||||
"features_title": "المميزات",
|
"features_title": "Features",
|
||||||
"home_sections_title": "أقسام الرئيسية",
|
"home_sections_title": "Home Sections",
|
||||||
"enable_movie_recommendations": "توصيات الأفلام",
|
"enable_movie_recommendations": "Movie Recommendations",
|
||||||
"enable_series_recommendations": "توصيات المسلسلات",
|
"enable_series_recommendations": "Series Recommendations",
|
||||||
"enable_promoted_watchlists": "قوائم مشاهدة مختارة",
|
"enable_promoted_watchlists": "Promoted Watchlists",
|
||||||
"hide_watchlists_tab": "إخفاء تبويب قوائم المشاهدة",
|
"hide_watchlists_tab": "Hide Watchlists Tab",
|
||||||
"home_sections_hint": "إظهار التوصيات المخصصة وقوائم المشاهدة المختارة من Streamystats في الصفحة الرئيسية.",
|
"home_sections_hint": "Show personalized recommendations and promoted watchlists from Streamystats on the home page.",
|
||||||
"recommended_movies": "أفلام موصى بها",
|
"recommended_movies": "Recommended Movies",
|
||||||
"recommended_series": "مسلسلات موصى بها",
|
"recommended_series": "Recommended Series",
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"saved": "تم الحفظ",
|
"saved": "Saved",
|
||||||
"refreshed": "تم تحديث الإعدادات من الخادم",
|
"refreshed": "Settings refreshed from server",
|
||||||
"disabled": "تم تعطيل Streamystats"
|
"disabled": "Streamystats disabled"
|
||||||
},
|
},
|
||||||
"refresh_from_server": "تحديث الإعدادات من الخادم"
|
"refresh_from_server": "Refresh Settings from Server"
|
||||||
},
|
},
|
||||||
"kefinTweaks": {
|
"kefinTweaks": {
|
||||||
"watchlist_enabler": "تفعيل الربط مع قائمة المشاهدة الخاصة بنا",
|
"watchlist_enabler": "Enable our Watchlist integration",
|
||||||
"watchlist_button": "تبديل حالة ربط قائمة المشاهدة"
|
"watchlist_button": "Toggle Watchlist integration"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"storage": {
|
"storage": {
|
||||||
@@ -455,21 +398,15 @@
|
|||||||
"device_usage": "الجهاز {{availableSpace}}%",
|
"device_usage": "الجهاز {{availableSpace}}%",
|
||||||
"size_used": "تم استخدام {{used}} من {{total}}",
|
"size_used": "تم استخدام {{used}} من {{total}}",
|
||||||
"delete_all_downloaded_files": "حذف جميع الملفات التي تم تنزيلها",
|
"delete_all_downloaded_files": "حذف جميع الملفات التي تم تنزيلها",
|
||||||
"music_cache_title": "التخزين المؤقت للموسيقى",
|
"music_cache_title": "Music Cache",
|
||||||
"music_cache_description": "تخزين الأغاني تلقائياً أثناء الاستماع لضمان تشغيل أكثر سلاسة ودعم الاستماع بدون اتصال",
|
"music_cache_description": "Automatically cache songs as you listen for smoother playback and offline support",
|
||||||
"enable_music_cache": "تمكين التخزين المؤقت للموسيقى",
|
"enable_music_cache": "Enable Music Cache",
|
||||||
"clear_music_cache": "مسح التخزين المؤقت للموسيقى",
|
"clear_music_cache": "Clear Music Cache",
|
||||||
"music_cache_size": "تم تخزين {{size}} مؤقتاً",
|
"music_cache_size": "{{size}} cached",
|
||||||
"music_cache_cleared": "تم مسح التخزين المؤقت للموسيقى",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "حذف جميع الأغاني التي تم تنزيلها",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "تم تنزيل {{size}}",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "تم حذف الأغاني التي تم تنزيلها",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "المقدمة",
|
"title": "المقدمة",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "خطأ في حذف الملفات",
|
"error_deleting_files": "خطأ في حذف الملفات",
|
||||||
"background_downloads_enabled": "تم تفعيل التنزيلات في الخلفية",
|
"background_downloads_enabled": "تم تفعيل التنزيلات في الخلفية",
|
||||||
"background_downloads_disabled": "تم تعطيل التنزيلات في الخلفية"
|
"background_downloads_disabled": "تم تعطيل التنزيلات في الخلفية"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "يتطلب التحديث الجديد تنزيل المحتوى مرة أخرى. يرجى إزالة كل المحتوى الذي تم تنزيله والمحاولة مرة أخرى.",
|
"new_app_version_requires_re_download_description": "يتطلب التحديث الجديد تنزيل المحتوى مرة أخرى. يرجى إزالة كل المحتوى الذي تم تنزيله والمحاولة مرة أخرى.",
|
||||||
"back": "رجوع",
|
"back": "رجوع",
|
||||||
"delete": "حذف",
|
"delete": "حذف",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "حدث خطأ ما",
|
"something_went_wrong": "حدث خطأ ما",
|
||||||
"could_not_get_stream_url_from_jellyfin": "تعذر الحصول على رابط البث من Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "تعذر الحصول على رابط البث من Jellyfin",
|
||||||
"eta": "الوقت المتبقي {{eta}}",
|
"eta": "الوقت المتبقي {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "اختر",
|
"select": "اختر",
|
||||||
"no_trailer_available": "لا يوجد مقطع دعائي متوفر",
|
"no_trailer_available": "لا يوجد مقطع دعائي متوفر",
|
||||||
"video": "فيديو",
|
"video": "فيديو",
|
||||||
"audio": "الصوت",
|
"audio": "الصوت",
|
||||||
"subtitle": "الترجمة",
|
"subtitle": "الترجمة",
|
||||||
"play": "تشغيل",
|
"play": "تشغيل",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "لا شيء",
|
"none": "لا شيء",
|
||||||
"track": "أغنية",
|
"track": "Track",
|
||||||
"cancel": "إلغاء",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
"delete": "Delete",
|
||||||
"delete": "حذف",
|
"ok": "OK",
|
||||||
"ok": "حسناً",
|
"remove": "Remove",
|
||||||
"remove": "إزالة",
|
"next": "Next",
|
||||||
"next": "التالي",
|
"back": "Back",
|
||||||
"back": "رجوع",
|
"continue": "Continue",
|
||||||
"continue": "متابعة",
|
"verifying": "Verifying..."
|
||||||
"verifying": "جارٍ التحقق...",
|
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "بحث...",
|
"search": "بحث...",
|
||||||
@@ -606,10 +521,10 @@
|
|||||||
"episodes": "حلقات",
|
"episodes": "حلقات",
|
||||||
"collections": "مجموعات",
|
"collections": "مجموعات",
|
||||||
"actors": "ممثلون",
|
"actors": "ممثلون",
|
||||||
"artists": "الفنانون",
|
"artists": "Artists",
|
||||||
"albums": "الألبومات",
|
"albums": "Albums",
|
||||||
"songs": "الأغاني",
|
"songs": "Songs",
|
||||||
"playlists": "قوائم التشغيل",
|
"playlists": "Playlists",
|
||||||
"request_movies": "طلب أفلام",
|
"request_movies": "طلب أفلام",
|
||||||
"request_series": "طلب مسلسلات",
|
"request_series": "طلب مسلسلات",
|
||||||
"recently_added": "أضيف مؤخرًا",
|
"recently_added": "أضيف مؤخرًا",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "أفلام",
|
"movies": "أفلام",
|
||||||
"series": "مسلسلات",
|
"series": "مسلسلات",
|
||||||
"boxsets": "مجموعات",
|
"boxsets": "مجموعات",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "عناصر"
|
"items": "عناصر"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,20 +566,15 @@
|
|||||||
"poster": "ملصق",
|
"poster": "ملصق",
|
||||||
"cover": "غلاف",
|
"cover": "غلاف",
|
||||||
"show_titles": "إظهار العناوين",
|
"show_titles": "إظهار العناوين",
|
||||||
"show_stats": "إظهار الإحصائيات",
|
"show_stats": "إظهار الإحصائيات"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "الأنواع",
|
"genres": "الأنواع",
|
||||||
"years": "السنوات",
|
"years": "السنوات",
|
||||||
"sort_by": "ترتيب حسب",
|
"sort_by": "ترتيب حسب",
|
||||||
"filter_by": "تصفية حسب",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "اتجاه الترتيب",
|
"sort_order": "اتجاه الترتيب",
|
||||||
"tags": "الوسوم",
|
"tags": "الوسوم"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "لا توجد روابط"
|
"no_links": "لا توجد روابط"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "خطأ",
|
"error": "خطأ",
|
||||||
"failed_to_get_stream_url": "فشل في الحصول على رابط البث",
|
"failed_to_get_stream_url": "فشل في الحصول على رابط البث",
|
||||||
"an_error_occured_while_playing_the_video": "حدث خطأ أثناء تشغيل الفيديو. تحقق من السجلات في الإعدادات.",
|
"an_error_occured_while_playing_the_video": "حدث خطأ أثناء تشغيل الفيديو. تحقق من السجلات في الإعدادات.",
|
||||||
@@ -697,39 +604,11 @@
|
|||||||
"index": "الفِهْرِس:",
|
"index": "الفِهْرِس:",
|
||||||
"continue_watching": "متابعة المشاهدة",
|
"continue_watching": "متابعة المشاهدة",
|
||||||
"go_back": "رجوع",
|
"go_back": "رجوع",
|
||||||
"downloaded_file_title": "تم تنزيل هذا الملف",
|
"downloaded_file_title": "You have this file downloaded",
|
||||||
"downloaded_file_message": "هل تريد تشغيل الملف الذي تم تنزيله؟",
|
"downloaded_file_message": "هل تريد تشغيل الملف الذي تم تنزيله؟",
|
||||||
"downloaded_file_yes": "نعم",
|
"downloaded_file_yes": "نعم",
|
||||||
"downloaded_file_no": "لا",
|
"downloaded_file_no": "لا",
|
||||||
"downloaded_file_cancel": "إلغاء",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "التالي",
|
"next_up": "التالي",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "مسلسلات",
|
"series": "مسلسلات",
|
||||||
"seasons": "مواسم",
|
"seasons": "مواسم",
|
||||||
"season": "موسم",
|
"season": "موسم",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "لا توجد حلقات لهذا الموسم",
|
"no_episodes_for_this_season": "لا توجد حلقات لهذا الموسم",
|
||||||
"overview": "نظرة عامة",
|
"overview": "نظرة عامة",
|
||||||
"more_with": "المزيد مع {{name}}",
|
"more_with": "المزيد مع {{name}}",
|
||||||
@@ -750,24 +624,13 @@
|
|||||||
"no_similar_items_found": "لم يتم العثور على عناصر مشابهة",
|
"no_similar_items_found": "لم يتم العثور على عناصر مشابهة",
|
||||||
"video": "فيديو",
|
"video": "فيديو",
|
||||||
"more_details": "المزيد من التفاصيل",
|
"more_details": "المزيد من التفاصيل",
|
||||||
"media_options": "خيارات الوسائط",
|
"media_options": "Media Options",
|
||||||
"quality": "الجودة",
|
"quality": "الجودة",
|
||||||
"audio": "الصوت",
|
"audio": "الصوت",
|
||||||
"subtitles": {
|
"subtitles": "الترجمة",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "عرض المزيد",
|
"show_more": "عرض المزيد",
|
||||||
"show_less": "عرض أقل",
|
"show_less": "عرض أقل",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "ظهر في",
|
"appeared_in": "ظهر في",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "تعذر تحميل العنصر",
|
"could_not_load_item": "تعذر تحميل العنصر",
|
||||||
"none": "لا شيء",
|
"none": "لا شيء",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "تنزيل {{item_count}} عناصر",
|
"download_x_item": "تنزيل {{item_count}} عناصر",
|
||||||
"download_unwatched_only": "غير المشاهدة فقط",
|
"download_unwatched_only": "غير المشاهدة فقط",
|
||||||
"download_button": "تنزيل"
|
"download_button": "تنزيل"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "التالي",
|
"next": "التالي",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "أفلام",
|
"movies": "أفلام",
|
||||||
"sports": "رياضة",
|
"sports": "رياضة",
|
||||||
"for_kids": "للأطفال",
|
"for_kids": "للأطفال",
|
||||||
"news": "أخبار",
|
"news": "أخبار"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "تأكيد",
|
"confirm": "تأكيد",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "رفض",
|
"decline": "رفض",
|
||||||
"requested_by": "مطلوب من {{user}}",
|
"requested_by": "مطلوب من {{user}}",
|
||||||
"unknown_user": "مستخدم غير معروف",
|
"unknown_user": "مستخدم غير معروف",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "خادم Seerr لا يستوفي الحد الأدنى للإصدار المطلوب! يرجى التحديث إلى إصدار 2.0.0 على الأقل",
|
"jellyseer_does_not_meet_requirements": "خادم Seerr لا يستوفي الحد الأدنى للإصدار المطلوب! يرجى التحديث إلى إصدار 2.0.0 على الأقل",
|
||||||
"jellyseerr_test_failed": "فشل اختبار Seerr. يرجى المحاولة مرة أخرى.",
|
"jellyseerr_test_failed": "فشل اختبار Seerr. يرجى المحاولة مرة أخرى.",
|
||||||
@@ -876,162 +716,130 @@
|
|||||||
"search": "بحث",
|
"search": "بحث",
|
||||||
"library": "المكتبة",
|
"library": "المكتبة",
|
||||||
"custom_links": "روابط مخصصة",
|
"custom_links": "روابط مخصصة",
|
||||||
"favorites": "المفضلة",
|
"favorites": "المفضلة"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "الموسيقى",
|
"title": "Music",
|
||||||
"tabs": {
|
"tabs": {
|
||||||
"suggestions": "الإقتراحات",
|
"suggestions": "Suggestions",
|
||||||
"albums": "الألبومات",
|
"albums": "Albums",
|
||||||
"artists": "الفنانون",
|
"artists": "Artists",
|
||||||
"playlists": "قوائم التشغيل",
|
"playlists": "Playlists",
|
||||||
"tracks": "الأغاني"
|
"tracks": "tracks"
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"all": "الكل"
|
"all": "All"
|
||||||
},
|
},
|
||||||
"recently_added": "أضيف مؤخرًا",
|
"recently_added": "Recently Added",
|
||||||
"recently_played": "تم تشغيله مؤخرًا",
|
"recently_played": "Recently Played",
|
||||||
"frequently_played": "الأكثر تشغيلاً",
|
"frequently_played": "Frequently Played",
|
||||||
"explore": "اكتشف",
|
"explore": "Explore",
|
||||||
"top_tracks": "أفضل الأغاني",
|
"top_tracks": "Top Tracks",
|
||||||
"play": "تشغيل",
|
"play": "Play",
|
||||||
"shuffle": "ترتيب عشوائي",
|
"shuffle": "Shuffle",
|
||||||
"play_top_tracks": "تشغيل أفضل الأغاني",
|
"play_top_tracks": "Play Top Tracks",
|
||||||
"no_suggestions": "لا توجد مقترحات متاحة",
|
"no_suggestions": "No suggestions available",
|
||||||
"no_albums": "لا توجد ألبومات",
|
"no_albums": "No albums found",
|
||||||
"no_artists": "لا يوجد فنانون",
|
"no_artists": "No artists found",
|
||||||
"no_playlists": "لا توجد قوائم تشغيل",
|
"no_playlists": "No playlists found",
|
||||||
"album_not_found": "الألبوم غير موجود",
|
"album_not_found": "Album not found",
|
||||||
"artist_not_found": "الفنان غير موجود",
|
"artist_not_found": "Artist not found",
|
||||||
"playlist_not_found": "قائمة التشغيل غير موجودة",
|
"playlist_not_found": "Playlist not found",
|
||||||
"track_options": {
|
"track_options": {
|
||||||
"play_next": "تشغيل التالي",
|
"play_next": "Play Next",
|
||||||
"add_to_queue": "إضافة إلى قائمة الانتظار",
|
"add_to_queue": "Add to Queue",
|
||||||
"add_to_playlist": "أضف إلى قائمة التشغيل",
|
"add_to_playlist": "Add to Playlist",
|
||||||
"download": "تنزيل",
|
"download": "Download",
|
||||||
"downloaded": "تم التنزيل",
|
"downloaded": "Downloaded",
|
||||||
"downloading": "جارٍ التنزيل...",
|
"downloading": "Downloading...",
|
||||||
"cached": "تم التخزين مؤقتاً",
|
"cached": "Cached",
|
||||||
"delete_download": "حذف ملف التنزيل",
|
"delete_download": "Delete Download",
|
||||||
"delete_cache": "إزالة من التخزين المؤقت",
|
"delete_cache": "Remove from Cache",
|
||||||
"go_to_artist": "انتقال إلى الفنان",
|
"go_to_artist": "Go to Artist",
|
||||||
"go_to_album": "انتقال إلى الألبوم",
|
"go_to_album": "Go to Album",
|
||||||
"add_to_favorites": "إضافة إلى المفضلة",
|
"add_to_favorites": "Add to Favorites",
|
||||||
"remove_from_favorites": "إزالة من المفضلة",
|
"remove_from_favorites": "Remove from Favorites",
|
||||||
"remove_from_playlist": "إزالة من قائمة التشغيل"
|
"remove_from_playlist": "Remove from Playlist"
|
||||||
},
|
},
|
||||||
"playlists": {
|
"playlists": {
|
||||||
"create_playlist": "إنشاء قائمة التشغيل",
|
"create_playlist": "Create Playlist",
|
||||||
"playlist_name": "اسم قائمة التشغيل",
|
"playlist_name": "Playlist Name",
|
||||||
"enter_name": "أدخل اسم قائمة التشغيل",
|
"enter_name": "Enter playlist name",
|
||||||
"create": "إنشاء",
|
"create": "Create",
|
||||||
"search_playlists": "البحث عن قوائم التشغيل...",
|
"search_playlists": "Search playlists...",
|
||||||
"added_to": "تمت الإضافة إلى {{name}}",
|
"added_to": "Added to {{name}}",
|
||||||
"added": "تمت الإضافة إلى قائمة التشغيل",
|
"added": "Added to playlist",
|
||||||
"removed_from": "تمت الإزالة من {{name}}",
|
"removed_from": "Removed from {{name}}",
|
||||||
"removed": "تمت الازالة من قائمة التشغيل",
|
"removed": "Removed from playlist",
|
||||||
"created": "تم إنشاء قائمة التشغيل",
|
"created": "Playlist created",
|
||||||
"create_new": "إنشاء قائمة تشغيل جديدة",
|
"create_new": "Create New Playlist",
|
||||||
"failed_to_add": "فشلت الإضافة إلى قائمة التشغيل",
|
"failed_to_add": "Failed to add to playlist",
|
||||||
"failed_to_remove": "فشلت الإزالة من قائمة التشغيل",
|
"failed_to_remove": "Failed to remove from playlist",
|
||||||
"failed_to_create": "فشل إنشاء قائمة التشغيل",
|
"failed_to_create": "Failed to create playlist",
|
||||||
"delete_playlist": "حذف قائمة التشغيل",
|
"delete_playlist": "Delete Playlist",
|
||||||
"delete_confirm": "هل أنت متأكد من رغبتك في حذف {{name}}؟ لا يمكن التراجع عن هذا الإجراء.",
|
"delete_confirm": "Are you sure you want to delete \"{{name}}\"? This action cannot be undone.",
|
||||||
"deleted": "تم حذف قائمة التشغيل",
|
"deleted": "Playlist deleted",
|
||||||
"failed_to_delete": "فشل إنشاء قائمة التشغيل"
|
"failed_to_delete": "Failed to delete playlist"
|
||||||
},
|
},
|
||||||
"sort": {
|
"sort": {
|
||||||
"title": "ترتيب حسب",
|
"title": "Sort By",
|
||||||
"alphabetical": "أبجدي",
|
"alphabetical": "Alphabetical",
|
||||||
"date_created": "تاريخ الإنشاء"
|
"date_created": "Date Created"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"watchlists": {
|
"watchlists": {
|
||||||
"title": "قوائم المشاهدة",
|
"title": "Watchlists",
|
||||||
"my_watchlists": "قوائم المشاهدة الخاصة بي",
|
"my_watchlists": "My Watchlists",
|
||||||
"public_watchlists": "قوائم مشاهدة عامة",
|
"public_watchlists": "Public Watchlists",
|
||||||
"create_title": "إنشاء قائمة مشاهدة",
|
"create_title": "Create Watchlist",
|
||||||
"edit_title": "تعديل قائمة المشاهدة",
|
"edit_title": "Edit Watchlist",
|
||||||
"create_button": "إنشاء قائمة مشاهدة",
|
"create_button": "Create Watchlist",
|
||||||
"save_button": "حفظ التغييرات",
|
"save_button": "Save Changes",
|
||||||
"delete_button": "حذف",
|
"delete_button": "Delete",
|
||||||
"remove_button": "إزالة",
|
"remove_button": "Remove",
|
||||||
"cancel_button": "إلغاء",
|
"cancel_button": "Cancel",
|
||||||
"name_label": "الاسم",
|
"name_label": "Name",
|
||||||
"name_placeholder": "أدخل اسم قائمة المشاهدة",
|
"name_placeholder": "Enter watchlist name",
|
||||||
"description_label": "الوصف",
|
"description_label": "Description",
|
||||||
"description_placeholder": "أدخل الوصف (اختياري)",
|
"description_placeholder": "Enter description (optional)",
|
||||||
"is_public_label": "قائمة مشاهدة عامة",
|
"is_public_label": "Public Watchlist",
|
||||||
"is_public_description": "السماح للآخرين بعرض قائمة المشاهدة هذه",
|
"is_public_description": "Allow others to view this watchlist",
|
||||||
"allowed_type_label": "نوع المحتوى",
|
"allowed_type_label": "Content Type",
|
||||||
"sort_order_label": "الترتيب الافتراضي",
|
"sort_order_label": "Default Sort Order",
|
||||||
"empty_title": "لا توجد قوائم مشاهدة",
|
"empty_title": "No Watchlists",
|
||||||
"empty_description": "قم بإنشاء أول قائمة مشاهدة لبدء تنظيم الوسائط الخاصة بك",
|
"empty_description": "Create your first watchlist to start organizing your media",
|
||||||
"empty_watchlist": "قائمة المشاهدة هذه فارغة",
|
"empty_watchlist": "This watchlist is empty",
|
||||||
"empty_watchlist_hint": "إضافة عناصر من مكتبتك إلى قائمة المشاهدة هذه",
|
"empty_watchlist_hint": "Add items from your library to this watchlist",
|
||||||
"not_configured_title": "لم يتم ضبط Streamystats",
|
"not_configured_title": "Streamystats Not Configured",
|
||||||
"not_configured_description": "اضبط Streamystats في الإعدادات لاستخدام قوائم المشاهدة",
|
"not_configured_description": "Configure Streamystats in settings to use watchlists",
|
||||||
"go_to_settings": "الذهاب إلى الإعدادات",
|
"go_to_settings": "Go to Settings",
|
||||||
"add_to_watchlist": "إضافة إلى قائمة المشاهدة",
|
"add_to_watchlist": "Add to Watchlist",
|
||||||
"remove_from_watchlist": "إزالة من قائمة المشاهدة",
|
"remove_from_watchlist": "Remove from Watchlist",
|
||||||
"select_watchlist": "تحديد قائمة المشاهدة",
|
"select_watchlist": "Select Watchlist",
|
||||||
"create_new": "إنشاء قائمة مشاهدة جديدة",
|
"create_new": "Create New Watchlist",
|
||||||
"item": "عنصر",
|
"item": "item",
|
||||||
"items": "عناصر",
|
"items": "items",
|
||||||
"public": "عامة",
|
"public": "Public",
|
||||||
"private": "خاصة",
|
"private": "Private",
|
||||||
"you": "أنت",
|
"you": "You",
|
||||||
"by_owner": "بواسطة مستخدم آخر",
|
"by_owner": "By another user",
|
||||||
"not_found": "قائمة المشاهدة غير موجودة",
|
"not_found": "Watchlist not found",
|
||||||
"delete_confirm_title": "حذف قائمة المشاهدة",
|
"delete_confirm_title": "Delete Watchlist",
|
||||||
"delete_confirm_message": "هل أنت متأكد من رغبتك في حذف \"{{name}}\"؟ لا يمكن التراجع عن هذا الإجراء.",
|
"delete_confirm_message": "Are you sure you want to delete \"{{name}}\"? This action cannot be undone.",
|
||||||
"remove_item_title": "إزالة من قائمة المشاهدة",
|
"remove_item_title": "Remove from Watchlist",
|
||||||
"remove_item_message": "إزالة \"{{name}}\" من قائمة المشاهدة هذه؟",
|
"remove_item_message": "Remove \"{{name}}\" from this watchlist?",
|
||||||
"loading": "تحميل قوائم المشاهدة...",
|
"loading": "Loading watchlists...",
|
||||||
"no_compatible_watchlists": "لا توجد قوائم مشاهدة متوافقة",
|
"no_compatible_watchlists": "No compatible watchlists",
|
||||||
"create_one_first": "إنشاء قائمة مشاهدة تقبل نوع المحتوى هذا"
|
"create_one_first": "Create a watchlist that accepts this content type"
|
||||||
},
|
},
|
||||||
"playback_speed": {
|
"playback_speed": {
|
||||||
"title": "سرعة التشغيل",
|
"title": "Playback Speed",
|
||||||
"apply_to": "تطبيق على",
|
"apply_to": "Apply To",
|
||||||
"speed": "السرعة",
|
"speed": "Speed",
|
||||||
"scope": {
|
"scope": {
|
||||||
"media": "الوسائط هذه فقط",
|
"media": "This media only",
|
||||||
"show": "هذا المسلسل",
|
"show": "This show",
|
||||||
"all": "جميع الوسائط (الافتراضي)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Error",
|
"error_title": "Error",
|
||||||
"login_title": "Inicia sessió",
|
"login_title": "Inicia sessió",
|
||||||
"login_to_title": "Inicia sessió a",
|
"login_to_title": "Inicia sessió a",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Nom d'usuari",
|
"username_placeholder": "Nom d'usuari",
|
||||||
"password_placeholder": "Contrasenya",
|
"password_placeholder": "Contrasenya",
|
||||||
"login_button": "Inicia sessió",
|
"login_button": "Inicia sessió",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Oops!",
|
"oops": "Oops!",
|
||||||
"error_message": "Alguna cosa ha anat malament.\nTanqueu la sessió i torneu-la a iniciar.",
|
"error_message": "Alguna cosa ha anat malament.\nTanqueu la sessió i torneu-la a iniciar.",
|
||||||
"continue_watching": "Continua veient",
|
"continue_watching": "Continua veient",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "A continuació",
|
"next_up": "A continuació",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Afegit recentment a {{libraryName}}",
|
"recently_added_in": "Afegit recentment a {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Configuració",
|
"settings_title": "Configuració",
|
||||||
"log_out_button": "Tanca sessió",
|
"log_out_button": "Tanca sessió",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categories"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Durada del rebobinat",
|
"rewind_length": "Durada del rebobinat",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Gesture Controls",
|
"gesture_controls_title": "Gesture Controls",
|
||||||
"horizontal_swipe_skip": "Horizontal Swipe to Skip",
|
"horizontal_swipe_skip": "Horizontal Swipe to Skip",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Error en suprimir fitxers",
|
"error_deleting_files": "Error en suprimir fitxers",
|
||||||
"background_downloads_enabled": "Descàrregues en segon pla activades",
|
"background_downloads_enabled": "Descàrregues en segon pla activades",
|
||||||
"background_downloads_disabled": "Descàrregues en segon pla desactivades"
|
"background_downloads_disabled": "Descàrregues en segon pla desactivades"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "L'actualització nova requereix que el contingut es torni a descarregar. Suprimiu tot el contingut descarregat i torneu-ho a provar.",
|
"new_app_version_requires_re_download_description": "L'actualització nova requereix que el contingut es torni a descarregar. Suprimiu tot el contingut descarregat i torneu-ho a provar.",
|
||||||
"back": "Enrere",
|
"back": "Enrere",
|
||||||
"delete": "Suprimeix",
|
"delete": "Suprimeix",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Alguna cosa ha anat malament",
|
"something_went_wrong": "Alguna cosa ha anat malament",
|
||||||
"could_not_get_stream_url_from_jellyfin": "No s'ha pogut obtenir l'URL del flux de Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "No s'ha pogut obtenir l'URL del flux de Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Select",
|
"select": "Select",
|
||||||
"no_trailer_available": "No trailer available",
|
"no_trailer_available": "No trailer available",
|
||||||
"video": "Vídeo",
|
"video": "Vídeo",
|
||||||
"audio": "Àudio",
|
"audio": "Àudio",
|
||||||
"subtitle": "Subtítols",
|
"subtitle": "Subtítols",
|
||||||
"play": "Play",
|
"play": "Play",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Cerca...",
|
"search": "Cerca...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "pel·lícules",
|
"movies": "pel·lícules",
|
||||||
"series": "sèries",
|
"series": "sèries",
|
||||||
"boxsets": "col·leccions",
|
"boxsets": "col·leccions",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "elements"
|
"items": "elements"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Cartell",
|
"poster": "Cartell",
|
||||||
"cover": "Coberta",
|
"cover": "Coberta",
|
||||||
"show_titles": "Mostrar títols",
|
"show_titles": "Mostrar títols",
|
||||||
"show_stats": "Mostrar estadístiques",
|
"show_stats": "Mostrar estadístiques"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Gèneres",
|
"genres": "Gèneres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Ordenar per",
|
"sort_by": "Ordenar per",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Ordre",
|
"sort_order": "Ordre",
|
||||||
"tags": "Etiquetes",
|
"tags": "Etiquetes"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "No hi ha enllaços"
|
"no_links": "No hi ha enllaços"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Error",
|
"error": "Error",
|
||||||
"failed_to_get_stream_url": "No s'ha pogut obtenir l'URL del flux",
|
"failed_to_get_stream_url": "No s'ha pogut obtenir l'URL del flux",
|
||||||
"an_error_occured_while_playing_the_video": "S'ha produït un error en reproduir el vídeo. Consulteu els registres a la configuració.",
|
"an_error_occured_while_playing_the_video": "S'ha produït un error en reproduir el vídeo. Consulteu els registres a la configuració.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "A continuació",
|
"next_up": "A continuació",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Sèries",
|
"series": "Sèries",
|
||||||
"seasons": "Temporades",
|
"seasons": "Temporades",
|
||||||
"season": "Temporada",
|
"season": "Temporada",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "No hi ha episodis per a aquesta temporada",
|
"no_episodes_for_this_season": "No hi ha episodis per a aquesta temporada",
|
||||||
"overview": "Descripció general",
|
"overview": "Descripció general",
|
||||||
"more_with": "Més amb {{name}}",
|
"more_with": "Més amb {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Qualitat",
|
"quality": "Qualitat",
|
||||||
"audio": "Àudio",
|
"audio": "Àudio",
|
||||||
"subtitles": {
|
"subtitles": "Subtítols",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Mostra més",
|
"show_more": "Mostra més",
|
||||||
"show_less": "Mostra menys",
|
"show_less": "Mostra menys",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Va aparèixer a",
|
"appeared_in": "Va aparèixer a",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "No s'ha pogut carregar l'element",
|
"could_not_load_item": "No s'ha pogut carregar l'element",
|
||||||
"none": "Cap",
|
"none": "Cap",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Descarrega {{item_count}} elements",
|
"download_x_item": "Descarrega {{item_count}} elements",
|
||||||
"download_unwatched_only": "Unwatched Only",
|
"download_unwatched_only": "Unwatched Only",
|
||||||
"download_button": "Descarrega"
|
"download_button": "Descarrega"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Següent",
|
"next": "Següent",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Pel·lícules",
|
"movies": "Pel·lícules",
|
||||||
"sports": "Esports",
|
"sports": "Esports",
|
||||||
"for_kids": "Infantil",
|
"for_kids": "Infantil",
|
||||||
"news": "Notícies",
|
"news": "Notícies"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Confirma",
|
"confirm": "Confirma",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "El servidor Jellyseerr no compleix els requisits mínims de versió! Actualitzeu-lo almenys a la versió 2.0.0",
|
"jellyseer_does_not_meet_requirements": "El servidor Jellyseerr no compleix els requisits mínims de versió! Actualitzeu-lo almenys a la versió 2.0.0",
|
||||||
"jellyseerr_test_failed": "Ha fallat la prova de Jellyseerr. Torneu-ho a provar.",
|
"jellyseerr_test_failed": "Ha fallat la prova de Jellyseerr. Torneu-ho a provar.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Cercar",
|
"search": "Cercar",
|
||||||
"library": "Biblioteca",
|
"library": "Biblioteca",
|
||||||
"custom_links": "Enllaços personalitzats",
|
"custom_links": "Enllaços personalitzats",
|
||||||
"favorites": "Preferits",
|
"favorites": "Preferits"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Chyba",
|
"error_title": "Chyba",
|
||||||
"login_title": "Přihlásit se",
|
"login_title": "Přihlásit se",
|
||||||
"login_to_title": "Přihlásit se do",
|
"login_to_title": "Přihlásit se do",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Uživatelské jméno",
|
"username_placeholder": "Uživatelské jméno",
|
||||||
"password_placeholder": "Heslo",
|
"password_placeholder": "Heslo",
|
||||||
"login_button": "Přihlásit se",
|
"login_button": "Přihlásit se",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Jejda!",
|
"oops": "Jejda!",
|
||||||
"error_message": "Něco se pokazilo.\nOdhlaste se a znovu se prosím.",
|
"error_message": "Něco se pokazilo.\nOdhlaste se a znovu se prosím.",
|
||||||
"continue_watching": "Pokračovat ve sledování",
|
"continue_watching": "Pokračovat ve sledování",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Další nahoru",
|
"next_up": "Další nahoru",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Nedávno přidané v {{libraryName}}",
|
"recently_added_in": "Nedávno přidané v {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Nastavení",
|
"settings_title": "Nastavení",
|
||||||
"log_out_button": "Odhlásit se",
|
"log_out_button": "Odhlásit se",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categories"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Délka zpětného větru",
|
"rewind_length": "Délka zpětného větru",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Ovládání gest",
|
"gesture_controls_title": "Ovládání gest",
|
||||||
"horizontal_swipe_skip": "Horizontální přejetím přeskočit",
|
"horizontal_swipe_skip": "Horizontální přejetím přeskočit",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Chyba při mazání souborů",
|
"error_deleting_files": "Chyba při mazání souborů",
|
||||||
"background_downloads_enabled": "Stahování na pozadí povoleno",
|
"background_downloads_enabled": "Stahování na pozadí povoleno",
|
||||||
"background_downloads_disabled": "Stahování na pozadí zakázáno"
|
"background_downloads_disabled": "Stahování na pozadí zakázáno"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Nová aktualizace vyžaduje opětovné stažení obsahu. Odstraňte prosím veškerý stažený obsah a zkuste to znovu.",
|
"new_app_version_requires_re_download_description": "Nová aktualizace vyžaduje opětovné stažení obsahu. Odstraňte prosím veškerý stažený obsah a zkuste to znovu.",
|
||||||
"back": "Zpět",
|
"back": "Zpět",
|
||||||
"delete": "Vymazat",
|
"delete": "Vymazat",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Něco se pokazilo",
|
"something_went_wrong": "Něco se pokazilo",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Nelze získat URL streamu z Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Nelze získat URL streamu z Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Vybrat",
|
"select": "Vybrat",
|
||||||
"no_trailer_available": "Přípojné vozidlo není k dispozici",
|
"no_trailer_available": "Přípojné vozidlo není k dispozici",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Zvuk",
|
"audio": "Zvuk",
|
||||||
"subtitle": "Podtitulek",
|
"subtitle": "Podtitulek",
|
||||||
"play": "Hrát",
|
"play": "Hrát",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Hledat...",
|
"search": "Hledat...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Filmy",
|
"movies": "Filmy",
|
||||||
"series": "Série",
|
"series": "Série",
|
||||||
"boxsets": "Sada boxů",
|
"boxsets": "Sada boxů",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Položky"
|
"items": "Položky"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Plakát",
|
"poster": "Plakát",
|
||||||
"cover": "Kryt",
|
"cover": "Kryt",
|
||||||
"show_titles": "Zobrazit názvy",
|
"show_titles": "Zobrazit názvy",
|
||||||
"show_stats": "Zobrazit statistiky",
|
"show_stats": "Zobrazit statistiky"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Seřadit podle",
|
"sort_by": "Seřadit podle",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Řazení",
|
"sort_order": "Řazení",
|
||||||
"tags": "Štítky",
|
"tags": "Štítky"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Žádné odkazy"
|
"no_links": "Žádné odkazy"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Chyba",
|
"error": "Chyba",
|
||||||
"failed_to_get_stream_url": "Nepodařilo se získat URL streamu",
|
"failed_to_get_stream_url": "Nepodařilo se získat URL streamu",
|
||||||
"an_error_occured_while_playing_the_video": "Při přehrávání videa došlo k chybě. Zkontrolujte logy v nastavení.",
|
"an_error_occured_while_playing_the_video": "Při přehrávání videa došlo k chybě. Zkontrolujte logy v nastavení.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Další nahoru",
|
"next_up": "Další nahoru",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Série",
|
"series": "Série",
|
||||||
"seasons": "Série",
|
"seasons": "Série",
|
||||||
"season": "Sezóna",
|
"season": "Sezóna",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Žádné epizody pro tuto sezónu",
|
"no_episodes_for_this_season": "Žádné epizody pro tuto sezónu",
|
||||||
"overview": "Přehled",
|
"overview": "Přehled",
|
||||||
"more_with": "Více s {{name}}",
|
"more_with": "Více s {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Kvalita",
|
"quality": "Kvalita",
|
||||||
"audio": "Zvuk",
|
"audio": "Zvuk",
|
||||||
"subtitles": {
|
"subtitles": "Podtitulek",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Zobrazit více",
|
"show_more": "Zobrazit více",
|
||||||
"show_less": "Zobrazit méně",
|
"show_less": "Zobrazit méně",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Zobrazeno v",
|
"appeared_in": "Zobrazeno v",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Nelze načíst položku",
|
"could_not_load_item": "Nelze načíst položku",
|
||||||
"none": "Nic",
|
"none": "Nic",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Stáhnout položky {{item_count}}",
|
"download_x_item": "Stáhnout položky {{item_count}}",
|
||||||
"download_unwatched_only": "Pouze nezhlédnuté",
|
"download_unwatched_only": "Pouze nezhlédnuté",
|
||||||
"download_button": "Stáhnout"
|
"download_button": "Stáhnout"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Další",
|
"next": "Další",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Filmy",
|
"movies": "Filmy",
|
||||||
"sports": "Sporty",
|
"sports": "Sporty",
|
||||||
"for_kids": "Pro děti",
|
"for_kids": "Pro děti",
|
||||||
"news": "Novinky",
|
"news": "Novinky"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Potvrdit",
|
"confirm": "Potvrdit",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Seerr server nesplňuje minimální požadavky na verzi! Aktualizujte prosím alespoň na 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Seerr server nesplňuje minimální požadavky na verzi! Aktualizujte prosím alespoň na 2.0.0",
|
||||||
"jellyseerr_test_failed": "Seerr test se nezdařil. Zkuste to prosím znovu.",
|
"jellyseerr_test_failed": "Seerr test se nezdařil. Zkuste to prosím znovu.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Hledat",
|
"search": "Hledat",
|
||||||
"library": "Knihovna",
|
"library": "Knihovna",
|
||||||
"custom_links": "Vlastní odkazy",
|
"custom_links": "Vlastní odkazy",
|
||||||
"favorites": "Oblíbené",
|
"favorites": "Oblíbené"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Fejl",
|
"error_title": "Fejl",
|
||||||
"login_title": "Log ind",
|
"login_title": "Log ind",
|
||||||
"login_to_title": "Log ind på",
|
"login_to_title": "Log ind på",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Brugernavn",
|
"username_placeholder": "Brugernavn",
|
||||||
"password_placeholder": "Adgangskode",
|
"password_placeholder": "Adgangskode",
|
||||||
"login_button": "Log ind",
|
"login_button": "Log ind",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Ups!",
|
"oops": "Ups!",
|
||||||
"error_message": "Noget gik galt.\nLog venligst ud og ind igen.",
|
"error_message": "Noget gik galt.\nLog venligst ud og ind igen.",
|
||||||
"continue_watching": "Fortsæt med at se",
|
"continue_watching": "Fortsæt med at se",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Næste",
|
"next_up": "Næste",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Senest tilføjet i {{libraryName}}",
|
"recently_added_in": "Senest tilføjet i {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Indstillinger",
|
"settings_title": "Indstillinger",
|
||||||
"log_out_button": "Log ud",
|
"log_out_button": "Log ud",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categories"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Spol tilbage længde",
|
"rewind_length": "Spol tilbage længde",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Bevægelsesstyring",
|
"gesture_controls_title": "Bevægelsesstyring",
|
||||||
"horizontal_swipe_skip": "Vandret Stryg for at springe over",
|
"horizontal_swipe_skip": "Vandret Stryg for at springe over",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Fejl ved sletning af filer",
|
"error_deleting_files": "Fejl ved sletning af filer",
|
||||||
"background_downloads_enabled": "Baggrundsdownloads aktiveret",
|
"background_downloads_enabled": "Baggrundsdownloads aktiveret",
|
||||||
"background_downloads_disabled": "Baggrundsdownloads deaktiveret"
|
"background_downloads_disabled": "Baggrundsdownloads deaktiveret"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Den nye opdatering kræver, at indhold downloades igen. Fjern venligst alt downloadet indhold og prøv igen.",
|
"new_app_version_requires_re_download_description": "Den nye opdatering kræver, at indhold downloades igen. Fjern venligst alt downloadet indhold og prøv igen.",
|
||||||
"back": "Tilbage",
|
"back": "Tilbage",
|
||||||
"delete": "Slet",
|
"delete": "Slet",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Noget gik galt",
|
"something_went_wrong": "Noget gik galt",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Kunne ikke hente stream URL'en fra Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Kunne ikke hente stream URL'en fra Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Vælg",
|
"select": "Vælg",
|
||||||
"no_trailer_available": "Intet påhængskøretøj tilgængeligt",
|
"no_trailer_available": "Intet påhængskøretøj tilgængeligt",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Lyd",
|
"audio": "Lyd",
|
||||||
"subtitle": "Undertekster",
|
"subtitle": "Undertekster",
|
||||||
"play": "Afspil",
|
"play": "Afspil",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Søg...",
|
"search": "Søg...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "film",
|
"movies": "film",
|
||||||
"series": "serier",
|
"series": "serier",
|
||||||
"boxsets": "box sæt",
|
"boxsets": "box sæt",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "elementer"
|
"items": "elementer"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Plakat",
|
"poster": "Plakat",
|
||||||
"cover": "Omslag",
|
"cover": "Omslag",
|
||||||
"show_titles": "Vis titler",
|
"show_titles": "Vis titler",
|
||||||
"show_stats": "Vis statistik",
|
"show_stats": "Vis statistik"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genrer",
|
"genres": "Genrer",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Sortér efter",
|
"sort_by": "Sortér efter",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Sorteringsrækkefølge",
|
"sort_order": "Sorteringsrækkefølge",
|
||||||
"tags": "Mærker",
|
"tags": "Mærker"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Ingen links"
|
"no_links": "Ingen links"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Fejl",
|
"error": "Fejl",
|
||||||
"failed_to_get_stream_url": "Kunne ikke hente stream URL'en",
|
"failed_to_get_stream_url": "Kunne ikke hente stream URL'en",
|
||||||
"an_error_occured_while_playing_the_video": "Der opstod en fejl under afspilning af videoen. Tjek logfilerne i indstillinger.",
|
"an_error_occured_while_playing_the_video": "Der opstod en fejl under afspilning af videoen. Tjek logfilerne i indstillinger.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Næste",
|
"next_up": "Næste",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Serier",
|
"series": "Serier",
|
||||||
"seasons": "Sæsoner",
|
"seasons": "Sæsoner",
|
||||||
"season": "Sæson",
|
"season": "Sæson",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Ingen episoder for denne sæson",
|
"no_episodes_for_this_season": "Ingen episoder for denne sæson",
|
||||||
"overview": "Oversigt",
|
"overview": "Oversigt",
|
||||||
"more_with": "Mere med {{name}}",
|
"more_with": "Mere med {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Kvalitet",
|
"quality": "Kvalitet",
|
||||||
"audio": "Lyd",
|
"audio": "Lyd",
|
||||||
"subtitles": {
|
"subtitles": "Undertekster",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Vis mere",
|
"show_more": "Vis mere",
|
||||||
"show_less": "Vis mindre",
|
"show_less": "Vis mindre",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Medvirket i",
|
"appeared_in": "Medvirket i",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Kunne ikke indlæse elementet",
|
"could_not_load_item": "Kunne ikke indlæse elementet",
|
||||||
"none": "Ingen",
|
"none": "Ingen",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Download {{item_count}} elementer",
|
"download_x_item": "Download {{item_count}} elementer",
|
||||||
"download_unwatched_only": "Kun Usete",
|
"download_unwatched_only": "Kun Usete",
|
||||||
"download_button": "Hent"
|
"download_button": "Hent"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Næste",
|
"next": "Næste",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Film",
|
"movies": "Film",
|
||||||
"sports": "Sport",
|
"sports": "Sport",
|
||||||
"for_kids": "For børn",
|
"for_kids": "For børn",
|
||||||
"news": "Nyheder",
|
"news": "Nyheder"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Bekræft",
|
"confirm": "Bekræft",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Jellyseerr serveren opfylder ikke minimumskravene! Opdater venligst til mindst 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Jellyseerr serveren opfylder ikke minimumskravene! Opdater venligst til mindst 2.0.0",
|
||||||
"jellyseerr_test_failed": "Jellyseerr test mislykkedes. Prøv venligst igen.",
|
"jellyseerr_test_failed": "Jellyseerr test mislykkedes. Prøv venligst igen.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Søg",
|
"search": "Søg",
|
||||||
"library": "Bibliotek",
|
"library": "Bibliotek",
|
||||||
"custom_links": "Tilpassede links",
|
"custom_links": "Tilpassede links",
|
||||||
"favorites": "Favoritter",
|
"favorites": "Favoritter"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Fehler",
|
"error_title": "Fehler",
|
||||||
"login_title": "Anmelden",
|
"login_title": "Anmelden",
|
||||||
"login_to_title": "Anmelden bei",
|
"login_to_title": "Anmelden bei",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Benutzername",
|
"username_placeholder": "Benutzername",
|
||||||
"password_placeholder": "Passwort",
|
"password_placeholder": "Passwort",
|
||||||
"login_button": "Anmelden",
|
"login_button": "Anmelden",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} Konten",
|
"accounts_count": "{{count}} Konten",
|
||||||
"select_account": "Konto auswählen",
|
"select_account": "Konto auswählen",
|
||||||
"add_account": "Konto hinzufügen",
|
"add_account": "Konto hinzufügen",
|
||||||
"remove_account_description": "Hiermit werden die gespeicherten Zugangsdaten für {{username}} entfernt.",
|
"remove_account_description": "Hiermit werden die gespeicherten Zugangsdaten für {{username}} entfernt."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Konto speichern",
|
"title": "Konto speichern",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Ups!",
|
"oops": "Ups!",
|
||||||
"error_message": "Etwas ist schiefgelaufen.\nBitte melde dich ab und wieder an.",
|
"error_message": "Etwas ist schiefgelaufen.\nBitte melde dich ab und wieder an.",
|
||||||
"continue_watching": "Weiterschauen",
|
"continue_watching": "Weiterschauen",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Als nächstes",
|
"next_up": "Als nächstes",
|
||||||
"continue_and_next_up": "\"Weiterschauen\" und \"Als Nächstes\"",
|
"continue_and_next_up": "\"Weiterschauen\" und \"Als Nächstes\"",
|
||||||
"recently_added_in": "Kürzlich hinzugefügt in {{libraryName}}",
|
"recently_added_in": "Kürzlich hinzugefügt in {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Einstellungen",
|
"settings_title": "Einstellungen",
|
||||||
"log_out_button": "Abmelden",
|
"log_out_button": "Abmelden",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Kategorien"
|
"title": "Kategorien"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Aussehen",
|
"title": "Aussehen",
|
||||||
"merge_next_up_continue_watching": "\"Weiterschauen\" und \"Als Nächstes\" kombinieren",
|
"merge_next_up_continue_watching": "\"Weiterschauen\" und \"Als Nächstes\" kombinieren",
|
||||||
"hide_remote_session_button": "Button für Remote-Sitzung ausblenden",
|
"hide_remote_session_button": "Button für Remote-Sitzung ausblenden"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Netzwerk",
|
"title": "Netzwerk",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Rückspullänge",
|
"rewind_length": "Rückspullänge",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Gestensteuerung",
|
"gesture_controls_title": "Gestensteuerung",
|
||||||
"horizontal_swipe_skip": "Horizontal Wischen zum Überspringen",
|
"horizontal_swipe_skip": "Horizontal Wischen zum Überspringen",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Untertitel-Schriftart",
|
"subtitle_font": "Untertitel-Schriftart",
|
||||||
"ksplayer_title": "KSPlayer Einstellungen",
|
"ksplayer_title": "KSPlayer Einstellungen",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Hardwarebeschleunigung für Video Decoding verwenden. Deaktivieren wenn Wiedergabeprobleme auftreten.",
|
"hardware_decode_description": "Hardwarebeschleunigung für Video Decoding verwenden. Deaktivieren wenn Wiedergabeprobleme auftreten."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Untertitel-Einstellungen",
|
"title": "VLC Untertitel-Einstellungen",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Musik-Cache geleert",
|
"music_cache_cleared": "Musik-Cache geleert",
|
||||||
"delete_all_downloaded_songs": "Alle heruntergeladenen Titel löschen",
|
"delete_all_downloaded_songs": "Alle heruntergeladenen Titel löschen",
|
||||||
"downloaded_songs_size": "{{size}} heruntergeladen",
|
"downloaded_songs_size": "{{size}} heruntergeladen",
|
||||||
"downloaded_songs_deleted": "Heruntergeladene Titel gelöscht",
|
"downloaded_songs_deleted": "Heruntergeladene Titel gelöscht"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Einführung",
|
"title": "Einführung",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Fehler beim Löschen von Dateien",
|
"error_deleting_files": "Fehler beim Löschen von Dateien",
|
||||||
"background_downloads_enabled": "Hintergrunddownloads aktiviert",
|
"background_downloads_enabled": "Hintergrunddownloads aktiviert",
|
||||||
"background_downloads_disabled": "Hintergrunddownloads deaktiviert"
|
"background_downloads_disabled": "Hintergrunddownloads deaktiviert"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Die neue App-Version erfordert das erneute Herunterladen von Filmen und Serien. Bitte lösche alle heruntergeladenen Elemente und starte den Download erneut.",
|
"new_app_version_requires_re_download_description": "Die neue App-Version erfordert das erneute Herunterladen von Filmen und Serien. Bitte lösche alle heruntergeladenen Elemente und starte den Download erneut.",
|
||||||
"back": "Zurück",
|
"back": "Zurück",
|
||||||
"delete": "Löschen",
|
"delete": "Löschen",
|
||||||
"delete_download": "Download löschen",
|
|
||||||
"something_went_wrong": "Etwas ist schiefgelaufen",
|
"something_went_wrong": "Etwas ist schiefgelaufen",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Konnte keine Stream-URL von Jellyfin erhalten",
|
"could_not_get_stream_url_from_jellyfin": "Konnte keine Stream-URL von Jellyfin erhalten",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Auswählen",
|
"select": "Auswählen",
|
||||||
"no_trailer_available": "Kein Trailer verfügbar",
|
"no_trailer_available": "Kein Trailer verfügbar",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Untertitel",
|
"subtitle": "Untertitel",
|
||||||
"play": "Abspielen",
|
"play": "Abspielen",
|
||||||
"mark_as_played": "Als gesehen markieren",
|
|
||||||
"mark_as_not_played": "Als ungesehen markieren",
|
|
||||||
"none": "Keine",
|
"none": "Keine",
|
||||||
"track": "Spur",
|
"track": "Spur",
|
||||||
"cancel": "Abbrechen",
|
"cancel": "Abbrechen",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Löschen",
|
"delete": "Löschen",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Entfernen",
|
"remove": "Entfernen",
|
||||||
"next": "Weiter",
|
"next": "Weiter",
|
||||||
"back": "Zurück",
|
"back": "Zurück",
|
||||||
"continue": "Fortsetzen",
|
"continue": "Fortsetzen",
|
||||||
"verifying": "Verifiziere...",
|
"verifying": "Verifiziere..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Suchen...",
|
"search": "Suchen...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Filme",
|
"movies": "Filme",
|
||||||
"series": "Serien",
|
"series": "Serien",
|
||||||
"boxsets": "Boxsets",
|
"boxsets": "Boxsets",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Elemente"
|
"items": "Elemente"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Poster",
|
"poster": "Poster",
|
||||||
"cover": "Cover",
|
"cover": "Cover",
|
||||||
"show_titles": "Titel anzeigen",
|
"show_titles": "Titel anzeigen",
|
||||||
"show_stats": "Statistiken anzeigen",
|
"show_stats": "Statistiken anzeigen"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Sortieren nach",
|
"sort_by": "Sortieren nach",
|
||||||
"filter_by": "Filtern nach",
|
"filter_by": "Filtern nach",
|
||||||
"sort_order": "Sortierreihenfolge",
|
"sort_order": "Sortierreihenfolge",
|
||||||
"tags": "Tags",
|
"tags": "Tags"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Keine Links"
|
"no_links": "Keine Links"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Fehler",
|
"error": "Fehler",
|
||||||
"failed_to_get_stream_url": "Fehler beim Abrufen der Stream-URL",
|
"failed_to_get_stream_url": "Fehler beim Abrufen der Stream-URL",
|
||||||
"an_error_occured_while_playing_the_video": "Ein Fehler ist beim Abspielen des Videos aufgetreten. Logs in den Einstellungen überprüfen.",
|
"an_error_occured_while_playing_the_video": "Ein Fehler ist beim Abspielen des Videos aufgetreten. Logs in den Einstellungen überprüfen.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Heruntergeladene Datei abspielen?",
|
"downloaded_file_message": "Heruntergeladene Datei abspielen?",
|
||||||
"downloaded_file_yes": "Ja",
|
"downloaded_file_yes": "Ja",
|
||||||
"downloaded_file_no": "Nein",
|
"downloaded_file_no": "Nein",
|
||||||
"downloaded_file_cancel": "Abbrechen",
|
"downloaded_file_cancel": "Abbrechen"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Endet um {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Als Nächstes",
|
"next_up": "Als Nächstes",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Serien",
|
"series": "Serien",
|
||||||
"seasons": "Staffeln",
|
"seasons": "Staffeln",
|
||||||
"season": "Staffel",
|
"season": "Staffel",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Keine Episoden für diese Staffel",
|
"no_episodes_for_this_season": "Keine Episoden für diese Staffel",
|
||||||
"overview": "Überblick",
|
"overview": "Überblick",
|
||||||
"more_with": "Mehr mit {{name}}",
|
"more_with": "Mehr mit {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Medienoptionen",
|
"media_options": "Medienoptionen",
|
||||||
"quality": "Qualität",
|
"quality": "Qualität",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitles": {
|
"subtitles": "Untertitel",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Mehr anzeigen",
|
"show_more": "Mehr anzeigen",
|
||||||
"show_less": "Weniger anzeigen",
|
"show_less": "Weniger anzeigen",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Erschien in",
|
"appeared_in": "Erschien in",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Konnte Element nicht laden",
|
"could_not_load_item": "Konnte Element nicht laden",
|
||||||
"none": "Keine",
|
"none": "Keine",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "{{item_count}} Elemente herunterladen",
|
"download_x_item": "{{item_count}} Elemente herunterladen",
|
||||||
"download_unwatched_only": "Nur Ungesehene",
|
"download_unwatched_only": "Nur Ungesehene",
|
||||||
"download_button": "Herunterladen"
|
"download_button": "Herunterladen"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Nächste",
|
"next": "Nächste",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Filme",
|
"movies": "Filme",
|
||||||
"sports": "Sport",
|
"sports": "Sport",
|
||||||
"for_kids": "Für Kinder",
|
"for_kids": "Für Kinder",
|
||||||
"news": "Nachrichten",
|
"news": "Nachrichten"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Bestätigen",
|
"confirm": "Bestätigen",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Ablehnen",
|
"decline": "Ablehnen",
|
||||||
"requested_by": "Angefragt von {{user}}",
|
"requested_by": "Angefragt von {{user}}",
|
||||||
"unknown_user": "Unbekannter Nutzer",
|
"unknown_user": "Unbekannter Nutzer",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Seerr-Server erfüllt nicht die minimalen Versionsanforderungen. Bitte den Seerr-Server auf mindestens 2.0.0 aktualisieren.",
|
"jellyseer_does_not_meet_requirements": "Seerr-Server erfüllt nicht die minimalen Versionsanforderungen. Bitte den Seerr-Server auf mindestens 2.0.0 aktualisieren.",
|
||||||
"jellyseerr_test_failed": "Seerr-Test fehlgeschlagen. Bitte erneut versuchen.",
|
"jellyseerr_test_failed": "Seerr-Test fehlgeschlagen. Bitte erneut versuchen.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Suche",
|
"search": "Suche",
|
||||||
"library": "Bibliothek",
|
"library": "Bibliothek",
|
||||||
"custom_links": "Links",
|
"custom_links": "Links",
|
||||||
"favorites": "Favoriten",
|
"favorites": "Favoriten"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Musik",
|
"title": "Musik",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "Nur diese Serie",
|
"show": "Nur diese Serie",
|
||||||
"all": "Alle (Standard)"
|
"all": "Alle (Standard)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Σφάλμα",
|
"error_title": "Σφάλμα",
|
||||||
"login_title": "Σύνδεση",
|
"login_title": "Σύνδεση",
|
||||||
"login_to_title": "Συνδεθείτε στο",
|
"login_to_title": "Συνδεθείτε στο",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Όνομα Χρήστη",
|
"username_placeholder": "Όνομα Χρήστη",
|
||||||
"password_placeholder": "Κωδικός",
|
"password_placeholder": "Κωδικός",
|
||||||
"login_button": "Σύνδεση",
|
"login_button": "Σύνδεση",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Ωχ!",
|
"oops": "Ωχ!",
|
||||||
"error_message": "Something went wrong.\nPlease log out and in again.",
|
"error_message": "Something went wrong.\nPlease log out and in again.",
|
||||||
"continue_watching": "Συνέχεια Παρακολούθησης",
|
"continue_watching": "Συνέχεια Παρακολούθησης",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Επόμενο Επάνω",
|
"next_up": "Επόμενο Επάνω",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Προστέθηκε πρόσφατα στο {{libraryName}}",
|
"recently_added_in": "Προστέθηκε πρόσφατα στο {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Ρυθμίσεις",
|
"settings_title": "Ρυθμίσεις",
|
||||||
"log_out_button": "Αποσύνδεση",
|
"log_out_button": "Αποσύνδεση",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categories"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Επαναφορά Μήκους",
|
"rewind_length": "Επαναφορά Μήκους",
|
||||||
"seconds_unit": "ίνα"
|
"seconds_unit": "ίνα"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Έλεγχοι Χειρονομιών",
|
"gesture_controls_title": "Έλεγχοι Χειρονομιών",
|
||||||
"horizontal_swipe_skip": "Οριζόντια σάρωση για παράλειψη",
|
"horizontal_swipe_skip": "Οριζόντια σάρωση για παράλειψη",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Σφάλμα Διαγραφής Αρχείων",
|
"error_deleting_files": "Σφάλμα Διαγραφής Αρχείων",
|
||||||
"background_downloads_enabled": "Οι λήψεις στο παρασκήνιο ενεργοποιήθηκαν",
|
"background_downloads_enabled": "Οι λήψεις στο παρασκήνιο ενεργοποιήθηκαν",
|
||||||
"background_downloads_disabled": "Οι λήψεις παρασκηνίου απενεργοποιήθηκαν"
|
"background_downloads_disabled": "Οι λήψεις παρασκηνίου απενεργοποιήθηκαν"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Η νέα ενημέρωση απαιτεί λήψη περιεχομένου ξανά. Καταργήστε όλο το κατεβασμένο περιεχόμενο και προσπαθήστε ξανά.",
|
"new_app_version_requires_re_download_description": "Η νέα ενημέρωση απαιτεί λήψη περιεχομένου ξανά. Καταργήστε όλο το κατεβασμένο περιεχόμενο και προσπαθήστε ξανά.",
|
||||||
"back": "Πίσω",
|
"back": "Πίσω",
|
||||||
"delete": "Διαγραφή",
|
"delete": "Διαγραφή",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Κάτι Πήγε Λάθος",
|
"something_went_wrong": "Κάτι Πήγε Λάθος",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Αδυναμία λήψης του URL ροής από το Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Αδυναμία λήψης του URL ροής από το Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Επιλογή",
|
"select": "Επιλογή",
|
||||||
"no_trailer_available": "Δεν υπάρχει διαθέσιμο ρυμουλκούμενο",
|
"no_trailer_available": "Δεν υπάρχει διαθέσιμο ρυμουλκούμενο",
|
||||||
"video": "Βίντεο",
|
"video": "Βίντεο",
|
||||||
"audio": "Ήχος",
|
"audio": "Ήχος",
|
||||||
"subtitle": "Υπότιτλος",
|
"subtitle": "Υπότιτλος",
|
||||||
"play": "Αναπαραγωγή",
|
"play": "Αναπαραγωγή",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Αναζήτηση...",
|
"search": "Αναζήτηση...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Ταινίες",
|
"movies": "Ταινίες",
|
||||||
"series": "Σειρά",
|
"series": "Σειρά",
|
||||||
"boxsets": "Σύνολα Πλαισίων",
|
"boxsets": "Σύνολα Πλαισίων",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Στοιχεία"
|
"items": "Στοιχεία"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Αφίσα",
|
"poster": "Αφίσα",
|
||||||
"cover": "Εξώφυλλο",
|
"cover": "Εξώφυλλο",
|
||||||
"show_titles": "Εμφάνιση Τίτλων",
|
"show_titles": "Εμφάνιση Τίτλων",
|
||||||
"show_stats": "Εμφάνιση Στατιστικών",
|
"show_stats": "Εμφάνιση Στατιστικών"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Ταξινόμηση Κατά",
|
"sort_by": "Ταξινόμηση Κατά",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Σειρά Ταξινόμησης",
|
"sort_order": "Σειρά Ταξινόμησης",
|
||||||
"tags": "Ετικέτες",
|
"tags": "Ετικέτες"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Δεν Υπάρχουν Σύνδεσμοι"
|
"no_links": "Δεν Υπάρχουν Σύνδεσμοι"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Σφάλμα",
|
"error": "Σφάλμα",
|
||||||
"failed_to_get_stream_url": "Αποτυχία λήψης του URL ροής",
|
"failed_to_get_stream_url": "Αποτυχία λήψης του URL ροής",
|
||||||
"an_error_occured_while_playing_the_video": "Παρουσιάστηκε σφάλμα κατά την αναπαραγωγή του βίντεο. Ελέγξτε τα αρχεία καταγραφής στις ρυθμίσεις.",
|
"an_error_occured_while_playing_the_video": "Παρουσιάστηκε σφάλμα κατά την αναπαραγωγή του βίντεο. Ελέγξτε τα αρχεία καταγραφής στις ρυθμίσεις.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Επόμενο Επάνω",
|
"next_up": "Επόμενο Επάνω",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Σειρά",
|
"series": "Σειρά",
|
||||||
"seasons": "Περίοδοι",
|
"seasons": "Περίοδοι",
|
||||||
"season": "Σεζόν",
|
"season": "Σεζόν",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Δεν υπάρχουν επεισόδια για αυτή τη σεζόν",
|
"no_episodes_for_this_season": "Δεν υπάρχουν επεισόδια για αυτή τη σεζόν",
|
||||||
"overview": "Επισκόπηση",
|
"overview": "Επισκόπηση",
|
||||||
"more_with": "More with {{name}}",
|
"more_with": "More with {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Ποιότητα",
|
"quality": "Ποιότητα",
|
||||||
"audio": "Ήχος",
|
"audio": "Ήχος",
|
||||||
"subtitles": {
|
"subtitles": "Υπότιτλος",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Εμφάνιση Περισσότερων",
|
"show_more": "Εμφάνιση Περισσότερων",
|
||||||
"show_less": "Εμφάνιση Λιγότερων",
|
"show_less": "Εμφάνιση Λιγότερων",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Εμφανίστηκε Σε",
|
"appeared_in": "Εμφανίστηκε Σε",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Αδύνατη Η Φόρτωση Του Στοιχείου",
|
"could_not_load_item": "Αδύνατη Η Φόρτωση Του Στοιχείου",
|
||||||
"none": "Κανένα",
|
"none": "Κανένα",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Λήψη Αντικειμένων {{item_count}}",
|
"download_x_item": "Λήψη Αντικειμένων {{item_count}}",
|
||||||
"download_unwatched_only": "Μόνο Χωρίς Παρακολούθηση",
|
"download_unwatched_only": "Μόνο Χωρίς Παρακολούθηση",
|
||||||
"download_button": "Λήψη"
|
"download_button": "Λήψη"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Επόμενο",
|
"next": "Επόμενο",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Ταινίες",
|
"movies": "Ταινίες",
|
||||||
"sports": "Αθλητισμός",
|
"sports": "Αθλητισμός",
|
||||||
"for_kids": "Για Παιδιά",
|
"for_kids": "Για Παιδιά",
|
||||||
"news": "Νέα",
|
"news": "Νέα"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Επιβεβαίωση",
|
"confirm": "Επιβεβαίωση",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Ο διακομιστής Seerr δεν πληροί τις ελάχιστες απαιτήσεις έκδοσης! Παρακαλούμε ενημερώστε τουλάχιστον σε 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Ο διακομιστής Seerr δεν πληροί τις ελάχιστες απαιτήσεις έκδοσης! Παρακαλούμε ενημερώστε τουλάχιστον σε 2.0.0",
|
||||||
"jellyseerr_test_failed": "Ο έλεγχος Seerr απέτυχε. Παρακαλώ προσπαθήστε ξανά.",
|
"jellyseerr_test_failed": "Ο έλεγχος Seerr απέτυχε. Παρακαλώ προσπαθήστε ξανά.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Αναζήτηση",
|
"search": "Αναζήτηση",
|
||||||
"library": "Βιβλιοθήκη",
|
"library": "Βιβλιοθήκη",
|
||||||
"custom_links": "Προσαρμοσμένοι Σύνδεσμοι",
|
"custom_links": "Προσαρμοσμένοι Σύνδεσμοι",
|
||||||
"favorites": "Αγαπημένα",
|
"favorites": "Αγαπημένα"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -534,7 +534,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "The new update requires content to be downloaded again. Please remove all downloaded content and try again.",
|
"new_app_version_requires_re_download_description": "The new update requires content to be downloaded again. Please remove all downloaded content and try again.",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Something Went Wrong",
|
"something_went_wrong": "Something Went Wrong",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Could not get the stream URL from Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Could not get the stream URL from Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -578,8 +577,6 @@
|
|||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Subtitle",
|
"subtitle": "Subtitle",
|
||||||
"play": "Play",
|
"play": "Play",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
@@ -652,8 +649,7 @@
|
|||||||
"poster": "Poster",
|
"poster": "Poster",
|
||||||
"cover": "Cover",
|
"cover": "Cover",
|
||||||
"show_titles": "Show Titles",
|
"show_titles": "Show Titles",
|
||||||
"show_stats": "Show Stats",
|
"show_stats": "Show Stats"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -683,7 +679,6 @@
|
|||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
"live": "LIVE",
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Error",
|
"error": "Error",
|
||||||
"failed_to_get_stream_url": "Failed to get the stream URL",
|
"failed_to_get_stream_url": "Failed to get the stream URL",
|
||||||
"an_error_occured_while_playing_the_video": "An error occurred while playing the video. Check logs in settings.",
|
"an_error_occured_while_playing_the_video": "An error occurred while playing the video. Check logs in settings.",
|
||||||
@@ -703,7 +698,7 @@
|
|||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel",
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
"swipe_down_settings": "Swipe down for settings",
|
||||||
"ends_at": "Ends at {{time}}",
|
"ends_at": "ends at",
|
||||||
"search_subtitles": "Search Subtitles",
|
"search_subtitles": "Search Subtitles",
|
||||||
"subtitle_tracks": "Tracks",
|
"subtitle_tracks": "Tracks",
|
||||||
"subtitle_search": "Search & Download",
|
"subtitle_search": "Search & Download",
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Error",
|
"error_title": "Error",
|
||||||
"login_title": "Iniciar sesión",
|
"login_title": "Iniciar sesión",
|
||||||
"login_to_title": "Iniciar sesión en",
|
"login_to_title": "Iniciar sesión en",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Nombre de usuario",
|
"username_placeholder": "Nombre de usuario",
|
||||||
"password_placeholder": "Contraseña",
|
"password_placeholder": "Contraseña",
|
||||||
"login_button": "Iniciar sesión",
|
"login_button": "Iniciar sesión",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} cuentas",
|
"accounts_count": "{{count}} cuentas",
|
||||||
"select_account": "Seleccione una cuenta",
|
"select_account": "Seleccione una cuenta",
|
||||||
"add_account": "Añadir cuenta",
|
"add_account": "Añadir cuenta",
|
||||||
"remove_account_description": "Esto eliminará las credenciales guardadas para {{username}}.",
|
"remove_account_description": "Esto eliminará las credenciales guardadas para {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Guardar Cuenta",
|
"title": "Guardar Cuenta",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "¡Vaya!",
|
"oops": "¡Vaya!",
|
||||||
"error_message": "Algo ha salido mal.\nPor favor, cierra la sesión y vuelve a iniciar.",
|
"error_message": "Algo ha salido mal.\nPor favor, cierra la sesión y vuelve a iniciar.",
|
||||||
"continue_watching": "Seguir viendo",
|
"continue_watching": "Seguir viendo",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "A continuación",
|
"next_up": "A continuación",
|
||||||
"continue_and_next_up": "Continuar y siguiente",
|
"continue_and_next_up": "Continuar y siguiente",
|
||||||
"recently_added_in": "Recientemente añadido en {{libraryName}}",
|
"recently_added_in": "Recientemente añadido en {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Configuración",
|
"settings_title": "Configuración",
|
||||||
"log_out_button": "Cerrar sesión",
|
"log_out_button": "Cerrar sesión",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categorías"
|
"title": "Categorías"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Apariencia",
|
"title": "Apariencia",
|
||||||
"merge_next_up_continue_watching": "Fusionar continuar viendo y siguiente",
|
"merge_next_up_continue_watching": "Fusionar continuar viendo y siguiente",
|
||||||
"hide_remote_session_button": "Ocultar botón de sesión remota",
|
"hide_remote_session_button": "Ocultar botón de sesión remota"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Cadena",
|
"title": "Cadena",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Longitud de retroceso",
|
"rewind_length": "Longitud de retroceso",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Controles de gestos",
|
"gesture_controls_title": "Controles de gestos",
|
||||||
"horizontal_swipe_skip": "Deslizar horizontal para omitir",
|
"horizontal_swipe_skip": "Deslizar horizontal para omitir",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Fuente de los subtítulos",
|
"subtitle_font": "Fuente de los subtítulos",
|
||||||
"ksplayer_title": "Ajustes de KSPlayer",
|
"ksplayer_title": "Ajustes de KSPlayer",
|
||||||
"hardware_decode": "Decodificación de hardware",
|
"hardware_decode": "Decodificación de hardware",
|
||||||
"hardware_decode_description": "Utilizar la aceleración de hardware para la decodificación de vídeo. Deshabilite si experimenta problemas de reproducción.",
|
"hardware_decode_description": "Utilizar la aceleración de hardware para la decodificación de vídeo. Deshabilite si experimenta problemas de reproducción."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "Configuración de subtítulos VLC",
|
"title": "Configuración de subtítulos VLC",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Caché de música eliminado",
|
"music_cache_cleared": "Caché de música eliminado",
|
||||||
"delete_all_downloaded_songs": "Eliminar todas las descargas",
|
"delete_all_downloaded_songs": "Eliminar todas las descargas",
|
||||||
"downloaded_songs_size": "{{tamaño}} descargado",
|
"downloaded_songs_size": "{{tamaño}} descargado",
|
||||||
"downloaded_songs_deleted": "Canciones descargadas eliminadas",
|
"downloaded_songs_deleted": "Canciones descargadas eliminadas"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Error al eliminar archivos",
|
"error_deleting_files": "Error al eliminar archivos",
|
||||||
"background_downloads_enabled": "Descargas en segundo plano habilitadas",
|
"background_downloads_enabled": "Descargas en segundo plano habilitadas",
|
||||||
"background_downloads_disabled": "Descargas en segundo plano deshabilitadas"
|
"background_downloads_disabled": "Descargas en segundo plano deshabilitadas"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "La nueva actualización requiere volver a descargar el contenido. Por favor, elimina todo el código descargado y vuélvelo a intentar.",
|
"new_app_version_requires_re_download_description": "La nueva actualización requiere volver a descargar el contenido. Por favor, elimina todo el código descargado y vuélvelo a intentar.",
|
||||||
"back": "Atrás",
|
"back": "Atrás",
|
||||||
"delete": "Borrar",
|
"delete": "Borrar",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Algo ha salido mal",
|
"something_went_wrong": "Algo ha salido mal",
|
||||||
"could_not_get_stream_url_from_jellyfin": "No se pudo obtener la URL del stream de Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "No se pudo obtener la URL del stream de Jellyfin",
|
||||||
"eta": "{{eta}} restante",
|
"eta": "{{eta}} restante",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Seleccionar",
|
"select": "Seleccionar",
|
||||||
"no_trailer_available": "No hay tráiler disponible",
|
"no_trailer_available": "No hay tráiler disponible",
|
||||||
"video": "Vídeo",
|
"video": "Vídeo",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Subtítulos",
|
"subtitle": "Subtítulos",
|
||||||
"play": "Jugar",
|
"play": "Jugar",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "Nada",
|
"none": "Nada",
|
||||||
"track": "Pista",
|
"track": "Pista",
|
||||||
"cancel": "Cancelar",
|
"cancel": "Cancelar",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Borrar",
|
"delete": "Borrar",
|
||||||
"ok": "Aceptar",
|
"ok": "Aceptar",
|
||||||
"remove": "Eliminar",
|
"remove": "Eliminar",
|
||||||
"next": "Siguiente",
|
"next": "Siguiente",
|
||||||
"back": "Atrás",
|
"back": "Atrás",
|
||||||
"continue": "Continuar",
|
"continue": "Continuar",
|
||||||
"verifying": "Verificando...",
|
"verifying": "Verificando..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Buscar...",
|
"search": "Buscar...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Películas",
|
"movies": "Películas",
|
||||||
"series": "Series",
|
"series": "Series",
|
||||||
"boxsets": "Colecciones",
|
"boxsets": "Colecciones",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Elementos"
|
"items": "Elementos"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Póster",
|
"poster": "Póster",
|
||||||
"cover": "Portada",
|
"cover": "Portada",
|
||||||
"show_titles": "Mostrar títulos",
|
"show_titles": "Mostrar títulos",
|
||||||
"show_stats": "Mostrar estadísticas",
|
"show_stats": "Mostrar estadísticas"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Géneros",
|
"genres": "Géneros",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Ordenar por",
|
"sort_by": "Ordenar por",
|
||||||
"filter_by": "Filtrar por",
|
"filter_by": "Filtrar por",
|
||||||
"sort_order": "Ordenar",
|
"sort_order": "Ordenar",
|
||||||
"tags": "Etiquetas",
|
"tags": "Etiquetas"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Sin enlaces"
|
"no_links": "Sin enlaces"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Error",
|
"error": "Error",
|
||||||
"failed_to_get_stream_url": "Error al obtener la URL del Steam",
|
"failed_to_get_stream_url": "Error al obtener la URL del Steam",
|
||||||
"an_error_occured_while_playing_the_video": "Ha ocurrido un error al reproducir el vídeo. Comprueba los registros en la configuración.",
|
"an_error_occured_while_playing_the_video": "Ha ocurrido un error al reproducir el vídeo. Comprueba los registros en la configuración.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "¿Quieres reproducir el archivo descargado?",
|
"downloaded_file_message": "¿Quieres reproducir el archivo descargado?",
|
||||||
"downloaded_file_yes": "Sí",
|
"downloaded_file_yes": "Sí",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancelar",
|
"downloaded_file_cancel": "Cancelar"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "A continuación",
|
"next_up": "A continuación",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Series",
|
"series": "Series",
|
||||||
"seasons": "Temporadas",
|
"seasons": "Temporadas",
|
||||||
"season": "Temporada",
|
"season": "Temporada",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "No hay episodios para esta temporada",
|
"no_episodes_for_this_season": "No hay episodios para esta temporada",
|
||||||
"overview": "Resumen",
|
"overview": "Resumen",
|
||||||
"more_with": "Más con {{name}}",
|
"more_with": "Más con {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Opciones de medios",
|
"media_options": "Opciones de medios",
|
||||||
"quality": "Calidad",
|
"quality": "Calidad",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitles": {
|
"subtitles": "Subtítulos",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Mostrar más",
|
"show_more": "Mostrar más",
|
||||||
"show_less": "Mostrar menos",
|
"show_less": "Mostrar menos",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Apareció en",
|
"appeared_in": "Apareció en",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "No se pudo cargar el ítem",
|
"could_not_load_item": "No se pudo cargar el ítem",
|
||||||
"none": "Ninguno",
|
"none": "Ninguno",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Descargar {{item_count}} ítems",
|
"download_x_item": "Descargar {{item_count}} ítems",
|
||||||
"download_unwatched_only": "No visto",
|
"download_unwatched_only": "No visto",
|
||||||
"download_button": "Descargar"
|
"download_button": "Descargar"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Siguiente",
|
"next": "Siguiente",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Películas",
|
"movies": "Películas",
|
||||||
"sports": "Deportes",
|
"sports": "Deportes",
|
||||||
"for_kids": "Para niños",
|
"for_kids": "Para niños",
|
||||||
"news": "Noticias",
|
"news": "Noticias"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Confirmar",
|
"confirm": "Confirmar",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Rechazar",
|
"decline": "Rechazar",
|
||||||
"requested_by": "Solicitado por {{user}}",
|
"requested_by": "Solicitado por {{user}}",
|
||||||
"unknown_user": "Usuario desconocido",
|
"unknown_user": "Usuario desconocido",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "¡Jellyseer no cumple con los requisitos! Por favor, actualízalo al menos a la versión 2.0.0.",
|
"jellyseer_does_not_meet_requirements": "¡Jellyseer no cumple con los requisitos! Por favor, actualízalo al menos a la versión 2.0.0.",
|
||||||
"jellyseerr_test_failed": "La prueba de Jellyseerr ha fallado. Por favor inténtalo de nuevo.",
|
"jellyseerr_test_failed": "La prueba de Jellyseerr ha fallado. Por favor inténtalo de nuevo.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Buscar",
|
"search": "Buscar",
|
||||||
"library": "Bibliotecas",
|
"library": "Bibliotecas",
|
||||||
"custom_links": "Enlaces personalizados",
|
"custom_links": "Enlaces personalizados",
|
||||||
"favorites": "Favoritos",
|
"favorites": "Favoritos"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Música",
|
"title": "Música",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "Este programa",
|
"show": "Este programa",
|
||||||
"all": "Todos los medios (por defecto)"
|
"all": "Todos los medios (por defecto)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Virhe",
|
"error_title": "Virhe",
|
||||||
"login_title": "Kirjaudu sisään",
|
"login_title": "Kirjaudu sisään",
|
||||||
"login_to_title": "Kirjaudu sisään palveluun",
|
"login_to_title": "Kirjaudu sisään palveluun",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Käyttäjätunnus",
|
"username_placeholder": "Käyttäjätunnus",
|
||||||
"password_placeholder": "Salasana",
|
"password_placeholder": "Salasana",
|
||||||
"login_button": "Kirjaudu sisään",
|
"login_button": "Kirjaudu sisään",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Ups!",
|
"oops": "Ups!",
|
||||||
"error_message": "Jotain meni pieleen.\nKirjaudu ulos ja takaisin sisään.",
|
"error_message": "Jotain meni pieleen.\nKirjaudu ulos ja takaisin sisään.",
|
||||||
"continue_watching": "Jatka katsomista",
|
"continue_watching": "Jatka katsomista",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Seuraavaksi",
|
"next_up": "Seuraavaksi",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Äskettäin lisätty {{libraryName}}-kirjastoon",
|
"recently_added_in": "Äskettäin lisätty {{libraryName}}-kirjastoon",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Asetukset",
|
"settings_title": "Asetukset",
|
||||||
"log_out_button": "Kirjaudu ulos",
|
"log_out_button": "Kirjaudu ulos",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Kategoriat"
|
"title": "Kategoriat"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Ulkoasu",
|
"title": "Ulkoasu",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Taaksepäin hyppäämisen pituus",
|
"rewind_length": "Taaksepäin hyppäämisen pituus",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Ele Ohjaus",
|
"gesture_controls_title": "Ele Ohjaus",
|
||||||
"horizontal_swipe_skip": "Ohita vaakatasossa pyyhkäisemällä",
|
"horizontal_swipe_skip": "Ohita vaakatasossa pyyhkäisemällä",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Esittely",
|
"title": "Esittely",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Virhe tiedostojen poistamisessa",
|
"error_deleting_files": "Virhe tiedostojen poistamisessa",
|
||||||
"background_downloads_enabled": "Taustalataukset käytössä",
|
"background_downloads_enabled": "Taustalataukset käytössä",
|
||||||
"background_downloads_disabled": "Taustalataukset pois käytöstä"
|
"background_downloads_disabled": "Taustalataukset pois käytöstä"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Uusi päivitys vaatii sisällön lataamista uudelleen. Poista kaikki ladattu sisältö ja yritä uudelleen.",
|
"new_app_version_requires_re_download_description": "Uusi päivitys vaatii sisällön lataamista uudelleen. Poista kaikki ladattu sisältö ja yritä uudelleen.",
|
||||||
"back": "Takaisin",
|
"back": "Takaisin",
|
||||||
"delete": "Poista",
|
"delete": "Poista",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Jotain meni pieleen",
|
"something_went_wrong": "Jotain meni pieleen",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Ei voitu saada suoratoiston URL:ia Jellyfinilta",
|
"could_not_get_stream_url_from_jellyfin": "Ei voitu saada suoratoiston URL:ia Jellyfinilta",
|
||||||
"eta": "Arvio {{eta}}",
|
"eta": "Arvio {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Valitse",
|
"select": "Valitse",
|
||||||
"no_trailer_available": "Perävaunua ei saatavilla",
|
"no_trailer_available": "Perävaunua ei saatavilla",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Ääni",
|
"audio": "Ääni",
|
||||||
"subtitle": "Tekstitys",
|
"subtitle": "Tekstitys",
|
||||||
"play": "Toista",
|
"play": "Toista",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "Ei mitään",
|
"none": "Ei mitään",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Haku...",
|
"search": "Haku...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "elokuvat",
|
"movies": "elokuvat",
|
||||||
"series": "sarjat",
|
"series": "sarjat",
|
||||||
"boxsets": "bokset",
|
"boxsets": "bokset",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "kohteet"
|
"items": "kohteet"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Juliste",
|
"poster": "Juliste",
|
||||||
"cover": "Kansi",
|
"cover": "Kansi",
|
||||||
"show_titles": "Näytä otsikot",
|
"show_titles": "Näytä otsikot",
|
||||||
"show_stats": "Näytä tilastot",
|
"show_stats": "Näytä tilastot"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genret",
|
"genres": "Genret",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Lajittele",
|
"sort_by": "Lajittele",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Lajittelujärjestys",
|
"sort_order": "Lajittelujärjestys",
|
||||||
"tags": "Tunnisteet",
|
"tags": "Tunnisteet"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Ei Linkkejä"
|
"no_links": "Ei Linkkejä"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Virhe",
|
"error": "Virhe",
|
||||||
"failed_to_get_stream_url": "Lähetyksen URL-osoitteen haku epäonnistui",
|
"failed_to_get_stream_url": "Lähetyksen URL-osoitteen haku epäonnistui",
|
||||||
"an_error_occured_while_playing_the_video": "Videon toiston yhteydessä tapahtui virhe. Tarkista lokit asetuksista.",
|
"an_error_occured_while_playing_the_video": "Videon toiston yhteydessä tapahtui virhe. Tarkista lokit asetuksista.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Haluatko toistaa ladatun tiedoston?",
|
"downloaded_file_message": "Haluatko toistaa ladatun tiedoston?",
|
||||||
"downloaded_file_yes": "Kyllä",
|
"downloaded_file_yes": "Kyllä",
|
||||||
"downloaded_file_no": "Ei",
|
"downloaded_file_no": "Ei",
|
||||||
"downloaded_file_cancel": "Peruuta",
|
"downloaded_file_cancel": "Peruuta"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Seuraavaksi",
|
"next_up": "Seuraavaksi",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Sarjat",
|
"series": "Sarjat",
|
||||||
"seasons": "Kaudet",
|
"seasons": "Kaudet",
|
||||||
"season": "Kausi",
|
"season": "Kausi",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Ei jaksoja tälle kaudelle",
|
"no_episodes_for_this_season": "Ei jaksoja tälle kaudelle",
|
||||||
"overview": "Yleiskatsaus",
|
"overview": "Yleiskatsaus",
|
||||||
"more_with": "Enemmän {{name}} kanssa",
|
"more_with": "Enemmän {{name}} kanssa",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media-asetukset",
|
"media_options": "Media-asetukset",
|
||||||
"quality": "Laatu",
|
"quality": "Laatu",
|
||||||
"audio": "Ääni",
|
"audio": "Ääni",
|
||||||
"subtitles": {
|
"subtitles": "Tekstitys",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Näytä Lisää",
|
"show_more": "Näytä Lisää",
|
||||||
"show_less": "Näytä Vähemmän",
|
"show_less": "Näytä Vähemmän",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Esiintyy Sisään",
|
"appeared_in": "Esiintyy Sisään",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Kohdetta Ei Voitu Ladata",
|
"could_not_load_item": "Kohdetta Ei Voitu Ladata",
|
||||||
"none": "Ei mitään",
|
"none": "Ei mitään",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Lataa {{item_count}} Kohteita",
|
"download_x_item": "Lataa {{item_count}} Kohteita",
|
||||||
"download_unwatched_only": "Vain Katsomattomat",
|
"download_unwatched_only": "Vain Katsomattomat",
|
||||||
"download_button": "Lataa"
|
"download_button": "Lataa"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Seuraava",
|
"next": "Seuraava",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Elokuvat",
|
"movies": "Elokuvat",
|
||||||
"sports": "Urheilu",
|
"sports": "Urheilu",
|
||||||
"for_kids": "Lapsille",
|
"for_kids": "Lapsille",
|
||||||
"news": "Uutiset",
|
"news": "Uutiset"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Vahvista",
|
"confirm": "Vahvista",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Hylkää",
|
"decline": "Hylkää",
|
||||||
"requested_by": "Käyttäjän {{user}} pyynnöstä",
|
"requested_by": "Käyttäjän {{user}} pyynnöstä",
|
||||||
"unknown_user": "Tuntematon käyttäjä",
|
"unknown_user": "Tuntematon käyttäjä",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Jellyseerr-palvelin ei täytä vähimmäisversiovaatimuksia! Päivitä vähintään versioon 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Jellyseerr-palvelin ei täytä vähimmäisversiovaatimuksia! Päivitä vähintään versioon 2.0.0",
|
||||||
"jellyseerr_test_failed": "Jellyseerr-testi epäonnistui. Yritä uudelleen.",
|
"jellyseerr_test_failed": "Jellyseerr-testi epäonnistui. Yritä uudelleen.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Haku",
|
"search": "Haku",
|
||||||
"library": "Kirjasto",
|
"library": "Kirjasto",
|
||||||
"custom_links": "Mukautetut linkit",
|
"custom_links": "Mukautetut linkit",
|
||||||
"favorites": "Suosikit",
|
"favorites": "Suosikit"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Erreur",
|
"error_title": "Erreur",
|
||||||
"login_title": "Se connecter",
|
"login_title": "Se connecter",
|
||||||
"login_to_title": "Se connecter à",
|
"login_to_title": "Se connecter à",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Nom d'utilisateur",
|
"username_placeholder": "Nom d'utilisateur",
|
||||||
"password_placeholder": "Mot de passe",
|
"password_placeholder": "Mot de passe",
|
||||||
"login_button": "Se connecter",
|
"login_button": "Se connecter",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "Comptes {{count}}",
|
"accounts_count": "Comptes {{count}}",
|
||||||
"select_account": "Sélectionnez un compte",
|
"select_account": "Sélectionnez un compte",
|
||||||
"add_account": "Ajouter un compte",
|
"add_account": "Ajouter un compte",
|
||||||
"remove_account_description": "Cela supprimera les identifiants enregistrés pour {{username}}.",
|
"remove_account_description": "Cela supprimera les identifiants enregistrés pour {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Sauvegarder le compte",
|
"title": "Sauvegarder le compte",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Oups !",
|
"oops": "Oups !",
|
||||||
"error_message": "Quelque chose s'est mal passé.\nVeuillez vous reconnecter à nouveau.",
|
"error_message": "Quelque chose s'est mal passé.\nVeuillez vous reconnecter à nouveau.",
|
||||||
"continue_watching": "Continuer à regarder",
|
"continue_watching": "Continuer à regarder",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "À suivre",
|
"next_up": "À suivre",
|
||||||
"continue_and_next_up": "Continuer de regarder et à suivre",
|
"continue_and_next_up": "Continuer de regarder et à suivre",
|
||||||
"recently_added_in": "Ajoutés récemment dans {{libraryName}}",
|
"recently_added_in": "Ajoutés récemment dans {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Paramètres",
|
"settings_title": "Paramètres",
|
||||||
"log_out_button": "Déconnexion",
|
"log_out_button": "Déconnexion",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Catégories"
|
"title": "Catégories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Apparence",
|
"title": "Apparence",
|
||||||
"merge_next_up_continue_watching": "Fusionner, continuer à regarder et à suivre",
|
"merge_next_up_continue_watching": "Fusionner, continuer à regarder et à suivre",
|
||||||
"hide_remote_session_button": "Masquer le bouton de session distante",
|
"hide_remote_session_button": "Masquer le bouton de session distante"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Réseau",
|
"title": "Réseau",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Durée de retour en arrière",
|
"rewind_length": "Durée de retour en arrière",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Commandes gestuelles",
|
"gesture_controls_title": "Commandes gestuelles",
|
||||||
"horizontal_swipe_skip": "Glisser horizontalement pour passer",
|
"horizontal_swipe_skip": "Glisser horizontalement pour passer",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Police des sous-titres",
|
"subtitle_font": "Police des sous-titres",
|
||||||
"ksplayer_title": "Paramètres de KSPlayer",
|
"ksplayer_title": "Paramètres de KSPlayer",
|
||||||
"hardware_decode": "Décodage matériel",
|
"hardware_decode": "Décodage matériel",
|
||||||
"hardware_decode_description": "Utilisez l’accélération matérielle pour le décodage vidéo. Désactivez si vous rencontrez des problèmes de lecture.",
|
"hardware_decode_description": "Utilisez l’accélération matérielle pour le décodage vidéo. Désactivez si vous rencontrez des problèmes de lecture."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "Paramètres des sous-titres VLC",
|
"title": "Paramètres des sous-titres VLC",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Cache de musique effacé",
|
"music_cache_cleared": "Cache de musique effacé",
|
||||||
"delete_all_downloaded_songs": "Supprimer toutes les musiques téléchargées",
|
"delete_all_downloaded_songs": "Supprimer toutes les musiques téléchargées",
|
||||||
"downloaded_songs_size": "{{size}} téléchargé",
|
"downloaded_songs_size": "{{size}} téléchargé",
|
||||||
"downloaded_songs_deleted": "Chansons téléchargées supprimées",
|
"downloaded_songs_deleted": "Chansons téléchargées supprimées"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Introduction",
|
"title": "Introduction",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Erreur lors de la suppression des fichiers",
|
"error_deleting_files": "Erreur lors de la suppression des fichiers",
|
||||||
"background_downloads_enabled": "Téléchargements en arrière-plan activés",
|
"background_downloads_enabled": "Téléchargements en arrière-plan activés",
|
||||||
"background_downloads_disabled": "Téléchargements en arrière-plan désactivés"
|
"background_downloads_disabled": "Téléchargements en arrière-plan désactivés"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "La nouvelle mise à jour nécessite que le contenu soit téléchargé à nouveau. Veuillez supprimer tout le contenu téléchargé et réessayer.",
|
"new_app_version_requires_re_download_description": "La nouvelle mise à jour nécessite que le contenu soit téléchargé à nouveau. Veuillez supprimer tout le contenu téléchargé et réessayer.",
|
||||||
"back": "Retour",
|
"back": "Retour",
|
||||||
"delete": "Supprimer",
|
"delete": "Supprimer",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Quelque chose s'est mal passé",
|
"something_went_wrong": "Quelque chose s'est mal passé",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Impossible d'obtenir l'URL du flux depuis Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Impossible d'obtenir l'URL du flux depuis Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Sélectionner",
|
"select": "Sélectionner",
|
||||||
"no_trailer_available": "Aucune bande-annonce disponible",
|
"no_trailer_available": "Aucune bande-annonce disponible",
|
||||||
"video": "Vidéo",
|
"video": "Vidéo",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Sous-titres",
|
"subtitle": "Sous-titres",
|
||||||
"play": "Lecture",
|
"play": "Lecture",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "Aucun",
|
"none": "Aucun",
|
||||||
"track": "Suivre",
|
"track": "Suivre",
|
||||||
"cancel": "Annuler",
|
"cancel": "Annuler",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Supprimer",
|
"delete": "Supprimer",
|
||||||
"ok": "Ok",
|
"ok": "Ok",
|
||||||
"remove": "Retirer",
|
"remove": "Retirer",
|
||||||
"next": "Suivant",
|
"next": "Suivant",
|
||||||
"back": "Précédent",
|
"back": "Précédent",
|
||||||
"continue": "Continuer",
|
"continue": "Continuer",
|
||||||
"verifying": "Vérification...",
|
"verifying": "Vérification..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Rechercher...",
|
"search": "Rechercher...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Films",
|
"movies": "Films",
|
||||||
"series": "Séries",
|
"series": "Séries",
|
||||||
"boxsets": "Coffrets ",
|
"boxsets": "Coffrets ",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Médias"
|
"items": "Médias"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Affiche",
|
"poster": "Affiche",
|
||||||
"cover": "Couverture",
|
"cover": "Couverture",
|
||||||
"show_titles": "Afficher les titres",
|
"show_titles": "Afficher les titres",
|
||||||
"show_stats": "Afficher les statistiques",
|
"show_stats": "Afficher les statistiques"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Trier par",
|
"sort_by": "Trier par",
|
||||||
"filter_by": "Filtrer par",
|
"filter_by": "Filtrer par",
|
||||||
"sort_order": "Ordre de tri",
|
"sort_order": "Ordre de tri",
|
||||||
"tags": "Tags",
|
"tags": "Tags"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Aucuns liens"
|
"no_links": "Aucuns liens"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Erreur",
|
"error": "Erreur",
|
||||||
"failed_to_get_stream_url": "Échec de l'obtention de l'URL du flux",
|
"failed_to_get_stream_url": "Échec de l'obtention de l'URL du flux",
|
||||||
"an_error_occured_while_playing_the_video": "Une erreur s’est produite lors de la lecture de la vidéo. Vérifiez les journaux dans les paramètres.",
|
"an_error_occured_while_playing_the_video": "Une erreur s’est produite lors de la lecture de la vidéo. Vérifiez les journaux dans les paramètres.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Voulez-vous lire le fichier téléchargé ?",
|
"downloaded_file_message": "Voulez-vous lire le fichier téléchargé ?",
|
||||||
"downloaded_file_yes": "Oui",
|
"downloaded_file_yes": "Oui",
|
||||||
"downloaded_file_no": "Non",
|
"downloaded_file_no": "Non",
|
||||||
"downloaded_file_cancel": "Annuler",
|
"downloaded_file_cancel": "Annuler"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "À suivre",
|
"next_up": "À suivre",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Séries",
|
"series": "Séries",
|
||||||
"seasons": "Saisons",
|
"seasons": "Saisons",
|
||||||
"season": "Saison",
|
"season": "Saison",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Aucun épisode pour cette saison",
|
"no_episodes_for_this_season": "Aucun épisode pour cette saison",
|
||||||
"overview": "Aperçu",
|
"overview": "Aperçu",
|
||||||
"more_with": "Plus avec {{name}}",
|
"more_with": "Plus avec {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Options média",
|
"media_options": "Options média",
|
||||||
"quality": "Qualité",
|
"quality": "Qualité",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitles": {
|
"subtitles": "Sous-titres",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Afficher plus",
|
"show_more": "Afficher plus",
|
||||||
"show_less": "Afficher moins",
|
"show_less": "Afficher moins",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Apparu dans",
|
"appeared_in": "Apparu dans",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Impossible de charger le média",
|
"could_not_load_item": "Impossible de charger le média",
|
||||||
"none": "Aucun",
|
"none": "Aucun",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Télécharger {{item_count}} médias",
|
"download_x_item": "Télécharger {{item_count}} médias",
|
||||||
"download_unwatched_only": "Non visionné uniquement",
|
"download_unwatched_only": "Non visionné uniquement",
|
||||||
"download_button": "Télécharger"
|
"download_button": "Télécharger"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Suivant",
|
"next": "Suivant",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Films",
|
"movies": "Films",
|
||||||
"sports": "Sports",
|
"sports": "Sports",
|
||||||
"for_kids": "Pour enfants",
|
"for_kids": "Pour enfants",
|
||||||
"news": "Actualités",
|
"news": "Actualités"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Confirmer",
|
"confirm": "Confirmer",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Refuser",
|
"decline": "Refuser",
|
||||||
"requested_by": "Demandé par {{user}}",
|
"requested_by": "Demandé par {{user}}",
|
||||||
"unknown_user": "Utilisateur inconnu",
|
"unknown_user": "Utilisateur inconnu",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Seerr ne répond pas aux exigences ! Veuillez mettre à jour au moins vers la version 2.0.0.",
|
"jellyseer_does_not_meet_requirements": "Seerr ne répond pas aux exigences ! Veuillez mettre à jour au moins vers la version 2.0.0.",
|
||||||
"jellyseerr_test_failed": "Le test Seerr a échoué. Veuillez réessayer.",
|
"jellyseerr_test_failed": "Le test Seerr a échoué. Veuillez réessayer.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Recherche",
|
"search": "Recherche",
|
||||||
"library": "Bibliothèque",
|
"library": "Bibliothèque",
|
||||||
"custom_links": "Liens personnalisés",
|
"custom_links": "Liens personnalisés",
|
||||||
"favorites": "Favoris",
|
"favorites": "Favoris"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Musique",
|
"title": "Musique",
|
||||||
@@ -949,8 +788,8 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"watchlists": {
|
"watchlists": {
|
||||||
"title": "Listes de lecture",
|
"title": "Watchlists",
|
||||||
"my_watchlists": "Mes listes de lecture",
|
"my_watchlists": "My Watchlists",
|
||||||
"public_watchlists": "Watchlist publique",
|
"public_watchlists": "Watchlist publique",
|
||||||
"create_title": "Créer une Watchlist",
|
"create_title": "Créer une Watchlist",
|
||||||
"edit_title": "Modifier la Watchlist",
|
"edit_title": "Modifier la Watchlist",
|
||||||
@@ -963,7 +802,7 @@
|
|||||||
"name_placeholder": "Entrer le nom de la playlist",
|
"name_placeholder": "Entrer le nom de la playlist",
|
||||||
"description_label": "Description",
|
"description_label": "Description",
|
||||||
"description_placeholder": "Entrez la description (facultatif)",
|
"description_placeholder": "Entrez la description (facultatif)",
|
||||||
"is_public_label": "Liste de lecture Publique",
|
"is_public_label": "Public Watchlist",
|
||||||
"is_public_description": "Autoriser d'autres personnes à voir cette liste de suivi",
|
"is_public_description": "Autoriser d'autres personnes à voir cette liste de suivi",
|
||||||
"allowed_type_label": "Type de contenu",
|
"allowed_type_label": "Type de contenu",
|
||||||
"sort_order_label": "Ordre de tri par défaut",
|
"sort_order_label": "Ordre de tri par défaut",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "Cette série",
|
"show": "Cette série",
|
||||||
"all": "Tous les médias (par défaut)"
|
"all": "Tous les médias (par défaut)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "שגיאה",
|
"error_title": "שגיאה",
|
||||||
"login_title": "התחבר",
|
"login_title": "התחבר",
|
||||||
"login_to_title": "התחבר אל",
|
"login_to_title": "התחבר אל",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "שם משתמש",
|
"username_placeholder": "שם משתמש",
|
||||||
"password_placeholder": "סיסמה",
|
"password_placeholder": "סיסמה",
|
||||||
"login_button": "התחבר",
|
"login_button": "התחבר",
|
||||||
@@ -42,16 +39,10 @@
|
|||||||
"please_login_again": "Your saved session has expired. Please log in again.",
|
"please_login_again": "Your saved session has expired. Please log in again.",
|
||||||
"remove_saved_login": "Remove Saved Login",
|
"remove_saved_login": "Remove Saved Login",
|
||||||
"remove_saved_login_description": "This will remove your saved credentials for this server. You'll need to enter your username and password again next time.",
|
"remove_saved_login_description": "This will remove your saved credentials for this server. You'll need to enter your username and password again next time.",
|
||||||
"accounts_count": "{{count}} חשבונות",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "אופס!",
|
"oops": "אופס!",
|
||||||
"error_message": "קרתה תקלה. אנא התנתק והתחבר מחדש.",
|
"error_message": "קרתה תקלה. אנא התנתק והתחבר מחדש.",
|
||||||
"continue_watching": "המשך לצפות",
|
"continue_watching": "המשך לצפות",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "הבא בתור",
|
"next_up": "הבא בתור",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "התווסף לאחרונה ב-{{libraryName}}",
|
"recently_added_in": "התווסף לאחרונה ב-{{libraryName}}",
|
||||||
@@ -119,34 +109,19 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "הגדרות",
|
"settings_title": "הגדרות",
|
||||||
"log_out_button": "התנתק",
|
"log_out_button": "התנתק",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "קטגוריות"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
"playback_controls": {
|
"playback_controls": {
|
||||||
"title": "Playback & Controls"
|
"title": "Playback & Controls"
|
||||||
},
|
},
|
||||||
"audio_subtitles": {
|
"audio_subtitles": {
|
||||||
"title": "שמע וכתוביות"
|
"title": "Audio & Subtitles"
|
||||||
},
|
},
|
||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "מראה",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "אורך הזזה אחורה",
|
"rewind_length": "אורך הזזה אחורה",
|
||||||
"seconds_unit": "שנ'"
|
"seconds_unit": "שנ'"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "פקדי מחוות",
|
"gesture_controls_title": "פקדי מחוות",
|
||||||
"horizontal_swipe_skip": "החלקה אופקית לדילוג",
|
"horizontal_swipe_skip": "החלקה אופקית לדילוג",
|
||||||
@@ -229,7 +188,7 @@
|
|||||||
"hide_brightness_slider_description": "Hide the brightness slider in the video player"
|
"hide_brightness_slider_description": "Hide the brightness slider in the video player"
|
||||||
},
|
},
|
||||||
"audio": {
|
"audio": {
|
||||||
"audio_title": "שמע",
|
"audio_title": "אודיו",
|
||||||
"set_audio_track": "בחר רצועת שמע מהפריט הקודם",
|
"set_audio_track": "בחר רצועת שמע מהפריט הקודם",
|
||||||
"audio_language": "שפת שמע",
|
"audio_language": "שפת שמע",
|
||||||
"audio_hint": "בחר שפת שמע אוטומטית.",
|
"audio_hint": "בחר שפת שמע אוטומטית.",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -328,8 +271,8 @@
|
|||||||
"margin": "Bottom Margin"
|
"margin": "Bottom Margin"
|
||||||
},
|
},
|
||||||
"video_player": {
|
"video_player": {
|
||||||
"title": "נגן וידאו",
|
"title": "Video Player",
|
||||||
"video_player": "נגן וידאו",
|
"video_player": "Video Player",
|
||||||
"video_player_description": "Choose which video player to use on iOS.",
|
"video_player_description": "Choose which video player to use on iOS.",
|
||||||
"ksplayer": "KSPlayer",
|
"ksplayer": "KSPlayer",
|
||||||
"vlc": "VLC"
|
"vlc": "VLC"
|
||||||
@@ -371,7 +314,7 @@
|
|||||||
"downloads_title": "הורדות"
|
"downloads_title": "הורדות"
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "מוזיקה",
|
"title": "Music",
|
||||||
"playback_title": "Playback",
|
"playback_title": "Playback",
|
||||||
"playback_description": "Configure how music is played.",
|
"playback_description": "Configure how music is played.",
|
||||||
"prefer_downloaded": "Prefer Downloaded Songs",
|
"prefer_downloaded": "Prefer Downloaded Songs",
|
||||||
@@ -463,16 +406,10 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "הקדמה",
|
"title": "Intro",
|
||||||
"show_intro": "הצג פתיח",
|
"show_intro": "הצג פתיח",
|
||||||
"reset_intro": "אפס פתיח"
|
"reset_intro": "אפס פתיח"
|
||||||
},
|
},
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "שגיאה במחיקת קבצים",
|
"error_deleting_files": "שגיאה במחיקת קבצים",
|
||||||
"background_downloads_enabled": "הורדה ברקע מופעלת",
|
"background_downloads_enabled": "הורדה ברקע מופעלת",
|
||||||
"background_downloads_disabled": "הורדה ברקע כבויה"
|
"background_downloads_disabled": "הורדה ברקע כבויה"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "הגרסה החדשה דורשת לתוכן לרדת מחדש. אנא מחק את כל התוכן שכבר הורדת ונסה שוב.",
|
"new_app_version_requires_re_download_description": "הגרסה החדשה דורשת לתוכן לרדת מחדש. אנא מחק את כל התוכן שכבר הורדת ונסה שוב.",
|
||||||
"back": "חזרה",
|
"back": "חזרה",
|
||||||
"delete": "מחק",
|
"delete": "מחק",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "משהו השתבש",
|
"something_went_wrong": "משהו השתבש",
|
||||||
"could_not_get_stream_url_from_jellyfin": "לא היה ניתן להגיע לקישור הזרם מהשרת Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "לא היה ניתן להגיע לקישור הזרם מהשרת Jellyfin",
|
||||||
"eta": "זמן משוער {{eta}}",
|
"eta": "זמן משוער {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "בחר",
|
"select": "בחר",
|
||||||
"no_trailer_available": "אין טריילר זמין",
|
"no_trailer_available": "אין טריילר זמין",
|
||||||
"video": "וידאו",
|
"video": "וידאו",
|
||||||
"audio": "שמע",
|
"audio": "אודיו",
|
||||||
"subtitle": "כתובית",
|
"subtitle": "כתובית",
|
||||||
"play": "נגן",
|
"play": "נגן",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "ללא",
|
"none": "ללא",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "חפש...",
|
"search": "חפש...",
|
||||||
@@ -606,9 +521,9 @@
|
|||||||
"episodes": "פרקים",
|
"episodes": "פרקים",
|
||||||
"collections": "אוספים",
|
"collections": "אוספים",
|
||||||
"actors": "שחקנים",
|
"actors": "שחקנים",
|
||||||
"artists": "אומנים",
|
"artists": "Artists",
|
||||||
"albums": "אלבומים",
|
"albums": "Albums",
|
||||||
"songs": "שירים",
|
"songs": "Songs",
|
||||||
"playlists": "Playlists",
|
"playlists": "Playlists",
|
||||||
"request_movies": "סרטים מבוקשים",
|
"request_movies": "סרטים מבוקשים",
|
||||||
"request_series": "סדרות מבוקשים",
|
"request_series": "סדרות מבוקשים",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "סרטים",
|
"movies": "סרטים",
|
||||||
"series": "סדרות",
|
"series": "סדרות",
|
||||||
"boxsets": "אוסף",
|
"boxsets": "אוסף",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "פריטים"
|
"items": "פריטים"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "פוסטר",
|
"poster": "פוסטר",
|
||||||
"cover": "עטיפה",
|
"cover": "עטיפה",
|
||||||
"show_titles": "הצג כותרות",
|
"show_titles": "הצג כותרות",
|
||||||
"show_stats": "הצג סטטיסטיקה",
|
"show_stats": "הצג סטטיסטיקה"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "סגנונות",
|
"genres": "סגנונות",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "מיין לפי",
|
"sort_by": "מיין לפי",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "סדר מיון",
|
"sort_order": "סדר מיון",
|
||||||
"tags": "תגים",
|
"tags": "תגים"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "אין קישורים"
|
"no_links": "אין קישורים"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "שגיאה",
|
"error": "שגיאה",
|
||||||
"failed_to_get_stream_url": "נכשל בהשגת קישור הזרם",
|
"failed_to_get_stream_url": "נכשל בהשגת קישור הזרם",
|
||||||
"an_error_occured_while_playing_the_video": "קרתה תקלה במהלך הניגון של הקובץ. בדוק את הלוגים בהגדרות.",
|
"an_error_occured_while_playing_the_video": "קרתה תקלה במהלך הניגון של הקובץ. בדוק את הלוגים בהגדרות.",
|
||||||
@@ -699,37 +606,9 @@
|
|||||||
"go_back": "חזור",
|
"go_back": "חזור",
|
||||||
"downloaded_file_title": "You have this file downloaded",
|
"downloaded_file_title": "You have this file downloaded",
|
||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "כן",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "לא",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "הבא בתור",
|
"next_up": "הבא בתור",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "סדרות",
|
"series": "סדרות",
|
||||||
"seasons": "עונות",
|
"seasons": "עונות",
|
||||||
"season": "עונה",
|
"season": "עונה",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "אין פרקים בעונה זו",
|
"no_episodes_for_this_season": "אין פרקים בעונה זו",
|
||||||
"overview": "סקירה",
|
"overview": "סקירה",
|
||||||
"more_with": "עוד עם {{name}}",
|
"more_with": "עוד עם {{name}}",
|
||||||
@@ -752,22 +626,11 @@
|
|||||||
"more_details": "פרטים נוספים",
|
"more_details": "פרטים נוספים",
|
||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "איכות",
|
"quality": "איכות",
|
||||||
"audio": "שמע",
|
"audio": "אודיו",
|
||||||
"subtitles": {
|
"subtitles": "כתובית",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "הצג עוד",
|
"show_more": "הצג עוד",
|
||||||
"show_less": "הצג פחות",
|
"show_less": "הצג פחות",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "הופיע ב-",
|
"appeared_in": "הופיע ב-",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "נכשל בטעינת פריט",
|
"could_not_load_item": "נכשל בטעינת פריט",
|
||||||
"none": "ללא",
|
"none": "ללא",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "הורד {{item_count}} פריטים",
|
"download_x_item": "הורד {{item_count}} פריטים",
|
||||||
"download_unwatched_only": "רק שלא נצפו",
|
"download_unwatched_only": "רק שלא נצפו",
|
||||||
"download_button": "הורד"
|
"download_button": "הורד"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "הבא",
|
"next": "הבא",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "סרטים",
|
"movies": "סרטים",
|
||||||
"sports": "ספורט",
|
"sports": "ספורט",
|
||||||
"for_kids": "לילדים",
|
"for_kids": "לילדים",
|
||||||
"news": "חדשות",
|
"news": "חדשות"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "אשר",
|
"confirm": "אשר",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "שרת ה-Seerr לא תואם את הגרסה המינימלית הנרדת! אנא עדכן לפחות לגרסה 2.0.0",
|
"jellyseer_does_not_meet_requirements": "שרת ה-Seerr לא תואם את הגרסה המינימלית הנרדת! אנא עדכן לפחות לגרסה 2.0.0",
|
||||||
"jellyseerr_test_failed": "בדיקת ה-Seerr נכשלה. אנא נסה שוב.",
|
"jellyseerr_test_failed": "בדיקת ה-Seerr נכשלה. אנא נסה שוב.",
|
||||||
@@ -876,14 +716,13 @@
|
|||||||
"search": "חיפוש",
|
"search": "חיפוש",
|
||||||
"library": "ספריה",
|
"library": "ספריה",
|
||||||
"custom_links": "קישורים מותאמים אישית",
|
"custom_links": "קישורים מותאמים אישית",
|
||||||
"favorites": "מועדפים",
|
"favorites": "מועדפים"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "מוזיקה",
|
"title": "Music",
|
||||||
"tabs": {
|
"tabs": {
|
||||||
"suggestions": "Suggestions",
|
"suggestions": "Suggestions",
|
||||||
"albums": "אלבומים",
|
"albums": "Albums",
|
||||||
"artists": "Artists",
|
"artists": "Artists",
|
||||||
"playlists": "Playlists",
|
"playlists": "Playlists",
|
||||||
"tracks": "tracks"
|
"tracks": "tracks"
|
||||||
@@ -959,9 +798,9 @@
|
|||||||
"delete_button": "Delete",
|
"delete_button": "Delete",
|
||||||
"remove_button": "Remove",
|
"remove_button": "Remove",
|
||||||
"cancel_button": "Cancel",
|
"cancel_button": "Cancel",
|
||||||
"name_label": "שם",
|
"name_label": "Name",
|
||||||
"name_placeholder": "Enter watchlist name",
|
"name_placeholder": "Enter watchlist name",
|
||||||
"description_label": "תיאור",
|
"description_label": "Description",
|
||||||
"description_placeholder": "Enter description (optional)",
|
"description_placeholder": "Enter description (optional)",
|
||||||
"is_public_label": "Public Watchlist",
|
"is_public_label": "Public Watchlist",
|
||||||
"is_public_description": "Allow others to view this watchlist",
|
"is_public_description": "Allow others to view this watchlist",
|
||||||
@@ -978,10 +817,10 @@
|
|||||||
"remove_from_watchlist": "Remove from Watchlist",
|
"remove_from_watchlist": "Remove from Watchlist",
|
||||||
"select_watchlist": "Select Watchlist",
|
"select_watchlist": "Select Watchlist",
|
||||||
"create_new": "Create New Watchlist",
|
"create_new": "Create New Watchlist",
|
||||||
"item": "פריט",
|
"item": "item",
|
||||||
"items": "פריטים",
|
"items": "items",
|
||||||
"public": "ציבורי",
|
"public": "Public",
|
||||||
"private": "פרטי",
|
"private": "Private",
|
||||||
"you": "You",
|
"you": "You",
|
||||||
"by_owner": "By another user",
|
"by_owner": "By another user",
|
||||||
"not_found": "Watchlist not found",
|
"not_found": "Watchlist not found",
|
||||||
@@ -996,42 +835,11 @@
|
|||||||
"playback_speed": {
|
"playback_speed": {
|
||||||
"title": "Playback Speed",
|
"title": "Playback Speed",
|
||||||
"apply_to": "Apply To",
|
"apply_to": "Apply To",
|
||||||
"speed": "מהירות",
|
"speed": "Speed",
|
||||||
"scope": {
|
"scope": {
|
||||||
"media": "This media only",
|
"media": "This media only",
|
||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Hiba",
|
"error_title": "Hiba",
|
||||||
"login_title": "Bejelentkezés",
|
"login_title": "Bejelentkezés",
|
||||||
"login_to_title": "Bejelentkezés ide",
|
"login_to_title": "Bejelentkezés ide",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Felhasználónév",
|
"username_placeholder": "Felhasználónév",
|
||||||
"password_placeholder": "Jelszó",
|
"password_placeholder": "Jelszó",
|
||||||
"login_button": "Bejelentkezés",
|
"login_button": "Bejelentkezés",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Hoppá!",
|
"oops": "Hoppá!",
|
||||||
"error_message": "Valami nem stimmel.\nKérjük, jelentkezz ki, majd újra be.",
|
"error_message": "Valami nem stimmel.\nKérjük, jelentkezz ki, majd újra be.",
|
||||||
"continue_watching": "Nézd Tovább",
|
"continue_watching": "Nézd Tovább",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Következő",
|
"next_up": "Következő",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Új a(z) {{libraryName}} könyvtárban",
|
"recently_added_in": "Új a(z) {{libraryName}} könyvtárban",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Beállítások",
|
"settings_title": "Beállítások",
|
||||||
"log_out_button": "Kijelentkezés",
|
"log_out_button": "Kijelentkezés",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categories"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Visszatekerés Hossza",
|
"rewind_length": "Visszatekerés Hossza",
|
||||||
"seconds_unit": "mp"
|
"seconds_unit": "mp"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Gesztusvezérlés",
|
"gesture_controls_title": "Gesztusvezérlés",
|
||||||
"horizontal_swipe_skip": "Vízszintes Húzás Ugráshoz",
|
"horizontal_swipe_skip": "Vízszintes Húzás Ugráshoz",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Hiba a Fájlok Törlésekor",
|
"error_deleting_files": "Hiba a Fájlok Törlésekor",
|
||||||
"background_downloads_enabled": "Background downloads enabled",
|
"background_downloads_enabled": "Background downloads enabled",
|
||||||
"background_downloads_disabled": "Background downloads disabled"
|
"background_downloads_disabled": "Background downloads disabled"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Az új frissítéshez az összes tartalmat újra le kell tölteni. Kérjük, töröld az összes letöltött tartalmat, majd próbáld újra.",
|
"new_app_version_requires_re_download_description": "Az új frissítéshez az összes tartalmat újra le kell tölteni. Kérjük, töröld az összes letöltött tartalmat, majd próbáld újra.",
|
||||||
"back": "Vissza",
|
"back": "Vissza",
|
||||||
"delete": "Törlés",
|
"delete": "Törlés",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Hiba Történt",
|
"something_went_wrong": "Hiba Történt",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Nem sikerült lekérni a stream URL-t a Jellyfinből",
|
"could_not_get_stream_url_from_jellyfin": "Nem sikerült lekérni a stream URL-t a Jellyfinből",
|
||||||
"eta": "Várható Idő: {{eta}}",
|
"eta": "Várható Idő: {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Select",
|
"select": "Select",
|
||||||
"no_trailer_available": "No trailer available",
|
"no_trailer_available": "No trailer available",
|
||||||
"video": "Videó",
|
"video": "Videó",
|
||||||
"audio": "Hang",
|
"audio": "Hang",
|
||||||
"subtitle": "Felirat",
|
"subtitle": "Felirat",
|
||||||
"play": "Play",
|
"play": "Play",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Keresés...",
|
"search": "Keresés...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Filmek",
|
"movies": "Filmek",
|
||||||
"series": "Sorozatok",
|
"series": "Sorozatok",
|
||||||
"boxsets": "Gyűjtemények",
|
"boxsets": "Gyűjtemények",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Elemek"
|
"items": "Elemek"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Poszter",
|
"poster": "Poszter",
|
||||||
"cover": "Borító",
|
"cover": "Borító",
|
||||||
"show_titles": "Címek Megjelenítése",
|
"show_titles": "Címek Megjelenítése",
|
||||||
"show_stats": "Statisztikák Megjelenítése",
|
"show_stats": "Statisztikák Megjelenítése"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Műfajok",
|
"genres": "Műfajok",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Rendezés",
|
"sort_by": "Rendezés",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Rendezés Iránya",
|
"sort_order": "Rendezés Iránya",
|
||||||
"tags": "Címkék",
|
"tags": "Címkék"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Nincsenek Linkek"
|
"no_links": "Nincsenek Linkek"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Hiba",
|
"error": "Hiba",
|
||||||
"failed_to_get_stream_url": "Nem sikerült lekérni a stream URL-t",
|
"failed_to_get_stream_url": "Nem sikerült lekérni a stream URL-t",
|
||||||
"an_error_occured_while_playing_the_video": "Hiba történt a videó lejátszása közben. Ellenőrizd a naplókat a beállításokban.",
|
"an_error_occured_while_playing_the_video": "Hiba történt a videó lejátszása közben. Ellenőrizd a naplókat a beállításokban.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Következő",
|
"next_up": "Következő",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Sorozat",
|
"series": "Sorozat",
|
||||||
"seasons": "Évadok",
|
"seasons": "Évadok",
|
||||||
"season": "Évad",
|
"season": "Évad",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Ehhez az évadhoz nincs epizód",
|
"no_episodes_for_this_season": "Ehhez az évadhoz nincs epizód",
|
||||||
"overview": "Áttekintés",
|
"overview": "Áttekintés",
|
||||||
"more_with": "További {{name}} Alkotások",
|
"more_with": "További {{name}} Alkotások",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Minőség",
|
"quality": "Minőség",
|
||||||
"audio": "Hang",
|
"audio": "Hang",
|
||||||
"subtitles": {
|
"subtitles": "Felirat",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Több Megjelenítése",
|
"show_more": "Több Megjelenítése",
|
||||||
"show_less": "Kevesebb Megjelenítése",
|
"show_less": "Kevesebb Megjelenítése",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Megjelent:",
|
"appeared_in": "Megjelent:",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Nem Sikerült Betölteni az Elemet",
|
"could_not_load_item": "Nem Sikerült Betölteni az Elemet",
|
||||||
"none": "Nincs",
|
"none": "Nincs",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "{{item_count}} Elem Letöltése",
|
"download_x_item": "{{item_count}} Elem Letöltése",
|
||||||
"download_unwatched_only": "Csak Nem Megtekintett",
|
"download_unwatched_only": "Csak Nem Megtekintett",
|
||||||
"download_button": "Letöltés"
|
"download_button": "Letöltés"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Következő",
|
"next": "Következő",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Filmek",
|
"movies": "Filmek",
|
||||||
"sports": "Sport",
|
"sports": "Sport",
|
||||||
"for_kids": "Gyerekeknek",
|
"for_kids": "Gyerekeknek",
|
||||||
"news": "Hírek",
|
"news": "Hírek"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Megerősítés",
|
"confirm": "Megerősítés",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "A Jellyseerr szerver nem felel meg a minimum verziókövetelményeknek! Kérlek frissítsd legalább 2.0.0-ra.",
|
"jellyseer_does_not_meet_requirements": "A Jellyseerr szerver nem felel meg a minimum verziókövetelményeknek! Kérlek frissítsd legalább 2.0.0-ra.",
|
||||||
"jellyseerr_test_failed": "A Jellyseerr teszt sikertelen. Próbáld újra.",
|
"jellyseerr_test_failed": "A Jellyseerr teszt sikertelen. Próbáld újra.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Keresés",
|
"search": "Keresés",
|
||||||
"library": "Könyvtár",
|
"library": "Könyvtár",
|
||||||
"custom_links": "Egyéni Linkek",
|
"custom_links": "Egyéni Linkek",
|
||||||
"favorites": "Kedvencek",
|
"favorites": "Kedvencek"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Errore",
|
"error_title": "Errore",
|
||||||
"login_title": "Accesso",
|
"login_title": "Accesso",
|
||||||
"login_to_title": "Accedi a",
|
"login_to_title": "Accedi a",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Nome utente",
|
"username_placeholder": "Nome utente",
|
||||||
"password_placeholder": "Password",
|
"password_placeholder": "Password",
|
||||||
"login_button": "Accedi",
|
"login_button": "Accedi",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Ops!",
|
"oops": "Ops!",
|
||||||
"error_message": "Qualcosa è andato storto. \nEffetturare il logout e riaccedere.",
|
"error_message": "Qualcosa è andato storto. \nEffetturare il logout e riaccedere.",
|
||||||
"continue_watching": "Continua a guardare",
|
"continue_watching": "Continua a guardare",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Prossimo",
|
"next_up": "Prossimo",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Aggiunti di recente a {{libraryName}}",
|
"recently_added_in": "Aggiunti di recente a {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Impostazioni",
|
"settings_title": "Impostazioni",
|
||||||
"log_out_button": "Esci",
|
"log_out_button": "Esci",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categorie"
|
"title": "Categorie"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Aspetto",
|
"title": "Aspetto",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -161,7 +136,7 @@
|
|||||||
"not_connected_to_wifi": "Not connected to WiFi",
|
"not_connected_to_wifi": "Not connected to WiFi",
|
||||||
"no_networks_configured": "No networks configured",
|
"no_networks_configured": "No networks configured",
|
||||||
"add_network_hint": "Add your home WiFi network to enable auto-switching",
|
"add_network_hint": "Add your home WiFi network to enable auto-switching",
|
||||||
"current_wifi": "WiFi Attuale",
|
"current_wifi": "Current WiFi",
|
||||||
"using_url": "Sta utilizzando",
|
"using_url": "Sta utilizzando",
|
||||||
"local": "Local URL",
|
"local": "Local URL",
|
||||||
"remote": "Remote URL",
|
"remote": "Remote URL",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Lunghezza del riavvolgimento",
|
"rewind_length": "Lunghezza del riavvolgimento",
|
||||||
"seconds_unit": "secondi"
|
"seconds_unit": "secondi"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Controlli Gesture",
|
"gesture_controls_title": "Controlli Gesture",
|
||||||
"horizontal_swipe_skip": "Scorrimento orizzontale per saltare",
|
"horizontal_swipe_skip": "Scorrimento orizzontale per saltare",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Errore nella cancellazione dei file",
|
"error_deleting_files": "Errore nella cancellazione dei file",
|
||||||
"background_downloads_enabled": "Scaricamento in background abilitato",
|
"background_downloads_enabled": "Scaricamento in background abilitato",
|
||||||
"background_downloads_disabled": "Scaricamento in background disabilitato"
|
"background_downloads_disabled": "Scaricamento in background disabilitato"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Il nuovo aggiornamento richiede di scaricare nuovamente i contenuti. Rimuovere tutti i contenuti scaricati e riprovare.",
|
"new_app_version_requires_re_download_description": "Il nuovo aggiornamento richiede di scaricare nuovamente i contenuti. Rimuovere tutti i contenuti scaricati e riprovare.",
|
||||||
"back": "Indietro",
|
"back": "Indietro",
|
||||||
"delete": "Cancella",
|
"delete": "Cancella",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Qualcosa è andato storto",
|
"something_went_wrong": "Qualcosa è andato storto",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Impossibile ottenere l'URL del flusso da Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Impossibile ottenere l'URL del flusso da Jellyfin",
|
||||||
"eta": "Tempo stimato di completamento {{eta}}",
|
"eta": "Tempo stimato di completamento {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Seleziona",
|
"select": "Seleziona",
|
||||||
"no_trailer_available": "Nessun trailer disponibile",
|
"no_trailer_available": "Nessun trailer disponibile",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Sottotitoli",
|
"subtitle": "Sottotitoli",
|
||||||
"play": "Gioca",
|
"play": "Gioca",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "Nulla",
|
"none": "Nulla",
|
||||||
"track": "Traccia",
|
"track": "Traccia",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Cerca...",
|
"search": "Cerca...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "film",
|
"movies": "film",
|
||||||
"series": "serie TV",
|
"series": "serie TV",
|
||||||
"boxsets": "cofanetti",
|
"boxsets": "cofanetti",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "elementi"
|
"items": "elementi"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Poster",
|
"poster": "Poster",
|
||||||
"cover": "Copertina",
|
"cover": "Copertina",
|
||||||
"show_titles": "Mostra titoli",
|
"show_titles": "Mostra titoli",
|
||||||
"show_stats": "Mostra statistiche",
|
"show_stats": "Mostra statistiche"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Generi",
|
"genres": "Generi",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Ordina per",
|
"sort_by": "Ordina per",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Criterio di ordinamento",
|
"sort_order": "Criterio di ordinamento",
|
||||||
"tags": "Tag",
|
"tags": "Tag"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Nessun link"
|
"no_links": "Nessun link"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Errore",
|
"error": "Errore",
|
||||||
"failed_to_get_stream_url": "Impossibile ottenere l'URL dello stream",
|
"failed_to_get_stream_url": "Impossibile ottenere l'URL dello stream",
|
||||||
"an_error_occured_while_playing_the_video": "Si è verificato un errore durante la riproduzione del video. Controllare i log nelle impostazioni.",
|
"an_error_occured_while_playing_the_video": "Si è verificato un errore durante la riproduzione del video. Controllare i log nelle impostazioni.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Il prossimo",
|
"next_up": "Il prossimo",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Serie",
|
"series": "Serie",
|
||||||
"seasons": "Stagioni",
|
"seasons": "Stagioni",
|
||||||
"season": "Stagione",
|
"season": "Stagione",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Nessun episodio per questa stagione",
|
"no_episodes_for_this_season": "Nessun episodio per questa stagione",
|
||||||
"overview": "Panoramica",
|
"overview": "Panoramica",
|
||||||
"more_with": "Altri con {{name}}",
|
"more_with": "Altri con {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Opzioni Media",
|
"media_options": "Opzioni Media",
|
||||||
"quality": "Qualità",
|
"quality": "Qualità",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitles": {
|
"subtitles": "Sottotitoli",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Mostra di più",
|
"show_more": "Mostra di più",
|
||||||
"show_less": "Mostra di meno",
|
"show_less": "Mostra di meno",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Apparso in",
|
"appeared_in": "Apparso in",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Impossibile caricare l'elemento",
|
"could_not_load_item": "Impossibile caricare l'elemento",
|
||||||
"none": "Nessuno",
|
"none": "Nessuno",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Scarica {{item_count}} elementi",
|
"download_x_item": "Scarica {{item_count}} elementi",
|
||||||
"download_unwatched_only": "Solo Non Visti",
|
"download_unwatched_only": "Solo Non Visti",
|
||||||
"download_button": "Scarica"
|
"download_button": "Scarica"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Prossimo",
|
"next": "Prossimo",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Film",
|
"movies": "Film",
|
||||||
"sports": "Sport",
|
"sports": "Sport",
|
||||||
"for_kids": "Per Bambini",
|
"for_kids": "Per Bambini",
|
||||||
"news": "Notiziari",
|
"news": "Notiziari"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Conferma",
|
"confirm": "Conferma",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Rifiuta",
|
"decline": "Rifiuta",
|
||||||
"requested_by": "Richiesto da {{user}}",
|
"requested_by": "Richiesto da {{user}}",
|
||||||
"unknown_user": "Utente Sconosciuto",
|
"unknown_user": "Utente Sconosciuto",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Il server Jellyseerr non soddisfa i requisiti minimi di versione! Aggiornare almeno alla versione 2.0.0.",
|
"jellyseer_does_not_meet_requirements": "Il server Jellyseerr non soddisfa i requisiti minimi di versione! Aggiornare almeno alla versione 2.0.0.",
|
||||||
"jellyseerr_test_failed": "Il test di Jellyseerr non è riuscito. Riprovare.",
|
"jellyseerr_test_failed": "Il test di Jellyseerr non è riuscito. Riprovare.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Cerca",
|
"search": "Cerca",
|
||||||
"library": "Libreria",
|
"library": "Libreria",
|
||||||
"custom_links": "Collegamenti personalizzati",
|
"custom_links": "Collegamenti personalizzati",
|
||||||
"favorites": "Preferiti",
|
"favorites": "Preferiti"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "エラー",
|
"error_title": "エラー",
|
||||||
"login_title": "ログイン",
|
"login_title": "ログイン",
|
||||||
"login_to_title": "ログイン先",
|
"login_to_title": "ログイン先",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "ユーザー名",
|
"username_placeholder": "ユーザー名",
|
||||||
"password_placeholder": "パスワード",
|
"password_placeholder": "パスワード",
|
||||||
"login_button": "ログイン",
|
"login_button": "ログイン",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "おっと!",
|
"oops": "おっと!",
|
||||||
"error_message": "何か問題が発生しました。\nログアウトして再度ログインしてください。",
|
"error_message": "何か問題が発生しました。\nログアウトして再度ログインしてください。",
|
||||||
"continue_watching": "続きを見る",
|
"continue_watching": "続きを見る",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "次の動画",
|
"next_up": "次の動画",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "{{libraryName}}に最近追加された",
|
"recently_added_in": "{{libraryName}}に最近追加された",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "設定",
|
"settings_title": "設定",
|
||||||
"log_out_button": "ログアウト",
|
"log_out_button": "ログアウト",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "カテゴリ"
|
"title": "カテゴリ"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "巻き戻しの長さ",
|
"rewind_length": "巻き戻しの長さ",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "ジェスチャーコントロール",
|
"gesture_controls_title": "ジェスチャーコントロール",
|
||||||
"horizontal_swipe_skip": "水平方向にスワイプしてスキップ",
|
"horizontal_swipe_skip": "水平方向にスワイプしてスキップ",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "イントロ",
|
"title": "イントロ",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "ファイルの削除エラー",
|
"error_deleting_files": "ファイルの削除エラー",
|
||||||
"background_downloads_enabled": "バックグラウンドでのダウンロードは有効です",
|
"background_downloads_enabled": "バックグラウンドでのダウンロードは有効です",
|
||||||
"background_downloads_disabled": "バックグラウンドでのダウンロードは無効です"
|
"background_downloads_disabled": "バックグラウンドでのダウンロードは無効です"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "新しいアップデートではコンテンツを再度ダウンロードする必要があります。ダウンロードしたコンテンツをすべて削除してもう一度お試しください。",
|
"new_app_version_requires_re_download_description": "新しいアップデートではコンテンツを再度ダウンロードする必要があります。ダウンロードしたコンテンツをすべて削除してもう一度お試しください。",
|
||||||
"back": "戻る",
|
"back": "戻る",
|
||||||
"delete": "削除",
|
"delete": "削除",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "問題が発生しました",
|
"something_went_wrong": "問題が発生しました",
|
||||||
"could_not_get_stream_url_from_jellyfin": "JellyfinからストリームURLを取得できませんでした",
|
"could_not_get_stream_url_from_jellyfin": "JellyfinからストリームURLを取得できませんでした",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "選択",
|
"select": "選択",
|
||||||
"no_trailer_available": "トレーラーがありません",
|
"no_trailer_available": "トレーラーがありません",
|
||||||
"video": "映像",
|
"video": "映像",
|
||||||
"audio": "音声",
|
"audio": "音声",
|
||||||
"subtitle": "字幕",
|
"subtitle": "字幕",
|
||||||
"play": "再生",
|
"play": "再生",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "検索...",
|
"search": "検索...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "映画",
|
"movies": "映画",
|
||||||
"series": "シリーズ",
|
"series": "シリーズ",
|
||||||
"boxsets": "ボックスセット",
|
"boxsets": "ボックスセット",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "アイテム"
|
"items": "アイテム"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "ポスター",
|
"poster": "ポスター",
|
||||||
"cover": "カバー",
|
"cover": "カバー",
|
||||||
"show_titles": "タイトルの表示",
|
"show_titles": "タイトルの表示",
|
||||||
"show_stats": "統計を表示",
|
"show_stats": "統計を表示"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "ジャンル",
|
"genres": "ジャンル",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "ソート",
|
"sort_by": "ソート",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "ソート順",
|
"sort_order": "ソート順",
|
||||||
"tags": "タグ",
|
"tags": "タグ"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "リンクがありません"
|
"no_links": "リンクがありません"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "エラー",
|
"error": "エラー",
|
||||||
"failed_to_get_stream_url": "ストリームURLを取得できませんでした",
|
"failed_to_get_stream_url": "ストリームURLを取得できませんでした",
|
||||||
"an_error_occured_while_playing_the_video": "動画の再生中にエラーが発生しました。設定でログを確認してください。",
|
"an_error_occured_while_playing_the_video": "動画の再生中にエラーが発生しました。設定でログを確認してください。",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "次",
|
"next_up": "次",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "シリーズ",
|
"series": "シリーズ",
|
||||||
"seasons": "シーズン",
|
"seasons": "シーズン",
|
||||||
"season": "シーズン",
|
"season": "シーズン",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "このシーズンのエピソードはありません",
|
"no_episodes_for_this_season": "このシーズンのエピソードはありません",
|
||||||
"overview": "ストーリー",
|
"overview": "ストーリー",
|
||||||
"more_with": "{{name}}の詳細",
|
"more_with": "{{name}}の詳細",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "画質",
|
"quality": "画質",
|
||||||
"audio": "音声",
|
"audio": "音声",
|
||||||
"subtitles": {
|
"subtitles": "字幕",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "もっと見る",
|
"show_more": "もっと見る",
|
||||||
"show_less": "少なく表示",
|
"show_less": "少なく表示",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "出演作品",
|
"appeared_in": "出演作品",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "アイテムを読み込めませんでした",
|
"could_not_load_item": "アイテムを読み込めませんでした",
|
||||||
"none": "なし",
|
"none": "なし",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "{{item_count}}のアイテムをダウンロード",
|
"download_x_item": "{{item_count}}のアイテムをダウンロード",
|
||||||
"download_unwatched_only": "未視聴のみ",
|
"download_unwatched_only": "未視聴のみ",
|
||||||
"download_button": "ダウンロード"
|
"download_button": "ダウンロード"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "次",
|
"next": "次",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "映画",
|
"movies": "映画",
|
||||||
"sports": "スポーツ",
|
"sports": "スポーツ",
|
||||||
"for_kids": "子供向け",
|
"for_kids": "子供向け",
|
||||||
"news": "ニュース",
|
"news": "ニュース"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "確認",
|
"confirm": "確認",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Jellyseerrサーバーは最小バージョン要件を満たしていません。少なくとも 2.0.0 に更新してください。",
|
"jellyseer_does_not_meet_requirements": "Jellyseerrサーバーは最小バージョン要件を満たしていません。少なくとも 2.0.0 に更新してください。",
|
||||||
"jellyseerr_test_failed": "Jellyseerrテストに失敗しました。もう一度お試しください。",
|
"jellyseerr_test_failed": "Jellyseerrテストに失敗しました。もう一度お試しください。",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "検索",
|
"search": "検索",
|
||||||
"library": "ライブラリ",
|
"library": "ライブラリ",
|
||||||
"custom_links": "カスタムリンク",
|
"custom_links": "カスタムリンク",
|
||||||
"favorites": "お気に入り",
|
"favorites": "お気に入り"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,228 +1,187 @@
|
|||||||
{
|
{
|
||||||
"login": {
|
"login": {
|
||||||
"username_required": "사용자 이름이 필요합니다",
|
"username_required": "Username Is Required",
|
||||||
"error_title": "오류",
|
"error_title": "Error",
|
||||||
"login_title": "로그인",
|
"login_title": "Log In",
|
||||||
"login_to_title": "다음 서비스에 연결 중",
|
"login_to_title": "Log in to",
|
||||||
"select_user": "Select a user to log in",
|
"username_placeholder": "Username",
|
||||||
"add_user_to_login": "Add a user to log in",
|
"password_placeholder": "Password",
|
||||||
"add_user": "Add User",
|
"login_button": "Log In",
|
||||||
"username_placeholder": "사용자 이름",
|
"quick_connect": "Quick Connect",
|
||||||
"password_placeholder": "비밀번호",
|
"enter_code_to_login": "Enter code {{code}} to login",
|
||||||
"login_button": "로그인",
|
"failed_to_initiate_quick_connect": "Failed to initiate Quick Connect",
|
||||||
"quick_connect": "퀵 커넥트",
|
"got_it": "Got It",
|
||||||
"enter_code_to_login": "로그인 하기 위해 코드{{code}}를 입력하세요",
|
"connection_failed": "Connection Failed",
|
||||||
"failed_to_initiate_quick_connect": "Quick Connect 연결을 시작하는 데 실패했습니다",
|
"could_not_connect_to_server": "Could not connect to the server. Please check the URL and your network connection.",
|
||||||
"got_it": "성공",
|
"an_unexpected_error_occured": "An Unexpected Error Occurred",
|
||||||
"connection_failed": "연결 실패",
|
"change_server": "Change Server",
|
||||||
"could_not_connect_to_server": "서버에 연결되지 않았습니다. URL과 네트워크 상태를 확인하세요.",
|
"invalid_username_or_password": "Invalid Username or Password",
|
||||||
"an_unexpected_error_occured": "예기치 않은 오류가 발생했습니다",
|
"user_does_not_have_permission_to_log_in": "User does not have permission to log in",
|
||||||
"change_server": "서버 변경",
|
"server_is_taking_too_long_to_respond_try_again_later": "Server is taking too long to respond, try again later",
|
||||||
"invalid_username_or_password": "잘못된 아이디 혹은 비밀번호입니다",
|
"server_received_too_many_requests_try_again_later": "Server received too many requests, try again later.",
|
||||||
"user_does_not_have_permission_to_log_in": "로그인 하기 위한 권한이 없습니다",
|
"there_is_a_server_error": "There is a server error",
|
||||||
"server_is_taking_too_long_to_respond_try_again_later": "서버 응답이 너무 느립니다. 나중에 다시 시도하세요",
|
"an_unexpected_error_occured_did_you_enter_the_correct_url": "An unexpected error occurred. Did you enter the server URL correctly?",
|
||||||
"server_received_too_many_requests_try_again_later": "서버가 너무 많은 요청을 받았습니다. 나중에 다시 시도하세요.",
|
|
||||||
"there_is_a_server_error": "서버 에러",
|
|
||||||
"an_unexpected_error_occured_did_you_enter_the_correct_url": "예기치 않은 오류가 발생했습니다. 서버 URL을 올바르게 입력하셨습니까?",
|
|
||||||
"too_old_server_text": "Unsupported Jellyfin Server Discovered",
|
"too_old_server_text": "Unsupported Jellyfin Server Discovered",
|
||||||
"too_old_server_description": "Please update Jellyfin to the latest version"
|
"too_old_server_description": "Please update Jellyfin to the latest version"
|
||||||
},
|
},
|
||||||
"server": {
|
"server": {
|
||||||
"enter_url_to_jellyfin_server": "Enter the URL to your Jellyfin server",
|
"enter_url_to_jellyfin_server": "Enter the URL to your Jellyfin server",
|
||||||
"server_url_placeholder": "http(s)://your-server.com",
|
"server_url_placeholder": "http(s)://your-server.com",
|
||||||
"connect_button": "연결",
|
"connect_button": "Connect",
|
||||||
"previous_servers": "이전 서버",
|
"previous_servers": "Previous Servers",
|
||||||
"clear_button": "모두 지우기",
|
"clear_button": "Clear all",
|
||||||
"swipe_to_remove": "스와이프해서 지우기",
|
"swipe_to_remove": "Swipe to remove",
|
||||||
"search_for_local_servers": "로컬 서버 찾기",
|
"search_for_local_servers": "Search for Local Servers",
|
||||||
"searching": "찾는 중...",
|
"searching": "Searching...",
|
||||||
"servers": "서버",
|
"servers": "Servers",
|
||||||
"saved": "저장됨",
|
"saved": "Saved",
|
||||||
"session_expired": "세션 만료됨",
|
"session_expired": "Session Expired",
|
||||||
"please_login_again": "사용자 세션이 만료되었습니다. 다시 로그인하십시오.",
|
"please_login_again": "Your saved session has expired. Please log in again.",
|
||||||
"remove_saved_login": "저장된 로그인 정보 삭제",
|
"remove_saved_login": "Remove Saved Login",
|
||||||
"remove_saved_login_description": "해당 서버에 저장된 자격 증명이 삭제됩니다. 다음에 접속할 때는 사용자 이름과 비밀번호를 다시 입력해야 합니다.",
|
"remove_saved_login_description": "This will remove your saved credentials for this server. You'll need to enter your username and password again next time.",
|
||||||
"accounts_count": "{{count}} 계정",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "계정 선택",
|
"select_account": "Select Account",
|
||||||
"add_account": "계정 추가",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "{{username}}에 저장된 자격 증명이 삭제됩니다.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "계정 저장",
|
"title": "Save Account",
|
||||||
"save_for_later": "이 계정 저장",
|
"save_for_later": "Save this account",
|
||||||
"security_option": "보안 설정",
|
"security_option": "Security Option",
|
||||||
"no_protection": "보안 없음",
|
"no_protection": "No protection",
|
||||||
"no_protection_desc": "인증 없이 빠른 로그인",
|
"no_protection_desc": "Quick login without authentication",
|
||||||
"pin_code": "PIN 코드",
|
"pin_code": "PIN code",
|
||||||
"pin_code_desc": "전환하려면 4자리 PIN 필요함",
|
"pin_code_desc": "4-digit PIN required when switching",
|
||||||
"password": "암호 확인",
|
"password": "Re-enter password",
|
||||||
"password_desc": "전환하려면 비밀번호 필요함",
|
"password_desc": "Password required when switching",
|
||||||
"save_button": "저장",
|
"save_button": "Save",
|
||||||
"cancel_button": "취소"
|
"cancel_button": "Cancel"
|
||||||
},
|
},
|
||||||
"pin": {
|
"pin": {
|
||||||
"enter_pin": "PIN 입력",
|
"enter_pin": "Enter PIN",
|
||||||
"enter_pin_for": "{{username}} PIN 입력",
|
"enter_pin_for": "Enter PIN for {{username}}",
|
||||||
"enter_4_digits": "4자리 입력",
|
"enter_4_digits": "Enter 4 digits",
|
||||||
"invalid_pin": "잘못된 PIN",
|
"invalid_pin": "Invalid PIN",
|
||||||
"setup_pin": "PIN 설정",
|
"setup_pin": "Set Up PIN",
|
||||||
"confirm_pin": "PIN 확인",
|
"confirm_pin": "Confirm PIN",
|
||||||
"pins_dont_match": "PIN이 일치하지 않습니다",
|
"pins_dont_match": "PINs don't match",
|
||||||
"forgot_pin": "PIN을 잊으셨나요?",
|
"forgot_pin": "Forgot PIN?",
|
||||||
"forgot_pin_desc": "저장된 계정 정보가 삭제됩니다"
|
"forgot_pin_desc": "Your saved credentials will be removed"
|
||||||
},
|
},
|
||||||
"password": {
|
"password": {
|
||||||
"enter_password": "비밀번호 입력",
|
"enter_password": "Enter Password",
|
||||||
"enter_password_for": "{{username}}의 비밀번호 입력",
|
"enter_password_for": "Enter password for {{username}}",
|
||||||
"invalid_password": "잘못된 비밀번호"
|
"invalid_password": "Invalid password"
|
||||||
},
|
},
|
||||||
"home": {
|
"home": {
|
||||||
"checking_server_connection": "서버 연결 체크중...",
|
"checking_server_connection": "Checking server connection...",
|
||||||
"no_internet": "인터넷에 연결되지 않음",
|
"no_internet": "No Internet",
|
||||||
"no_items": "항목 없음",
|
"no_items": "No Items",
|
||||||
"no_internet_message": "걱정마세요. 다운로드 된 컨텐츠는 여전히 볼 수 있습니다.",
|
"no_internet_message": "No worries, you can still watch\ndownloaded content.",
|
||||||
"checking_server_connection_message": "Checking connection to server",
|
"checking_server_connection_message": "Checking connection to server",
|
||||||
"go_to_downloads": "Go to Downloads",
|
"go_to_downloads": "Go to Downloads",
|
||||||
"retry": "재시도",
|
"retry": "Retry",
|
||||||
"server_unreachable": "서버에 연결할 수 없음",
|
"server_unreachable": "Server Unreachable",
|
||||||
"server_unreachable_message": "서버에 연결할 수 없습니다. 네트워크 상태를 체크하세요.",
|
"server_unreachable_message": "Could not reach the server.\nPlease check your network connection.",
|
||||||
"oops": "이런!",
|
"oops": "Oops!",
|
||||||
"error_message": "문제가 발생했습니다.\n로그아웃 후 다시 로그인해 주세요.",
|
"error_message": "Something went wrong.\nPlease log out and in again.",
|
||||||
"continue_watching": "이어서 보기",
|
"continue_watching": "Continue Watching",
|
||||||
"continue": "Continue",
|
"next_up": "Next Up",
|
||||||
"next_up": "다음 시청",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"continue_and_next_up": "이어서 보기 & 다음 시청",
|
"recently_added_in": "Recently Added in {{libraryName}}",
|
||||||
"recently_added_in": "최근에 추가된 {{libraryName}}",
|
"suggested_movies": "Suggested Movies",
|
||||||
"suggested_movies": "추천 영화",
|
"suggested_episodes": "Suggested Episodes",
|
||||||
"suggested_episodes": "추천 에피소드",
|
|
||||||
"intro": {
|
"intro": {
|
||||||
"welcome_to_streamyfin": "스트리미핀에 오신 것을 환영합니다",
|
"welcome_to_streamyfin": "Welcome to Streamyfin",
|
||||||
"a_free_and_open_source_client_for_jellyfin": "젤리핀을 위한 무료 오픈소스 클라이언트입니다.",
|
"a_free_and_open_source_client_for_jellyfin": "A Free and Open-Source Client for Jellyfin.",
|
||||||
"features_title": "기능",
|
"features_title": "Features",
|
||||||
"features_description": "스트리미핀은 다양한 기능을 제공하며 설정 메뉴에서 확인할 수 있는 여러 소프트웨어와 통합됩니다. 이러한 소프트웨어에는 다음이 포함됩니다:",
|
"features_description": "Streamyfin has a bunch of features and integrates with a wide array of software which you can find in the settings menu, these include:",
|
||||||
"jellyseerr_feature_description": "Seerr 인스턴스에 연결하여 앱에서 직접 영화를 요청할 수 있습니다.",
|
"jellyseerr_feature_description": "Connect to your Seerr instance and request movies directly in the app.",
|
||||||
"downloads_feature_title": "다운로드된 컨텐츠",
|
"downloads_feature_title": "Downloads",
|
||||||
"downloads_feature_description": "오프라인으로 보기위해 다운로드 하세요. 기본 다운로드 방식을 사용하거나, 백그라운드에서 파일을 다운로드하는 최적화 서버를 설치할 수 있습니다.",
|
"downloads_feature_description": "Download movies and tv-shows to view offline. Use either the default method or install the optimize server to download files in the background.",
|
||||||
"chromecast_feature_description": "영화와 TV 프로그램을 Chromecast 기기로 전송하기",
|
"chromecast_feature_description": "Cast movies and tv-shows to your Chromecast devices.",
|
||||||
"centralised_settings_plugin_title": "중앙 설정 플러그인",
|
"centralised_settings_plugin_title": "Centralised Settings Plugin",
|
||||||
"centralised_settings_plugin_description": "Jellyfin 서버의 중앙 집중식 위치에서 설정을 구성합니다. 모든 사용자의 모든 클라이언트 설정이 자동으로 동기화됩니다.",
|
"centralised_settings_plugin_description": "Configure settings from a centralised location on your Jellyfin server. All client settings for all users will be synced automatically.",
|
||||||
"done_button": "확인",
|
"done_button": "Done",
|
||||||
"go_to_settings_button": "설정으로 이동",
|
"go_to_settings_button": "Go to Settings",
|
||||||
"read_more": "자세히 보기"
|
"read_more": "Read More"
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "설정",
|
"settings_title": "Settings",
|
||||||
"log_out_button": "로그아웃",
|
"log_out_button": "Log Out",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "카테고리"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
"playback_controls": {
|
"playback_controls": {
|
||||||
"title": "재생 & 컨트롤"
|
"title": "Playback & Controls"
|
||||||
},
|
},
|
||||||
"audio_subtitles": {
|
"audio_subtitles": {
|
||||||
"title": "오디오 & 자막"
|
"title": "Audio & Subtitles"
|
||||||
},
|
},
|
||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "화면 스타일",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "[이어보기]와 [다음 보기] 합치기",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "원격 세션 버튼 숨기기",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "네트워크",
|
"title": "Network",
|
||||||
"local_network": "로컬 네트워크",
|
"local_network": "Local Network",
|
||||||
"auto_switch_enabled": "홈 네트워크 자동 전환",
|
"auto_switch_enabled": "Auto-switch when at home",
|
||||||
"auto_switch_description": "홈 WiFi에 연결되었을 때 로컬 URL로 자동 전환",
|
"auto_switch_description": "Automatically switch to local URL when connected to home WiFi",
|
||||||
"local_url": "로컬 URL",
|
"local_url": "Local URL",
|
||||||
"local_url_hint": "로컬 서버 주소를 입력하세요 (e.g., http://192.168.1.100:8096)",
|
"local_url_hint": "Enter your local server address (e.g., http://192.168.1.100:8096)",
|
||||||
"local_url_placeholder": "http://192.168.1.100:8096",
|
"local_url_placeholder": "http://192.168.1.100:8096",
|
||||||
"home_wifi_networks": "홈 WiFi 네트워크",
|
"home_wifi_networks": "Home WiFi Networks",
|
||||||
"add_current_network": "\"{{ssid}}\" 추가",
|
"add_current_network": "Add \"{{ssid}}\"",
|
||||||
"not_connected_to_wifi": "WiFi에 연결되지 않음",
|
"not_connected_to_wifi": "Not connected to WiFi",
|
||||||
"no_networks_configured": "구성된 네트워크가 없습니다",
|
"no_networks_configured": "No networks configured",
|
||||||
"add_network_hint": "자동 전환을 위한 홈 WiFi 추가",
|
"add_network_hint": "Add your home WiFi network to enable auto-switching",
|
||||||
"current_wifi": "현재 WiFi",
|
"current_wifi": "Current WiFi",
|
||||||
"using_url": "사용중",
|
"using_url": "Using",
|
||||||
"local": "로컬 URL",
|
"local": "Local URL",
|
||||||
"remote": "원격 URL",
|
"remote": "Remote URL",
|
||||||
"not_connected": "연결되지 않았습니다",
|
"not_connected": "Not connected",
|
||||||
"current_server": "현재 서버",
|
"current_server": "Current Server",
|
||||||
"remote_url": "원격 URL",
|
"remote_url": "Remote URL",
|
||||||
"active_url": "현재 사용 중인 URL",
|
"active_url": "Active URL",
|
||||||
"not_configured": "설정되지 않음",
|
"not_configured": "Not configured",
|
||||||
"network_added": "네트워크 추가됨",
|
"network_added": "Network added",
|
||||||
"network_already_added": "네트워크 이미 추가됨",
|
"network_already_added": "Network already added",
|
||||||
"no_wifi_connected": "WiFi에 연결되지 않음",
|
"no_wifi_connected": "Not connected to WiFi",
|
||||||
"permission_denied": "위치 권한이 거부되었습니다",
|
"permission_denied": "Location permission denied",
|
||||||
"permission_denied_explanation": "자동 전환 Wi-Fi 네트워크를 감지하려면 위치 권한이 필요합니다. 설정에서 위치 권한을 활성화해 주세요."
|
"permission_denied_explanation": "Location permission is required to detect WiFi network for auto-switching. Please enable it in Settings."
|
||||||
},
|
},
|
||||||
"user_info": {
|
"user_info": {
|
||||||
"user_info_title": "사용자 정보",
|
"user_info_title": "User Info",
|
||||||
"user": "사용자",
|
"user": "User",
|
||||||
"server": "서버",
|
"server": "Server",
|
||||||
"token": "토큰",
|
"token": "Token",
|
||||||
"app_version": "앱 버전"
|
"app_version": "App Version"
|
||||||
},
|
},
|
||||||
"quick_connect": {
|
"quick_connect": {
|
||||||
"quick_connect_title": "퀵 커넥트",
|
"quick_connect_title": "Quick Connect",
|
||||||
"authorize_button": "퀵 커넥트 승인",
|
"authorize_button": "Authorize Quick Connect",
|
||||||
"enter_the_quick_connect_code": "퀵 커넥트 코드 입력...",
|
"enter_the_quick_connect_code": "Enter the quick connect code...",
|
||||||
"success": "성공",
|
"success": "Success",
|
||||||
"quick_connect_autorized": "퀵 커넥트 승인됨",
|
"quick_connect_autorized": "Quick Connect Authorized",
|
||||||
"error": "오류",
|
"error": "Error",
|
||||||
"invalid_code": "유효하지 않은 코드",
|
"invalid_code": "Invalid Code",
|
||||||
"authorize": "승인"
|
"authorize": "Authorize"
|
||||||
},
|
},
|
||||||
"media_controls": {
|
"media_controls": {
|
||||||
"media_controls_title": "미디어 컨트롤",
|
"media_controls_title": "Media Controls",
|
||||||
"forward_skip_length": "앞으로 건너뛸 시간",
|
"forward_skip_length": "Forward Skip Length",
|
||||||
"rewind_length": "뒤로 되감을 시간",
|
"rewind_length": "Rewind Length",
|
||||||
"seconds_unit": "초"
|
"seconds_unit": "s"
|
||||||
},
|
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
},
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "제스처 제어",
|
"gesture_controls_title": "Gesture Controls",
|
||||||
"horizontal_swipe_skip": "좌/우로 스와이프하여 건너뛰기",
|
"horizontal_swipe_skip": "Horizontal Swipe to Skip",
|
||||||
"horizontal_swipe_skip_description": "컨트롤 숨김상태에서 좌/우로 스와이프하여 건너뛰기",
|
"horizontal_swipe_skip_description": "Swipe left/right when controls are hidden to skip",
|
||||||
"left_side_brightness": "왼쪽 영역 밝기 조정 컨트롤",
|
"left_side_brightness": "Left Side Brightness Control",
|
||||||
"left_side_brightness_description": "왼쪽 영역을 위/아래 스와이프하여 밝기 조절",
|
"left_side_brightness_description": "Swipe up/down on left side to adjust brightness",
|
||||||
"right_side_volume": "오른쪽 영역 볼륨 컨트롤",
|
"right_side_volume": "Right Side Volume Control",
|
||||||
"right_side_volume_description": "오른족 영역을 위/아래로 스와이프 하여 볼륨 조절",
|
"right_side_volume_description": "Swipe up/down on right side to adjust volume",
|
||||||
"hide_volume_slider": "Hide Volume Slider",
|
"hide_volume_slider": "Hide Volume Slider",
|
||||||
"hide_volume_slider_description": "Hide the volume slider in the video player",
|
"hide_volume_slider_description": "Hide the volume slider in the video player",
|
||||||
"hide_brightness_slider": "Hide Brightness Slider",
|
"hide_brightness_slider": "Hide Brightness Slider",
|
||||||
@@ -237,7 +196,7 @@
|
|||||||
"language": "Language",
|
"language": "Language",
|
||||||
"transcode_mode": {
|
"transcode_mode": {
|
||||||
"title": "Audio Transcoding",
|
"title": "Audio Transcoding",
|
||||||
"description": "서라운드 오디오(7.1, TrueHD, DTS-HD)를 어떻게 처리할지 설정합니다",
|
"description": "Controls how surround audio (7.1, TrueHD, DTS-HD) is handled",
|
||||||
"auto": "Auto",
|
"auto": "Auto",
|
||||||
"stereo": "Force Stereo",
|
"stereo": "Force Stereo",
|
||||||
"5_1": "Allow 5.1",
|
"5_1": "Allow 5.1",
|
||||||
@@ -269,68 +228,52 @@
|
|||||||
"outline_opacity": "Outline Opacity",
|
"outline_opacity": "Outline Opacity",
|
||||||
"bold_text": "Bold Text",
|
"bold_text": "Bold Text",
|
||||||
"colors": {
|
"colors": {
|
||||||
"Black": "검정색",
|
"Black": "Black",
|
||||||
"Gray": "회색",
|
"Gray": "Gray",
|
||||||
"Silver": "은색",
|
"Silver": "Silver",
|
||||||
"White": "흰색",
|
"White": "White",
|
||||||
"Maroon": "밤색",
|
"Maroon": "Maroon",
|
||||||
"Red": "빨간색",
|
"Red": "Red",
|
||||||
"Fuchsia": "분홍색",
|
"Fuchsia": "Fuchsia",
|
||||||
"Yellow": "노란색",
|
"Yellow": "Yellow",
|
||||||
"Olive": "올리브 색",
|
"Olive": "Olive",
|
||||||
"Green": "녹색",
|
"Green": "Green",
|
||||||
"Teal": "청록색",
|
"Teal": "Teal",
|
||||||
"Lime": "라임색",
|
"Lime": "Lime",
|
||||||
"Purple": "보라색",
|
"Purple": "Purple",
|
||||||
"Navy": "남색",
|
"Navy": "Navy",
|
||||||
"Blue": "파란색",
|
"Blue": "Blue",
|
||||||
"Aqua": "아쿠아색"
|
"Aqua": "Aqua"
|
||||||
},
|
},
|
||||||
"thickness": {
|
"thickness": {
|
||||||
"None": "없음",
|
"None": "None",
|
||||||
"Thin": "얇게",
|
"Thin": "Thin",
|
||||||
"Normal": "보통",
|
"Normal": "Normal",
|
||||||
"Thick": "굵게"
|
"Thick": "Thick"
|
||||||
},
|
},
|
||||||
"subtitle_color": "자막 색상",
|
"subtitle_color": "Subtitle Color",
|
||||||
"subtitle_background_color": "배경 색상",
|
"subtitle_background_color": "Background Color",
|
||||||
"subtitle_font": "자막 폰트",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer 설정",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "하드웨어 디코딩",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "비디오 디코딩에 하드웨어 가속을 사용하십시오. 재생 문제가 발생하는 경우 비활성화하십시오.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC 자막 설정",
|
"title": "VLC Subtitle Settings",
|
||||||
"hint": "VLC 플레이어의 자막 표시 방식을 설정하세요. 변경 사항은 다음 재생 시 적용됩니다.",
|
"hint": "Customize subtitle appearance for VLC player. Changes take effect on next playback.",
|
||||||
"text_color": "글자색",
|
"text_color": "Text Color",
|
||||||
"background_color": "배경 색상",
|
"background_color": "Background Color",
|
||||||
"background_opacity": "배경 투명도",
|
"background_opacity": "Background Opacity",
|
||||||
"outline_color": "외곽선 색상",
|
"outline_color": "Outline Color",
|
||||||
"outline_opacity": "외곽선 투명도",
|
"outline_opacity": "Outline Opacity",
|
||||||
"outline_thickness": "외곽선 굵기",
|
"outline_thickness": "Outline Thickness",
|
||||||
"bold": "굵은 글씨",
|
"bold": "Bold Text",
|
||||||
"margin": "아래쪽 여백"
|
"margin": "Bottom Margin"
|
||||||
},
|
},
|
||||||
"video_player": {
|
"video_player": {
|
||||||
"title": "비디오 플레이어",
|
"title": "Video Player",
|
||||||
"video_player": "비디오 플레이어",
|
"video_player": "Video Player",
|
||||||
"video_player_description": "iOS 사용자는 비디오 플레이어를 선택하세요.",
|
"video_player_description": "Choose which video player to use on iOS.",
|
||||||
"ksplayer": "KSPlayer",
|
"ksplayer": "KSPlayer",
|
||||||
"vlc": "VLC"
|
"vlc": "VLC"
|
||||||
},
|
},
|
||||||
@@ -345,20 +288,20 @@
|
|||||||
"PORTRAIT_UP": "Portrait Up",
|
"PORTRAIT_UP": "Portrait Up",
|
||||||
"PORTRAIT_DOWN": "Portrait Down",
|
"PORTRAIT_DOWN": "Portrait Down",
|
||||||
"LANDSCAPE": "Landscape",
|
"LANDSCAPE": "Landscape",
|
||||||
"LANDSCAPE_LEFT": "왼쪽 가로 모드",
|
"LANDSCAPE_LEFT": "Landscape Left",
|
||||||
"LANDSCAPE_RIGHT": "오른쪽 가로 모드",
|
"LANDSCAPE_RIGHT": "Landscape Right",
|
||||||
"OTHER": "Other",
|
"OTHER": "Other",
|
||||||
"UNKNOWN": "Unknown"
|
"UNKNOWN": "Unknown"
|
||||||
},
|
},
|
||||||
"safe_area_in_controls": "컨트롤 안전 영역",
|
"safe_area_in_controls": "Safe Area in Controls",
|
||||||
"video_player": "Video Player",
|
"video_player": "Video Player",
|
||||||
"video_players": {
|
"video_players": {
|
||||||
"VLC_3": "VLC 3",
|
"VLC_3": "VLC 3",
|
||||||
"VLC_4": "VLC 4 (Experimental + PiP)"
|
"VLC_4": "VLC 4 (Experimental + PiP)"
|
||||||
},
|
},
|
||||||
"show_custom_menu_links": "사용자 지정 메뉴 링크 표시",
|
"show_custom_menu_links": "Show Custom Menu Links",
|
||||||
"show_large_home_carousel": "대형 홈 슬라이드 배너 표시 (베타)",
|
"show_large_home_carousel": "Show Large Home Carousel (beta)",
|
||||||
"hide_libraries": "라이브러리 숨기기",
|
"hide_libraries": "Hide Libraries",
|
||||||
"select_liraries_you_want_to_hide": "Select the libraries you want to hide from the Library tab and home page sections.",
|
"select_liraries_you_want_to_hide": "Select the libraries you want to hide from the Library tab and home page sections.",
|
||||||
"disable_haptic_feedback": "Disable Haptic Feedback",
|
"disable_haptic_feedback": "Disable Haptic Feedback",
|
||||||
"default_quality": "Default Quality",
|
"default_quality": "Default Quality",
|
||||||
@@ -391,24 +334,24 @@
|
|||||||
"password": "Password",
|
"password": "Password",
|
||||||
"password_placeholder": "Enter password for Jellyfin user {{username}}",
|
"password_placeholder": "Enter password for Jellyfin user {{username}}",
|
||||||
"login_button": "Login",
|
"login_button": "Login",
|
||||||
"total_media_requests": "전체 미디어 요청 수",
|
"total_media_requests": "Total Media Requests",
|
||||||
"movie_quota_limit": "영화 요청 한도",
|
"movie_quota_limit": "Movie Quota Limit",
|
||||||
"movie_quota_days": "영화 요청 제한 기간",
|
"movie_quota_days": "Movie Quota Days",
|
||||||
"tv_quota_limit": "TV Quota Limit",
|
"tv_quota_limit": "TV Quota Limit",
|
||||||
"tv_quota_days": "TV 요청 제한 기간",
|
"tv_quota_days": "TV Quota Days",
|
||||||
"reset_jellyseerr_config_button": "Seerr 설정 초기화",
|
"reset_jellyseerr_config_button": "Reset Seerr Config",
|
||||||
"unlimited": "Unlimited",
|
"unlimited": "Unlimited",
|
||||||
"plus_n_more": "+{{n}}개 더",
|
"plus_n_more": "+{{n}} More",
|
||||||
"order_by": {
|
"order_by": {
|
||||||
"DEFAULT": "Default",
|
"DEFAULT": "Default",
|
||||||
"VOTE_COUNT_AND_AVERAGE": "평균 평점 및 투표 수",
|
"VOTE_COUNT_AND_AVERAGE": "Vote count and average",
|
||||||
"POPULARITY": "Popularity"
|
"POPULARITY": "Popularity"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"marlin_search": {
|
"marlin_search": {
|
||||||
"enable_marlin_search": "Marlin 검색 활성화",
|
"enable_marlin_search": "Enable Marlin Search",
|
||||||
"url": "URL",
|
"url": "URL",
|
||||||
"server_url_placeholder": "http(s)://도메인:포트",
|
"server_url_placeholder": "http(s)://domain.org:port",
|
||||||
"marlin_search_hint": "Enter the URL for the Marlin server. The URL should include http or https and optionally the port.",
|
"marlin_search_hint": "Enter the URL for the Marlin server. The URL should include http or https and optionally the port.",
|
||||||
"read_more_about_marlin": "Read More About Marlin.",
|
"read_more_about_marlin": "Read More About Marlin.",
|
||||||
"save_button": "Save",
|
"save_button": "Save",
|
||||||
@@ -431,28 +374,28 @@
|
|||||||
"features_title": "Features",
|
"features_title": "Features",
|
||||||
"home_sections_title": "Home Sections",
|
"home_sections_title": "Home Sections",
|
||||||
"enable_movie_recommendations": "Movie Recommendations",
|
"enable_movie_recommendations": "Movie Recommendations",
|
||||||
"enable_series_recommendations": "시리즈 추천",
|
"enable_series_recommendations": "Series Recommendations",
|
||||||
"enable_promoted_watchlists": "추천 관심 목록",
|
"enable_promoted_watchlists": "Promoted Watchlists",
|
||||||
"hide_watchlists_tab": "관심 목록 탭 숨기기",
|
"hide_watchlists_tab": "Hide Watchlists Tab",
|
||||||
"home_sections_hint": "홈 페이지에서 Streamystats의 개인 맞춤 추천 및 추천 관심 목록을 표시합니다.",
|
"home_sections_hint": "Show personalized recommendations and promoted watchlists from Streamystats on the home page.",
|
||||||
"recommended_movies": "Recommended Movies",
|
"recommended_movies": "Recommended Movies",
|
||||||
"recommended_series": "추천 시리즈",
|
"recommended_series": "Recommended Series",
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"saved": "Saved",
|
"saved": "Saved",
|
||||||
"refreshed": "서버에서 설정을 새로고침했습니다",
|
"refreshed": "Settings refreshed from server",
|
||||||
"disabled": "Streamystats 비활성화됨"
|
"disabled": "Streamystats disabled"
|
||||||
},
|
},
|
||||||
"refresh_from_server": "서버에서 설정 새로고침"
|
"refresh_from_server": "Refresh Settings from Server"
|
||||||
},
|
},
|
||||||
"kefinTweaks": {
|
"kefinTweaks": {
|
||||||
"watchlist_enabler": "관심 목록 통합 기능 활성화",
|
"watchlist_enabler": "Enable our Watchlist integration",
|
||||||
"watchlist_button": "관심 목록 연동 켜기/끄기"
|
"watchlist_button": "Toggle Watchlist integration"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"storage": {
|
"storage": {
|
||||||
"storage_title": "Storage",
|
"storage_title": "Storage",
|
||||||
"app_usage": "앱 {{usedSpace}}",
|
"app_usage": "App {{usedSpace}}%",
|
||||||
"device_usage": "디바이스 {{availableSpace}}%",
|
"device_usage": "Device {{availableSpace}}%",
|
||||||
"size_used": "{{used}} of {{total}} Used",
|
"size_used": "{{used}} of {{total}} Used",
|
||||||
"delete_all_downloaded_files": "Delete All Downloaded Files",
|
"delete_all_downloaded_files": "Delete All Downloaded Files",
|
||||||
"music_cache_title": "Music Cache",
|
"music_cache_title": "Music Cache",
|
||||||
@@ -460,16 +403,10 @@
|
|||||||
"enable_music_cache": "Enable Music Cache",
|
"enable_music_cache": "Enable Music Cache",
|
||||||
"clear_music_cache": "Clear Music Cache",
|
"clear_music_cache": "Clear Music Cache",
|
||||||
"music_cache_size": "{{size}} cached",
|
"music_cache_size": "{{size}} cached",
|
||||||
"music_cache_cleared": "음악 캐시가 삭제되었습니다",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "다운로드한 노래가 삭제되었습니다",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,26 +430,11 @@
|
|||||||
"error_deleting_files": "Error Deleting Files",
|
"error_deleting_files": "Error Deleting Files",
|
||||||
"background_downloads_enabled": "Background downloads enabled",
|
"background_downloads_enabled": "Background downloads enabled",
|
||||||
"background_downloads_disabled": "Background downloads disabled"
|
"background_downloads_disabled": "Background downloads disabled"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
"title": "Sessions",
|
"title": "Sessions",
|
||||||
"no_active_sessions": "세션 비활성화"
|
"no_active_sessions": "No Active Sessions"
|
||||||
},
|
},
|
||||||
"downloads": {
|
"downloads": {
|
||||||
"downloads_title": "Downloads",
|
"downloads_title": "Downloads",
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "The new update requires content to be downloaded again. Please remove all downloaded content and try again.",
|
"new_app_version_requires_re_download_description": "The new update requires content to be downloaded again. Please remove all downloaded content and try again.",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Something Went Wrong",
|
"something_went_wrong": "Something Went Wrong",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Could not get the stream URL from Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Could not get the stream URL from Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Select",
|
"select": "Select",
|
||||||
"no_trailer_available": "No trailer available",
|
"no_trailer_available": "No trailer available",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Subtitle",
|
"subtitle": "Subtitle",
|
||||||
"play": "Play",
|
"play": "Play",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Search...",
|
"search": "Search...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Movies",
|
"movies": "Movies",
|
||||||
"series": "Series",
|
"series": "Series",
|
||||||
"boxsets": "Box Sets",
|
"boxsets": "Box Sets",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Items"
|
"items": "Items"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Poster",
|
"poster": "Poster",
|
||||||
"cover": "Cover",
|
"cover": "Cover",
|
||||||
"show_titles": "Show Titles",
|
"show_titles": "Show Titles",
|
||||||
"show_stats": "Show Stats",
|
"show_stats": "Show Stats"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Sort By",
|
"sort_by": "Sort By",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Sort Order",
|
"sort_order": "Sort Order",
|
||||||
"tags": "Tags",
|
"tags": "Tags"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "No Links"
|
"no_links": "No Links"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Error",
|
"error": "Error",
|
||||||
"failed_to_get_stream_url": "Failed to get the stream URL",
|
"failed_to_get_stream_url": "Failed to get the stream URL",
|
||||||
"an_error_occured_while_playing_the_video": "An error occurred while playing the video. Check logs in settings.",
|
"an_error_occured_while_playing_the_video": "An error occurred while playing the video. Check logs in settings.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Next Up",
|
"next_up": "Next Up",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Series",
|
"series": "Series",
|
||||||
"seasons": "Seasons",
|
"seasons": "Seasons",
|
||||||
"season": "Season",
|
"season": "Season",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "No episodes for this season",
|
"no_episodes_for_this_season": "No episodes for this season",
|
||||||
"overview": "Overview",
|
"overview": "Overview",
|
||||||
"more_with": "More with {{name}}",
|
"more_with": "More with {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Quality",
|
"quality": "Quality",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitles": {
|
"subtitles": "Subtitle",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Show More",
|
"show_more": "Show More",
|
||||||
"show_less": "Show Less",
|
"show_less": "Show Less",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Appeared In",
|
"appeared_in": "Appeared In",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Could Not Load Item",
|
"could_not_load_item": "Could Not Load Item",
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Download {{item_count}} Items",
|
"download_x_item": "Download {{item_count}} Items",
|
||||||
"download_unwatched_only": "Unwatched Only",
|
"download_unwatched_only": "Unwatched Only",
|
||||||
"download_button": "Download"
|
"download_button": "Download"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Movies",
|
"movies": "Movies",
|
||||||
"sports": "Sports",
|
"sports": "Sports",
|
||||||
"for_kids": "For Kids",
|
"for_kids": "For Kids",
|
||||||
"news": "News",
|
"news": "News"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Confirm",
|
"confirm": "Confirm",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Seerr server does not meet minimum version requirements! Please update to at least 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Seerr server does not meet minimum version requirements! Please update to at least 2.0.0",
|
||||||
"jellyseerr_test_failed": "Seerr test failed. Please try again.",
|
"jellyseerr_test_failed": "Seerr test failed. Please try again.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Search",
|
"search": "Search",
|
||||||
"library": "Library",
|
"library": "Library",
|
||||||
"custom_links": "Custom Links",
|
"custom_links": "Custom Links",
|
||||||
"favorites": "Favorites",
|
"favorites": "Favorites"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Fout",
|
"error_title": "Fout",
|
||||||
"login_title": "Aanmelden",
|
"login_title": "Aanmelden",
|
||||||
"login_to_title": "Aanmelden bij",
|
"login_to_title": "Aanmelden bij",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Gebruikersnaam",
|
"username_placeholder": "Gebruikersnaam",
|
||||||
"password_placeholder": "Wachtwoord",
|
"password_placeholder": "Wachtwoord",
|
||||||
"login_button": "Aanmelden",
|
"login_button": "Aanmelden",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Account selecteren",
|
"select_account": "Account selecteren",
|
||||||
"add_account": "Account toevoegen",
|
"add_account": "Account toevoegen",
|
||||||
"remove_account_description": "Hiermee worden de opgeslagen inloggegevens voor {{username}} verwijderd.",
|
"remove_account_description": "Hiermee worden de opgeslagen inloggegevens voor {{username}} verwijderd."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Account opslaan",
|
"title": "Account opslaan",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Oeps!",
|
"oops": "Oeps!",
|
||||||
"error_message": "Er ging iets fout\nProbeer opnieuw in te loggen.",
|
"error_message": "Er ging iets fout\nProbeer opnieuw in te loggen.",
|
||||||
"continue_watching": "Verder Kijken",
|
"continue_watching": "Verder Kijken",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Volgende",
|
"next_up": "Volgende",
|
||||||
"continue_and_next_up": "Doorgaan & Volgende",
|
"continue_and_next_up": "Doorgaan & Volgende",
|
||||||
"recently_added_in": "Recent toegevoegd in {{libraryName}}",
|
"recently_added_in": "Recent toegevoegd in {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Instellingen",
|
"settings_title": "Instellingen",
|
||||||
"log_out_button": "Afmelden",
|
"log_out_button": "Afmelden",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categorieën"
|
"title": "Categorieën"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Weergave",
|
"title": "Weergave",
|
||||||
"merge_next_up_continue_watching": "Doorgaan met kijken & Volgende samenvoegen",
|
"merge_next_up_continue_watching": "Doorgaan met kijken & Volgende samenvoegen",
|
||||||
"hide_remote_session_button": "Verberg Knop voor Externe Sessie",
|
"hide_remote_session_button": "Verberg Knop voor Externe Sessie"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Netwerk",
|
"title": "Netwerk",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Duur terugspoelen",
|
"rewind_length": "Duur terugspoelen",
|
||||||
"seconds_unit": "s"
|
"seconds_unit": "s"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Gebaar Bediening",
|
"gesture_controls_title": "Gebaar Bediening",
|
||||||
"horizontal_swipe_skip": "Horizontale Swipe om over te slaan",
|
"horizontal_swipe_skip": "Horizontale Swipe om over te slaan",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Lettertype ondertitels",
|
"subtitle_font": "Lettertype ondertitels",
|
||||||
"ksplayer_title": "KSPlayer Instellingen",
|
"ksplayer_title": "KSPlayer Instellingen",
|
||||||
"hardware_decode": "Hardware Acceleratie",
|
"hardware_decode": "Hardware Acceleratie",
|
||||||
"hardware_decode_description": "Gebruik hardware acceleratie voor video-decodering. Uitschakelen als u problemen met afspelen ondervindt.",
|
"hardware_decode_description": "Gebruik hardware acceleratie voor video-decodering. Uitschakelen als u problemen met afspelen ondervindt."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC ondertitel instellingen",
|
"title": "VLC ondertitel instellingen",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Muziek cache gewist",
|
"music_cache_cleared": "Muziek cache gewist",
|
||||||
"delete_all_downloaded_songs": "Verwijder alle gedownloade liedjes",
|
"delete_all_downloaded_songs": "Verwijder alle gedownloade liedjes",
|
||||||
"downloaded_songs_size": "{{size}} gedownload",
|
"downloaded_songs_size": "{{size}} gedownload",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Fout bij het verwijderen van bestanden",
|
"error_deleting_files": "Fout bij het verwijderen van bestanden",
|
||||||
"background_downloads_enabled": "Downloads op de achtergrond ingeschakeld",
|
"background_downloads_enabled": "Downloads op de achtergrond ingeschakeld",
|
||||||
"background_downloads_disabled": "Downloads op de achtergrond uitgeschakeld"
|
"background_downloads_disabled": "Downloads op de achtergrond uitgeschakeld"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Voor de nieuwe update moet de content opnieuw worden gedownload. Verwijder alle gedownloade content en probeer het opnieuw.",
|
"new_app_version_requires_re_download_description": "Voor de nieuwe update moet de content opnieuw worden gedownload. Verwijder alle gedownloade content en probeer het opnieuw.",
|
||||||
"back": "Terug",
|
"back": "Terug",
|
||||||
"delete": "Verwijder",
|
"delete": "Verwijder",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Er ging iets mis",
|
"something_went_wrong": "Er ging iets mis",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Kon de URL van de stream niet krijgen van Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Kon de URL van de stream niet krijgen van Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Selecteren",
|
"select": "Selecteren",
|
||||||
"no_trailer_available": "Geen trailer beschikbaar",
|
"no_trailer_available": "Geen trailer beschikbaar",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitle": "Ondertitel",
|
"subtitle": "Ondertitel",
|
||||||
"play": "Afspelen",
|
"play": "Afspelen",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "Geen",
|
"none": "Geen",
|
||||||
"track": "Spoor",
|
"track": "Spoor",
|
||||||
"cancel": "Annuleren",
|
"cancel": "Annuleren",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Verwijderen",
|
"delete": "Verwijderen",
|
||||||
"ok": "Oké",
|
"ok": "Oké",
|
||||||
"remove": "Verwijderen",
|
"remove": "Verwijderen",
|
||||||
"next": "Volgende",
|
"next": "Volgende",
|
||||||
"back": "Terug",
|
"back": "Terug",
|
||||||
"continue": "Doorgaan",
|
"continue": "Doorgaan",
|
||||||
"verifying": "Verifiëren...",
|
"verifying": "Verifiëren..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Zoek...",
|
"search": "Zoek...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Films",
|
"movies": "Films",
|
||||||
"series": "Series",
|
"series": "Series",
|
||||||
"boxsets": "Boxsets",
|
"boxsets": "Boxsets",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "items"
|
"items": "items"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Poster",
|
"poster": "Poster",
|
||||||
"cover": "Omslag",
|
"cover": "Omslag",
|
||||||
"show_titles": "Toon titels",
|
"show_titles": "Toon titels",
|
||||||
"show_stats": "Toon statistieken",
|
"show_stats": "Toon statistieken"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Sorteren op",
|
"sort_by": "Sorteren op",
|
||||||
"filter_by": "Filteren op",
|
"filter_by": "Filteren op",
|
||||||
"sort_order": "Sorteer volgorde",
|
"sort_order": "Sorteer volgorde",
|
||||||
"tags": "Labels",
|
"tags": "Labels"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Geen links"
|
"no_links": "Geen links"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Fout",
|
"error": "Fout",
|
||||||
"failed_to_get_stream_url": "De stream-URL kon niet worden verkregen",
|
"failed_to_get_stream_url": "De stream-URL kon niet worden verkregen",
|
||||||
"an_error_occured_while_playing_the_video": "Er is een fout opgetreden tijdens het afspelen van de video. Controleer de logs in de instellingen.",
|
"an_error_occured_while_playing_the_video": "Er is een fout opgetreden tijdens het afspelen van de video. Controleer de logs in de instellingen.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Wil je het gedownloade bestand afspelen?",
|
"downloaded_file_message": "Wil je het gedownloade bestand afspelen?",
|
||||||
"downloaded_file_yes": "Ja",
|
"downloaded_file_yes": "Ja",
|
||||||
"downloaded_file_no": "Nee",
|
"downloaded_file_no": "Nee",
|
||||||
"downloaded_file_cancel": "Annuleren",
|
"downloaded_file_cancel": "Annuleren"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Volgende",
|
"next_up": "Volgende",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Series",
|
"series": "Series",
|
||||||
"seasons": "Seizoenen",
|
"seasons": "Seizoenen",
|
||||||
"season": "Seizoen",
|
"season": "Seizoen",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Geen afleveringen voor dit seizoen",
|
"no_episodes_for_this_season": "Geen afleveringen voor dit seizoen",
|
||||||
"overview": "Overzicht",
|
"overview": "Overzicht",
|
||||||
"more_with": "Meer met {{name}}",
|
"more_with": "Meer met {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media opties",
|
"media_options": "Media opties",
|
||||||
"quality": "Kwaliteit",
|
"quality": "Kwaliteit",
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
"subtitles": {
|
"subtitles": "Ondertitel",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Toon meer",
|
"show_more": "Toon meer",
|
||||||
"show_less": "Toon minder",
|
"show_less": "Toon minder",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Verschenen in",
|
"appeared_in": "Verschenen in",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Kon item niet laden",
|
"could_not_load_item": "Kon item niet laden",
|
||||||
"none": "Geen",
|
"none": "Geen",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Download {{item_count}} items",
|
"download_x_item": "Download {{item_count}} items",
|
||||||
"download_unwatched_only": "Alleen niet bekeken",
|
"download_unwatched_only": "Alleen niet bekeken",
|
||||||
"download_button": "Downloaden"
|
"download_button": "Downloaden"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Volgende ",
|
"next": "Volgende ",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Films",
|
"movies": "Films",
|
||||||
"sports": "Sport",
|
"sports": "Sport",
|
||||||
"for_kids": "Voor kinderen",
|
"for_kids": "Voor kinderen",
|
||||||
"news": "Nieuws",
|
"news": "Nieuws"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Bevestig",
|
"confirm": "Bevestig",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Weigeren",
|
"decline": "Weigeren",
|
||||||
"requested_by": "Aangevraagd door {{user}}",
|
"requested_by": "Aangevraagd door {{user}}",
|
||||||
"unknown_user": "Onbekende gebruiker",
|
"unknown_user": "Onbekende gebruiker",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Jellyseerr server voldoet niet aan de minimale versievereisten! Update naar minimaal 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Jellyseerr server voldoet niet aan de minimale versievereisten! Update naar minimaal 2.0.0",
|
||||||
"jellyseerr_test_failed": "Jellyseerr test mislukt. Probeer opnieuw.",
|
"jellyseerr_test_failed": "Jellyseerr test mislukt. Probeer opnieuw.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Zoeken",
|
"search": "Zoeken",
|
||||||
"library": "Bibliotheek",
|
"library": "Bibliotheek",
|
||||||
"custom_links": "Aangepaste links",
|
"custom_links": "Aangepaste links",
|
||||||
"favorites": "Favorieten",
|
"favorites": "Favorieten"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Muziek",
|
"title": "Muziek",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "Deze serie",
|
"show": "Deze serie",
|
||||||
"all": "Alle media (standaard)"
|
"all": "Alle media (standaard)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,6 @@
|
|||||||
"error_title": "Feil",
|
"error_title": "Feil",
|
||||||
"login_title": "Logg inn",
|
"login_title": "Logg inn",
|
||||||
"login_to_title": "Logg inn i",
|
"login_to_title": "Logg inn i",
|
||||||
"select_user": "Select a user to log in",
|
|
||||||
"add_user_to_login": "Add a user to log in",
|
|
||||||
"add_user": "Add User",
|
|
||||||
"username_placeholder": "Brukernavn",
|
"username_placeholder": "Brukernavn",
|
||||||
"password_placeholder": "Passord",
|
"password_placeholder": "Passord",
|
||||||
"login_button": "Logg inn",
|
"login_button": "Logg inn",
|
||||||
@@ -45,13 +42,7 @@
|
|||||||
"accounts_count": "{{count}} accounts",
|
"accounts_count": "{{count}} accounts",
|
||||||
"select_account": "Select Account",
|
"select_account": "Select Account",
|
||||||
"add_account": "Add Account",
|
"add_account": "Add Account",
|
||||||
"remove_account_description": "This will remove the saved credentials for {{username}}.",
|
"remove_account_description": "This will remove the saved credentials for {{username}}."
|
||||||
"remove_server": "Remove Server",
|
|
||||||
"remove_server_description": "This will remove {{server}} and all saved accounts from your list.",
|
|
||||||
"select_your_server": "Select Your Server",
|
|
||||||
"add_server_to_get_started": "Add a server to get started",
|
|
||||||
"add_server": "Add Server",
|
|
||||||
"change_server": "Change Server"
|
|
||||||
},
|
},
|
||||||
"save_account": {
|
"save_account": {
|
||||||
"title": "Save Account",
|
"title": "Save Account",
|
||||||
@@ -95,7 +86,6 @@
|
|||||||
"oops": "Oisann!",
|
"oops": "Oisann!",
|
||||||
"error_message": "Noe gikk galt.\nVennligst logg ut og inn igjen.",
|
"error_message": "Noe gikk galt.\nVennligst logg ut og inn igjen.",
|
||||||
"continue_watching": "Fortsett å se",
|
"continue_watching": "Fortsett å se",
|
||||||
"continue": "Continue",
|
|
||||||
"next_up": "Neste opp",
|
"next_up": "Neste opp",
|
||||||
"continue_and_next_up": "Continue & Next Up",
|
"continue_and_next_up": "Continue & Next Up",
|
||||||
"recently_added_in": "Nylig lagt til i {{libraryName}}",
|
"recently_added_in": "Nylig lagt til i {{libraryName}}",
|
||||||
@@ -119,12 +109,6 @@
|
|||||||
"settings": {
|
"settings": {
|
||||||
"settings_title": "Innstillinger",
|
"settings_title": "Innstillinger",
|
||||||
"log_out_button": "Logg ut",
|
"log_out_button": "Logg ut",
|
||||||
"switch_user": {
|
|
||||||
"title": "Switch User",
|
|
||||||
"account": "Account",
|
|
||||||
"switch_user": "Switch User on This Server",
|
|
||||||
"current": "current"
|
|
||||||
},
|
|
||||||
"categories": {
|
"categories": {
|
||||||
"title": "Categories"
|
"title": "Categories"
|
||||||
},
|
},
|
||||||
@@ -137,16 +121,7 @@
|
|||||||
"appearance": {
|
"appearance": {
|
||||||
"title": "Appearance",
|
"title": "Appearance",
|
||||||
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
"merge_next_up_continue_watching": "Merge Continue Watching & Next Up",
|
||||||
"hide_remote_session_button": "Hide Remote Session Button",
|
"hide_remote_session_button": "Hide Remote Session Button"
|
||||||
"show_home_backdrop": "Dynamic Home Backdrop",
|
|
||||||
"show_hero_carousel": "Hero Carousel",
|
|
||||||
"show_series_poster_on_episode": "Show Series Poster on Episodes",
|
|
||||||
"theme_music": "Theme Music",
|
|
||||||
"display_size": "Display Size",
|
|
||||||
"display_size_small": "Small",
|
|
||||||
"display_size_default": "Default",
|
|
||||||
"display_size_large": "Large",
|
|
||||||
"display_size_extra_large": "Extra Large"
|
|
||||||
},
|
},
|
||||||
"network": {
|
"network": {
|
||||||
"title": "Network",
|
"title": "Network",
|
||||||
@@ -199,22 +174,6 @@
|
|||||||
"rewind_length": "Omspar lengde",
|
"rewind_length": "Omspar lengde",
|
||||||
"seconds_unit": "S"
|
"seconds_unit": "S"
|
||||||
},
|
},
|
||||||
"buffer": {
|
|
||||||
"title": "Buffer Settings",
|
|
||||||
"cache_mode": "Cache Mode",
|
|
||||||
"cache_auto": "Auto",
|
|
||||||
"cache_yes": "Enabled",
|
|
||||||
"cache_no": "Disabled",
|
|
||||||
"buffer_duration": "Buffer Duration",
|
|
||||||
"max_cache_size": "Max Cache Size",
|
|
||||||
"max_backward_cache": "Max Backward Cache"
|
|
||||||
},
|
|
||||||
"vo_driver": {
|
|
||||||
"title": "Video Output",
|
|
||||||
"vo_mode": "VO Driver",
|
|
||||||
"gpu_next": "gpu-next (Recommended)",
|
|
||||||
"gpu": "gpu"
|
|
||||||
},
|
|
||||||
"gesture_controls": {
|
"gesture_controls": {
|
||||||
"gesture_controls_title": "Gest kontroller",
|
"gesture_controls_title": "Gest kontroller",
|
||||||
"horizontal_swipe_skip": "Vannrett sveip for å hoppe over",
|
"horizontal_swipe_skip": "Vannrett sveip for å hoppe over",
|
||||||
@@ -297,23 +256,7 @@
|
|||||||
"subtitle_font": "Subtitle Font",
|
"subtitle_font": "Subtitle Font",
|
||||||
"ksplayer_title": "KSPlayer Settings",
|
"ksplayer_title": "KSPlayer Settings",
|
||||||
"hardware_decode": "Hardware Decoding",
|
"hardware_decode": "Hardware Decoding",
|
||||||
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues.",
|
"hardware_decode_description": "Use hardware acceleration for video decoding. Disable if you experience playback issues."
|
||||||
"opensubtitles_title": "OpenSubtitles",
|
|
||||||
"opensubtitles_hint": "Enter your OpenSubtitles API key to enable client-side subtitle search as a fallback when your Jellyfin server doesn't have a subtitle provider configured.",
|
|
||||||
"opensubtitles_api_key": "API Key",
|
|
||||||
"opensubtitles_api_key_placeholder": "Enter API key...",
|
|
||||||
"opensubtitles_get_key": "Get your free API key at opensubtitles.com/en/consumers",
|
|
||||||
"mpv_subtitle_scale": "Subtitle Scale",
|
|
||||||
"mpv_subtitle_margin_y": "Vertical Margin",
|
|
||||||
"mpv_subtitle_align_x": "Horizontal Align",
|
|
||||||
"mpv_subtitle_align_y": "Vertical Align",
|
|
||||||
"align": {
|
|
||||||
"left": "Left",
|
|
||||||
"center": "Center",
|
|
||||||
"right": "Right",
|
|
||||||
"top": "Top",
|
|
||||||
"bottom": "Bottom"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"vlc_subtitles": {
|
"vlc_subtitles": {
|
||||||
"title": "VLC Subtitle Settings",
|
"title": "VLC Subtitle Settings",
|
||||||
@@ -463,13 +406,7 @@
|
|||||||
"music_cache_cleared": "Music cache cleared",
|
"music_cache_cleared": "Music cache cleared",
|
||||||
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
"delete_all_downloaded_songs": "Delete All Downloaded Songs",
|
||||||
"downloaded_songs_size": "{{size}} downloaded",
|
"downloaded_songs_size": "{{size}} downloaded",
|
||||||
"downloaded_songs_deleted": "Downloaded songs deleted",
|
"downloaded_songs_deleted": "Downloaded songs deleted"
|
||||||
"clear_all_cache": "Clear All Cache",
|
|
||||||
"clear_all_cache_confirm": "Clear All Cache?",
|
|
||||||
"clear_all_cache_confirm_desc": "Are you sure you want to clear all cached data? This will clear all cached images, music files, subtitles, and query caches. Your settings and login session will be kept.",
|
|
||||||
"clear_all_cache_success": "Cache Cleared",
|
|
||||||
"clear_all_cache_success_desc": "All cache has been cleared successfully.",
|
|
||||||
"clear_all_cache_error_desc": "An error occurred while clearing the cache."
|
|
||||||
},
|
},
|
||||||
"intro": {
|
"intro": {
|
||||||
"title": "Intro",
|
"title": "Intro",
|
||||||
@@ -493,21 +430,6 @@
|
|||||||
"error_deleting_files": "Feil ved sletting av filer",
|
"error_deleting_files": "Feil ved sletting av filer",
|
||||||
"background_downloads_enabled": "Nedlastinger av bakgrunn aktivert",
|
"background_downloads_enabled": "Nedlastinger av bakgrunn aktivert",
|
||||||
"background_downloads_disabled": "Bakgrunnsnedlastinger deaktivert"
|
"background_downloads_disabled": "Bakgrunnsnedlastinger deaktivert"
|
||||||
},
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"inactivity_timeout": {
|
|
||||||
"title": "Inactivity Timeout",
|
|
||||||
"description": "Auto logout after inactivity",
|
|
||||||
"disabled": "Disabled",
|
|
||||||
"1_minute": "1 minute",
|
|
||||||
"5_minutes": "5 minutes",
|
|
||||||
"15_minutes": "15 minutes",
|
|
||||||
"30_minutes": "30 minutes",
|
|
||||||
"1_hour": "1 hour",
|
|
||||||
"4_hours": "4 hours",
|
|
||||||
"24_hours": "24 hours"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"sessions": {
|
"sessions": {
|
||||||
@@ -534,7 +456,6 @@
|
|||||||
"new_app_version_requires_re_download_description": "Den nye oppdateringen krever at innholdet lastes ned på nytt. Fjern alt nedlastet innhold og prøv på nytt.",
|
"new_app_version_requires_re_download_description": "Den nye oppdateringen krever at innholdet lastes ned på nytt. Fjern alt nedlastet innhold og prøv på nytt.",
|
||||||
"back": "Tilbake",
|
"back": "Tilbake",
|
||||||
"delete": "Slett",
|
"delete": "Slett",
|
||||||
"delete_download": "Delete Download",
|
|
||||||
"something_went_wrong": "Noe gikk galt",
|
"something_went_wrong": "Noe gikk galt",
|
||||||
"could_not_get_stream_url_from_jellyfin": "Kunne ikke hente stream-URL fra Jellyfin",
|
"could_not_get_stream_url_from_jellyfin": "Kunne ikke hente stream-URL fra Jellyfin",
|
||||||
"eta": "ETA {{eta}}",
|
"eta": "ETA {{eta}}",
|
||||||
@@ -571,28 +492,22 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"no_results": "No Results",
|
|
||||||
"select": "Velg",
|
"select": "Velg",
|
||||||
"no_trailer_available": "Ingen trailer tilgjengelig",
|
"no_trailer_available": "Ingen trailer tilgjengelig",
|
||||||
"video": "Video",
|
"video": "Video",
|
||||||
"audio": "Lyd",
|
"audio": "Lyd",
|
||||||
"subtitle": "Undertittel",
|
"subtitle": "Undertittel",
|
||||||
"play": "Spill",
|
"play": "Spill",
|
||||||
"mark_as_played": "Mark as Played",
|
|
||||||
"mark_as_not_played": "Mark as not Played",
|
|
||||||
"none": "None",
|
"none": "None",
|
||||||
"track": "Track",
|
"track": "Track",
|
||||||
"cancel": "Cancel",
|
"cancel": "Cancel",
|
||||||
"stop": "Stop",
|
|
||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"ok": "OK",
|
"ok": "OK",
|
||||||
"remove": "Remove",
|
"remove": "Remove",
|
||||||
"next": "Next",
|
"next": "Next",
|
||||||
"back": "Back",
|
"back": "Back",
|
||||||
"continue": "Continue",
|
"continue": "Continue",
|
||||||
"verifying": "Verifying...",
|
"verifying": "Verifying..."
|
||||||
"login": "Login",
|
|
||||||
"refresh": "Refresh"
|
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"search": "Søk...",
|
"search": "Søk...",
|
||||||
@@ -641,7 +556,6 @@
|
|||||||
"movies": "Filmer",
|
"movies": "Filmer",
|
||||||
"series": "Serier",
|
"series": "Serier",
|
||||||
"boxsets": "Boks sett",
|
"boxsets": "Boks sett",
|
||||||
"playlists": "Playlists",
|
|
||||||
"items": "Elementer"
|
"items": "Elementer"
|
||||||
},
|
},
|
||||||
"options": {
|
"options": {
|
||||||
@@ -652,8 +566,7 @@
|
|||||||
"poster": "Plakat",
|
"poster": "Plakat",
|
||||||
"cover": "Omslag",
|
"cover": "Omslag",
|
||||||
"show_titles": "Vis titler",
|
"show_titles": "Vis titler",
|
||||||
"show_stats": "Vis statistikk",
|
"show_stats": "Vis statistikk"
|
||||||
"options_title": "Options"
|
|
||||||
},
|
},
|
||||||
"filters": {
|
"filters": {
|
||||||
"genres": "Genres",
|
"genres": "Genres",
|
||||||
@@ -661,11 +574,7 @@
|
|||||||
"sort_by": "Sorter etter",
|
"sort_by": "Sorter etter",
|
||||||
"filter_by": "Filter By",
|
"filter_by": "Filter By",
|
||||||
"sort_order": "Sorter etter",
|
"sort_order": "Sorter etter",
|
||||||
"tags": "Tagger",
|
"tags": "Tagger"
|
||||||
"all": "All",
|
|
||||||
"reset": "Reset",
|
|
||||||
"asc": "Ascending",
|
|
||||||
"desc": "Descending"
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
@@ -682,8 +591,6 @@
|
|||||||
"no_links": "Ingen lenke"
|
"no_links": "Ingen lenke"
|
||||||
},
|
},
|
||||||
"player": {
|
"player": {
|
||||||
"live": "LIVE",
|
|
||||||
"mpv_player_title": "MPV Player",
|
|
||||||
"error": "Feil",
|
"error": "Feil",
|
||||||
"failed_to_get_stream_url": "Kan ikke hente nettadressen for stream",
|
"failed_to_get_stream_url": "Kan ikke hente nettadressen for stream",
|
||||||
"an_error_occured_while_playing_the_video": "En feil oppstod under video. Sjekk loggene i innstillingene.",
|
"an_error_occured_while_playing_the_video": "En feil oppstod under video. Sjekk loggene i innstillingene.",
|
||||||
@@ -701,35 +608,7 @@
|
|||||||
"downloaded_file_message": "Do you want to play the downloaded file?",
|
"downloaded_file_message": "Do you want to play the downloaded file?",
|
||||||
"downloaded_file_yes": "Yes",
|
"downloaded_file_yes": "Yes",
|
||||||
"downloaded_file_no": "No",
|
"downloaded_file_no": "No",
|
||||||
"downloaded_file_cancel": "Cancel",
|
"downloaded_file_cancel": "Cancel"
|
||||||
"swipe_down_settings": "Swipe down for settings",
|
|
||||||
"ends_at": "Ends at {{time}}",
|
|
||||||
"search_subtitles": "Search Subtitles",
|
|
||||||
"subtitle_tracks": "Tracks",
|
|
||||||
"subtitle_search": "Search & Download",
|
|
||||||
"download": "Download",
|
|
||||||
"subtitle_download_hint": "Downloaded subtitles will be saved to your library",
|
|
||||||
"using_jellyfin_server": "Using Jellyfin Server",
|
|
||||||
"language": "Language",
|
|
||||||
"results": "Results",
|
|
||||||
"searching": "Searching...",
|
|
||||||
"search_failed": "Search failed",
|
|
||||||
"no_subtitle_provider": "No subtitle provider configured on server",
|
|
||||||
"no_subtitles_found": "No subtitles found",
|
|
||||||
"add_opensubtitles_key_hint": "Add OpenSubtitles API key in settings for client-side fallback",
|
|
||||||
"settings": "Settings",
|
|
||||||
"skip_intro": "Skip Intro",
|
|
||||||
"skip_credits": "Skip Credits",
|
|
||||||
"stopPlayback": "Stop Playback",
|
|
||||||
"stopPlayingTitle": "Stop playing \"{{title}}\"?",
|
|
||||||
"stopPlayingConfirm": "Are you sure you want to stop playback?",
|
|
||||||
"downloaded": "Downloaded"
|
|
||||||
},
|
|
||||||
"chapters": {
|
|
||||||
"title": "Chapters",
|
|
||||||
"chapter_number": "Chapter {{number}}",
|
|
||||||
"open": "Open chapters",
|
|
||||||
"close": "Close chapters"
|
|
||||||
},
|
},
|
||||||
"item_card": {
|
"item_card": {
|
||||||
"next_up": "Neste opp",
|
"next_up": "Neste opp",
|
||||||
@@ -738,11 +617,6 @@
|
|||||||
"series": "Serier",
|
"series": "Serier",
|
||||||
"seasons": "Sesonger",
|
"seasons": "Sesonger",
|
||||||
"season": "Sesong",
|
"season": "Sesong",
|
||||||
"from_this_series": "From This Series",
|
|
||||||
"more_from_this_season": "More from this Season",
|
|
||||||
"view_series": "View Series",
|
|
||||||
"view_season": "View Season",
|
|
||||||
"select_season": "Select Season",
|
|
||||||
"no_episodes_for_this_season": "Ingen episoder for denne sesongen",
|
"no_episodes_for_this_season": "Ingen episoder for denne sesongen",
|
||||||
"overview": "Oversikt",
|
"overview": "Oversikt",
|
||||||
"more_with": "Mer med {{name}}",
|
"more_with": "Mer med {{name}}",
|
||||||
@@ -753,21 +627,10 @@
|
|||||||
"media_options": "Media Options",
|
"media_options": "Media Options",
|
||||||
"quality": "Kvalitet",
|
"quality": "Kvalitet",
|
||||||
"audio": "Lyd",
|
"audio": "Lyd",
|
||||||
"subtitles": {
|
"subtitles": "Undertittel",
|
||||||
"label": "Subtitle",
|
|
||||||
"none": "None",
|
|
||||||
"tracks": "Tracks"
|
|
||||||
},
|
|
||||||
"show_more": "Vis mer",
|
"show_more": "Vis mer",
|
||||||
"show_less": "Vis mindre",
|
"show_less": "Vis mindre",
|
||||||
"left": "left",
|
|
||||||
"more_info": "More Info",
|
|
||||||
"director": "Director",
|
|
||||||
"cast": "Cast",
|
|
||||||
"technical_details": "Technical Details",
|
|
||||||
"appeared_in": "Ble brukt i",
|
"appeared_in": "Ble brukt i",
|
||||||
"movies": "Movies",
|
|
||||||
"shows": "Shows",
|
|
||||||
"could_not_load_item": "Kan ikke laste inn produkt",
|
"could_not_load_item": "Kan ikke laste inn produkt",
|
||||||
"none": "Ingen",
|
"none": "Ingen",
|
||||||
"download": {
|
"download": {
|
||||||
@@ -778,13 +641,7 @@
|
|||||||
"download_x_item": "Last ned {{item_count}} Objekter",
|
"download_x_item": "Last ned {{item_count}} Objekter",
|
||||||
"download_unwatched_only": "Bare usette",
|
"download_unwatched_only": "Bare usette",
|
||||||
"download_button": "Nedlasting"
|
"download_button": "Nedlasting"
|
||||||
},
|
}
|
||||||
"mark_played": "Mark as Watched",
|
|
||||||
"mark_unplayed": "Mark as Unwatched",
|
|
||||||
"resume_playback": "Resume Playback",
|
|
||||||
"resume_playback_description": "Do you want to continue where you left off or start from the beginning?",
|
|
||||||
"play_from_start": "Play from Start",
|
|
||||||
"continue_from": "Continue from {{time}}"
|
|
||||||
},
|
},
|
||||||
"live_tv": {
|
"live_tv": {
|
||||||
"next": "Neste",
|
"next": "Neste",
|
||||||
@@ -795,18 +652,7 @@
|
|||||||
"movies": "Filmer",
|
"movies": "Filmer",
|
||||||
"sports": "Sport",
|
"sports": "Sport",
|
||||||
"for_kids": "For barn",
|
"for_kids": "For barn",
|
||||||
"news": "Nyheter",
|
"news": "Nyheter"
|
||||||
"page_of": "Page {{current}} of {{total}}",
|
|
||||||
"no_programs": "No programs available",
|
|
||||||
"no_channels": "No channels available",
|
|
||||||
"tabs": {
|
|
||||||
"programs": "Programs",
|
|
||||||
"guide": "Guide",
|
|
||||||
"channels": "Channels",
|
|
||||||
"recordings": "Recordings",
|
|
||||||
"schedule": "Schedule",
|
|
||||||
"series": "Series"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"jellyseerr": {
|
"jellyseerr": {
|
||||||
"confirm": "Bekreft",
|
"confirm": "Bekreft",
|
||||||
@@ -851,12 +697,6 @@
|
|||||||
"decline": "Decline",
|
"decline": "Decline",
|
||||||
"requested_by": "Requested by {{user}}",
|
"requested_by": "Requested by {{user}}",
|
||||||
"unknown_user": "Unknown User",
|
"unknown_user": "Unknown User",
|
||||||
"select": "Select",
|
|
||||||
"request_all": "Request All",
|
|
||||||
"request_seasons": "Request Seasons",
|
|
||||||
"select_seasons": "Select Seasons",
|
|
||||||
"request_selected": "Request Selected",
|
|
||||||
"n_selected": "{{count}} selected",
|
|
||||||
"toasts": {
|
"toasts": {
|
||||||
"jellyseer_does_not_meet_requirements": "Seerr server oppfyller ikke minimumskravene til versjoner! Vennligst oppdater til minst 2.0.0",
|
"jellyseer_does_not_meet_requirements": "Seerr server oppfyller ikke minimumskravene til versjoner! Vennligst oppdater til minst 2.0.0",
|
||||||
"jellyseerr_test_failed": "Seerr-test mislyktes. Vennligst prøv på nytt.",
|
"jellyseerr_test_failed": "Seerr-test mislyktes. Vennligst prøv på nytt.",
|
||||||
@@ -876,8 +716,7 @@
|
|||||||
"search": "Søk",
|
"search": "Søk",
|
||||||
"library": "Bibliotek",
|
"library": "Bibliotek",
|
||||||
"custom_links": "Egendefinerte lenker",
|
"custom_links": "Egendefinerte lenker",
|
||||||
"favorites": "Favoritter",
|
"favorites": "Favoritter"
|
||||||
"settings": "Settings"
|
|
||||||
},
|
},
|
||||||
"music": {
|
"music": {
|
||||||
"title": "Music",
|
"title": "Music",
|
||||||
@@ -1002,36 +841,5 @@
|
|||||||
"show": "This show",
|
"show": "This show",
|
||||||
"all": "All media (default)"
|
"all": "All media (default)"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"companion_login": {
|
|
||||||
"title": "Pair with TV",
|
|
||||||
"align_qr": "Align the QR code within the frame",
|
|
||||||
"enter_code_manually": "Enter code manually",
|
|
||||||
"pairing_enter_credentials": "Enter credentials for TV",
|
|
||||||
"pairing_code_label": "Pairing code",
|
|
||||||
"server": "Server",
|
|
||||||
"authorize_button": "Authorize",
|
|
||||||
"authorizing": "Authorizing...",
|
|
||||||
"scan_again": "Scan Again",
|
|
||||||
"done": "Done",
|
|
||||||
"success_title": "Authorization Sent",
|
|
||||||
"pairing_tv_connecting": "The TV is connecting to your account",
|
|
||||||
"error_title": "Authorization Failed",
|
|
||||||
"error_invalid_qr": "Invalid QR code. Please scan the TV pairing code.",
|
|
||||||
"error_generic": "Something went wrong. Please try again.",
|
|
||||||
"error_permission_denied": "Camera permission is required to scan QR codes.",
|
|
||||||
"login_as": "Log in as {{username}}?",
|
|
||||||
"on_server": "on {{server}}",
|
|
||||||
"use_different_user": "Use a different user",
|
|
||||||
"open_settings": "Open Settings"
|
|
||||||
},
|
|
||||||
"pairing": {
|
|
||||||
"pair_with_phone": "Pair with Phone",
|
|
||||||
"pair_with_phone_title": "Login TV",
|
|
||||||
"pair_with_phone_description": "Scan the QR code displayed on your TV to log in",
|
|
||||||
"waiting_for_phone": "Waiting for phone...",
|
|
||||||
"scan_with_phone": "Scan with the Streamyfin app on your phone",
|
|
||||||
"logging_in": "Logging in...",
|
|
||||||
"logging_in_description": "Connecting to your server"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||