mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-12-31 11:38:47 -05:00
Compare commits
135 Commits
v2.21.0
...
sanitize_h
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9052ceedd3 | ||
|
|
4968864498 | ||
|
|
f44c2d9e11 | ||
|
|
0c8e334b1a | ||
|
|
abaa7b5ad0 | ||
|
|
df01e493ec | ||
|
|
949c8ce230 | ||
|
|
9eaa0c26cd | ||
|
|
d71f091e3e | ||
|
|
2589121908 | ||
|
|
ff425212e7 | ||
|
|
243baaf775 | ||
|
|
7275b1063b | ||
|
|
4fd97510b8 | ||
|
|
6e67b1d9dd | ||
|
|
0fc6afec26 | ||
|
|
c950ac7d69 | ||
|
|
8979e19e92 | ||
|
|
6a51cb07e8 | ||
|
|
846a8c3881 | ||
|
|
0cd698cc8d | ||
|
|
13d9462868 | ||
|
|
d8e2ff8b0e | ||
|
|
35c2a5c1a3 | ||
|
|
19dc096d22 | ||
|
|
535ebc10f0 | ||
|
|
7486a0659b | ||
|
|
273866fe92 | ||
|
|
6425d95deb | ||
|
|
68a39449a2 | ||
|
|
8e08458ea2 | ||
|
|
1119ddef8a | ||
|
|
3d0219a866 | ||
|
|
6ce1806359 | ||
|
|
f05a513767 | ||
|
|
d03c338b48 | ||
|
|
5e5a988f7a | ||
|
|
6d1f0b27df | ||
|
|
d01a7cb756 | ||
|
|
cae874ef05 | ||
|
|
733afc3e29 | ||
|
|
0772730336 | ||
|
|
8b02fe07c8 | ||
|
|
98f93a665c | ||
|
|
754566b221 | ||
|
|
f4f9adad35 | ||
|
|
16f7f1166e | ||
|
|
f527b0f4d5 | ||
|
|
4f41df53c9 | ||
|
|
8a15f775a2 | ||
|
|
5e83bcd283 | ||
|
|
2fd5dfcb66 | ||
|
|
872ce4fa38 | ||
|
|
ba792d91e5 | ||
|
|
4997c716db | ||
|
|
fd72d05280 | ||
|
|
241b56ad45 | ||
|
|
635c384952 | ||
|
|
ef930fd1b4 | ||
|
|
49997a1336 | ||
|
|
8d0434143c | ||
|
|
8e0319994e | ||
|
|
0ed6045d1e | ||
|
|
25c7e95a64 | ||
|
|
1781c4bbcb | ||
|
|
c4ce72d44e | ||
|
|
78813c4b28 | ||
|
|
990baa2dc6 | ||
|
|
c85f4467d2 | ||
|
|
59f7609054 | ||
|
|
2ef827e3fa | ||
|
|
5cadc8d90f | ||
|
|
40e7e36ef6 | ||
|
|
d60ad96f8a | ||
|
|
46ba342d49 | ||
|
|
ace6b2b81f | ||
|
|
fa7e2dfafe | ||
|
|
015310c15d | ||
|
|
f624f04dec | ||
|
|
7c13cfcda2 | ||
|
|
fc265dadae | ||
|
|
f9905f887e | ||
|
|
eb72bfbbc0 | ||
|
|
c268cace09 | ||
|
|
9666caf7a3 | ||
|
|
9e01e5c24e | ||
|
|
25e613a867 | ||
|
|
fe23a86eaa | ||
|
|
cb5a7d6aef | ||
|
|
7deb89ce7a | ||
|
|
1e300c77c9 | ||
|
|
ed7cc42959 | ||
|
|
f681ff68a1 | ||
|
|
ba112bf9c2 | ||
|
|
718434545a | ||
|
|
0e9a4c95a9 | ||
|
|
3c997c8468 | ||
|
|
eb49646256 | ||
|
|
c54b5eadfd | ||
|
|
659c671c25 | ||
|
|
0df5a7816d | ||
|
|
26c976b6b9 | ||
|
|
bdeb22615e | ||
|
|
257bf2ebe0 | ||
|
|
fc33da447a | ||
|
|
df45347690 | ||
|
|
b876256736 | ||
|
|
3ce6e45761 | ||
|
|
5ac6b85da1 | ||
|
|
69e0a0732a | ||
|
|
087835a9f3 | ||
|
|
1f7b181b7b | ||
|
|
1afb8840db | ||
|
|
d9531166b6 | ||
|
|
336de49d8d | ||
|
|
3cc527484d | ||
|
|
45987ffd63 | ||
|
|
1a1ef9c378 | ||
|
|
342d100f3e | ||
|
|
e0b90c6813 | ||
|
|
2706a9c4aa | ||
|
|
2cc9d1b7f8 | ||
|
|
2b7268c952 | ||
|
|
e097fe1e88 | ||
|
|
6819c0b108 | ||
|
|
58cd751b43 | ||
|
|
9f834a5345 | ||
|
|
5eaf9c69ad | ||
|
|
a1074e69ac | ||
|
|
65aec6a099 | ||
|
|
38957d4f32 | ||
|
|
a2dc76e190 | ||
|
|
dc3c978f8d | ||
|
|
13fac2d5bc | ||
|
|
fd0af6b2dd |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -23,3 +23,4 @@ sw.*
|
||||
.DS_STORE
|
||||
.idea/*
|
||||
tailwind.compiled.css
|
||||
tailwind.config.js
|
||||
|
||||
50
Dockerfile
50
Dockerfile
@@ -1,34 +1,32 @@
|
||||
ARG NUSQLITE3_DIR="/usr/local/lib/nusqlite3"
|
||||
ARG NUSQLITE3_PATH="${NUSQLITE3_DIR}/libnusqlite3.so"
|
||||
|
||||
### STAGE 0: Build client ###
|
||||
FROM node:20-alpine AS build
|
||||
FROM node:20-alpine AS build-client
|
||||
|
||||
WORKDIR /client
|
||||
COPY /client /client
|
||||
RUN npm ci && npm cache clean --force
|
||||
RUN npm run generate
|
||||
|
||||
### STAGE 1: Build server ###
|
||||
FROM node:20-alpine
|
||||
FROM node:20-alpine AS build-server
|
||||
|
||||
ARG NUSQLITE3_DIR
|
||||
ARG TARGETPLATFORM
|
||||
|
||||
ENV NODE_ENV=production
|
||||
|
||||
RUN apk update && \
|
||||
apk add --no-cache --update \
|
||||
RUN apk add --no-cache --update \
|
||||
curl \
|
||||
tzdata \
|
||||
ffmpeg \
|
||||
make \
|
||||
python3 \
|
||||
g++ \
|
||||
tini \
|
||||
unzip
|
||||
|
||||
COPY --from=build /client/dist /client/dist
|
||||
COPY index.js package* /
|
||||
COPY server server
|
||||
|
||||
ARG TARGETPLATFORM
|
||||
|
||||
ENV NUSQLITE3_DIR="/usr/local/lib/nusqlite3"
|
||||
ENV NUSQLITE3_PATH="${NUSQLITE3_DIR}/libnusqlite3.so"
|
||||
WORKDIR /server
|
||||
COPY index.js package* /server
|
||||
COPY /server /server/server
|
||||
|
||||
RUN case "$TARGETPLATFORM" in \
|
||||
"linux/amd64") \
|
||||
@@ -42,14 +40,34 @@ RUN case "$TARGETPLATFORM" in \
|
||||
|
||||
RUN npm ci --only=production
|
||||
|
||||
RUN apk del make python3 g++
|
||||
### STAGE 2: Create minimal runtime image ###
|
||||
FROM node:20-alpine
|
||||
|
||||
ARG NUSQLITE3_DIR
|
||||
ARG NUSQLITE3_PATH
|
||||
|
||||
# Install only runtime dependencies
|
||||
RUN apk add --no-cache --update \
|
||||
tzdata \
|
||||
ffmpeg \
|
||||
tini
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# Copy compiled frontend and server from build stages
|
||||
COPY --from=build-client /client/dist /app/client/dist
|
||||
COPY --from=build-server /server /app
|
||||
COPY --from=build-server /usr/local/lib/nusqlite3 /usr/local/lib/nusqlite3
|
||||
|
||||
EXPOSE 80
|
||||
|
||||
ENV PORT=80
|
||||
ENV NODE_ENV=production
|
||||
ENV CONFIG_PATH="/config"
|
||||
ENV METADATA_PATH="/metadata"
|
||||
ENV SOURCE="docker"
|
||||
ENV NUSQLITE3_DIR=${NUSQLITE3_DIR}
|
||||
ENV NUSQLITE3_PATH=${NUSQLITE3_PATH}
|
||||
|
||||
ENTRYPOINT ["tini", "--"]
|
||||
CMD ["node", "index.js"]
|
||||
|
||||
@@ -217,6 +217,16 @@ export default {
|
||||
})
|
||||
}
|
||||
|
||||
if (this.results.episodes?.length) {
|
||||
shelves.push({
|
||||
id: 'episodes',
|
||||
label: 'Episodes',
|
||||
labelStringKey: 'LabelEpisodes',
|
||||
type: 'episode',
|
||||
entities: this.results.episodes.map((res) => res.libraryItem)
|
||||
})
|
||||
}
|
||||
|
||||
if (this.results.series?.length) {
|
||||
shelves.push({
|
||||
id: 'series',
|
||||
|
||||
@@ -274,15 +274,10 @@ export default {
|
||||
isAuthorsPage() {
|
||||
return this.page === 'authors'
|
||||
},
|
||||
isAlbumsPage() {
|
||||
return this.page === 'albums'
|
||||
},
|
||||
numShowing() {
|
||||
return this.totalEntities
|
||||
},
|
||||
entityName() {
|
||||
if (this.isAlbumsPage) return 'Albums'
|
||||
|
||||
if (this.isPodcastLibrary) return this.$strings.LabelPodcasts
|
||||
if (!this.page) return this.$strings.LabelBooks
|
||||
if (this.isSeriesPage) return this.$strings.LabelSeries
|
||||
|
||||
@@ -0,0 +1,60 @@
|
||||
<template>
|
||||
<div class="flex items-center h-full px-1 overflow-hidden">
|
||||
<covers-book-cover :library-item="libraryItem" :width="coverWidth" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
||||
<div class="grow px-2 episodeSearchCardContent">
|
||||
<p class="truncate text-sm">{{ episodeTitle }}</p>
|
||||
<p class="text-xs text-gray-200 truncate">{{ podcastTitle }}</p>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
libraryItem: {
|
||||
type: Object,
|
||||
default: () => {}
|
||||
},
|
||||
episode: {
|
||||
type: Object,
|
||||
default: () => {}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {}
|
||||
},
|
||||
computed: {
|
||||
bookCoverAspectRatio() {
|
||||
return this.$store.getters['libraries/getBookCoverAspectRatio']
|
||||
},
|
||||
coverWidth() {
|
||||
if (this.bookCoverAspectRatio === 1) return 50 * 1.2
|
||||
return 50
|
||||
},
|
||||
media() {
|
||||
return this.libraryItem?.media || {}
|
||||
},
|
||||
mediaMetadata() {
|
||||
return this.media.metadata || {}
|
||||
},
|
||||
episodeTitle() {
|
||||
return this.episode.title || 'No Title'
|
||||
},
|
||||
podcastTitle() {
|
||||
return this.mediaMetadata.title || 'No Title'
|
||||
}
|
||||
},
|
||||
methods: {},
|
||||
mounted() {}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style>
|
||||
.episodeSearchCardContent {
|
||||
width: calc(100% - 80px);
|
||||
height: 75px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -1,142 +0,0 @@
|
||||
<template>
|
||||
<div ref="card" :id="`album-card-${index}`" :style="{ width: cardWidth + 'px' }" class="absolute top-0 left-0 rounded-xs z-30 cursor-pointer" @mousedown.prevent @mouseup.prevent @mousemove.prevent @mouseover="mouseover" @mouseleave="mouseleave" @click="clickCard">
|
||||
<div class="relative" :style="{ height: coverHeight + 'px' }">
|
||||
<div class="absolute top-0 left-0 w-full box-shadow-book shadow-height" />
|
||||
<div class="w-full h-full bg-primary relative rounded-sm overflow-hidden">
|
||||
<covers-preview-cover ref="cover" :src="coverSrc" :width="cardWidth" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="relative w-full">
|
||||
<div v-if="!isAlternativeBookshelfView" class="categoryPlacard absolute z-30 left-0 right-0 mx-auto -bottom-6e h-6e rounded-md text-center" :style="{ width: Math.min(200, cardWidth) + 'px' }">
|
||||
<div class="w-full h-full shinyBlack flex items-center justify-center rounded-xs border" :style="{ padding: `0em ${0.5}em` }">
|
||||
<p class="truncate" :style="{ fontSize: labelFontSize + 'em' }">{{ title }}</p>
|
||||
</div>
|
||||
</div>
|
||||
<div v-else class="absolute z-30 left-0 right-0 mx-auto -bottom-8e h-8e py-1e rounded-md text-center">
|
||||
<p class="truncate" :style="{ fontSize: labelFontSize + 'em' }">{{ title }}</p>
|
||||
<p class="truncate text-gray-400" :style="{ fontSize: 0.8 + 'em' }">{{ artist || ' ' }}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
index: Number,
|
||||
width: Number,
|
||||
height: {
|
||||
type: Number,
|
||||
default: 192
|
||||
},
|
||||
bookshelfView: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
albumMount: {
|
||||
type: Object,
|
||||
default: () => null
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
album: null,
|
||||
isSelectionMode: false,
|
||||
selected: false,
|
||||
isHovering: false
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
bookCoverAspectRatio() {
|
||||
return this.store.getters['libraries/getBookCoverAspectRatio']
|
||||
},
|
||||
cardWidth() {
|
||||
return this.width || this.coverHeight
|
||||
},
|
||||
coverHeight() {
|
||||
return this.height * this.sizeMultiplier
|
||||
},
|
||||
/*
|
||||
cardHeight() {
|
||||
return this.coverHeight + this.bottomTextHeight
|
||||
},
|
||||
bottomTextHeight() {
|
||||
if (!this.isAlternativeBookshelfView) return 0
|
||||
const lineHeight = 1.5
|
||||
const remSize = 16
|
||||
const baseHeight = this.sizeMultiplier * lineHeight * remSize
|
||||
const titleHeight = this.labelFontSize * baseHeight
|
||||
const paddingHeight = 4 * 2 * this.sizeMultiplier // py-1
|
||||
return titleHeight + paddingHeight
|
||||
},
|
||||
*/
|
||||
coverSrc() {
|
||||
const config = this.$config || this.$nuxt.$config
|
||||
if (!this.album || !this.album.libraryItemId) return `${config.routerBasePath}/book_placeholder.jpg`
|
||||
return this.store.getters['globals/getLibraryItemCoverSrcById'](this.album.libraryItemId)
|
||||
},
|
||||
labelFontSize() {
|
||||
if (this.width < 160) return 0.75
|
||||
return 0.9
|
||||
},
|
||||
sizeMultiplier() {
|
||||
return this.store.getters['user/getSizeMultiplier']
|
||||
},
|
||||
title() {
|
||||
return this.album ? this.album.title : ''
|
||||
},
|
||||
artist() {
|
||||
return this.album ? this.album.artist : ''
|
||||
},
|
||||
store() {
|
||||
return this.$store || this.$nuxt.$store
|
||||
},
|
||||
currentLibraryId() {
|
||||
return this.store.state.libraries.currentLibraryId
|
||||
},
|
||||
isAlternativeBookshelfView() {
|
||||
const constants = this.$constants || this.$nuxt.$constants
|
||||
return this.bookshelfView == constants.BookshelfView.DETAIL
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
setEntity(album) {
|
||||
this.album = album
|
||||
},
|
||||
setSelectionMode(val) {
|
||||
this.isSelectionMode = val
|
||||
},
|
||||
mouseover() {
|
||||
this.isHovering = true
|
||||
},
|
||||
mouseleave() {
|
||||
this.isHovering = false
|
||||
},
|
||||
clickCard() {
|
||||
if (!this.album) return
|
||||
// const router = this.$router || this.$nuxt.$router
|
||||
// router.push(`/album/${this.$encode(this.title)}`)
|
||||
},
|
||||
clickEdit() {
|
||||
this.$emit('edit', this.album)
|
||||
},
|
||||
destroy() {
|
||||
// destroy the vue listeners, etc
|
||||
this.$destroy()
|
||||
|
||||
// remove the element from the DOM
|
||||
if (this.$el && this.$el.parentNode) {
|
||||
this.$el.parentNode.removeChild(this.$el)
|
||||
} else if (this.$el && this.$el.remove) {
|
||||
this.$el.remove()
|
||||
}
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.albumMount) {
|
||||
this.setEntity(this.albumMount)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
@@ -39,6 +39,15 @@
|
||||
</li>
|
||||
</template>
|
||||
|
||||
<p v-if="episodeResults.length" class="uppercase text-xs text-gray-400 my-1 px-1 font-semibold">{{ $strings.LabelEpisodes }}</p>
|
||||
<template v-for="item in episodeResults">
|
||||
<li :key="item.libraryItem.recentEpisode.id" class="text-gray-50 select-none relative cursor-pointer hover:bg-black-400 py-1" role="option" @click="clickOption">
|
||||
<nuxt-link :to="`/item/${item.libraryItem.id}`">
|
||||
<cards-episode-search-card :episode="item.libraryItem.recentEpisode" :library-item="item.libraryItem" />
|
||||
</nuxt-link>
|
||||
</li>
|
||||
</template>
|
||||
|
||||
<p v-if="authorResults.length" class="uppercase text-xs text-gray-400 mb-1 mt-3 px-1 font-semibold">{{ $strings.LabelAuthors }}</p>
|
||||
<template v-for="item in authorResults">
|
||||
<li :key="item.id" class="text-gray-50 select-none relative cursor-pointer hover:bg-black-400 py-1" role="option" @click="clickOption">
|
||||
@@ -100,6 +109,7 @@ export default {
|
||||
isFetching: false,
|
||||
search: null,
|
||||
podcastResults: [],
|
||||
episodeResults: [],
|
||||
bookResults: [],
|
||||
authorResults: [],
|
||||
seriesResults: [],
|
||||
@@ -115,7 +125,7 @@ export default {
|
||||
return this.$store.state.libraries.currentLibraryId
|
||||
},
|
||||
totalResults() {
|
||||
return this.bookResults.length + this.seriesResults.length + this.authorResults.length + this.tagResults.length + this.genreResults.length + this.podcastResults.length + this.narratorResults.length
|
||||
return this.bookResults.length + this.seriesResults.length + this.authorResults.length + this.tagResults.length + this.genreResults.length + this.podcastResults.length + this.narratorResults.length + this.episodeResults.length
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
@@ -132,6 +142,7 @@ export default {
|
||||
this.search = null
|
||||
this.lastSearch = null
|
||||
this.podcastResults = []
|
||||
this.episodeResults = []
|
||||
this.bookResults = []
|
||||
this.authorResults = []
|
||||
this.seriesResults = []
|
||||
@@ -175,6 +186,7 @@ export default {
|
||||
if (!this.isFetching) return
|
||||
|
||||
this.podcastResults = searchResults.podcast || []
|
||||
this.episodeResults = searchResults.episodes || []
|
||||
this.bookResults = searchResults.book || []
|
||||
this.authorResults = searchResults.authors || []
|
||||
this.seriesResults = searchResults.series || []
|
||||
|
||||
@@ -276,6 +276,11 @@ export default {
|
||||
text: this.$strings.ButtonIssues,
|
||||
value: 'issues',
|
||||
sublist: false
|
||||
},
|
||||
{
|
||||
text: this.$strings.LabelRSSFeedOpen,
|
||||
value: 'feed-open',
|
||||
sublist: false
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
@@ -14,6 +14,7 @@
|
||||
<ui-text-input-with-label ref="sequenceInput" v-model="selectedSeries.sequence" :label="$strings.LabelSequence" />
|
||||
</div>
|
||||
</div>
|
||||
<div v-if="error" class="text-error text-sm mt-2 p-1">{{ error }}</div>
|
||||
<div class="flex justify-end mt-2 p-1">
|
||||
<ui-btn type="submit">{{ $strings.ButtonSubmit }}</ui-btn>
|
||||
</div>
|
||||
@@ -34,12 +35,17 @@ export default {
|
||||
existingSeriesNames: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
originalSeriesSequence: {
|
||||
type: String,
|
||||
default: null
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
el: null,
|
||||
content: null
|
||||
content: null,
|
||||
error: null
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
@@ -85,10 +91,17 @@ export default {
|
||||
}
|
||||
},
|
||||
submitSeriesForm() {
|
||||
this.error = null
|
||||
|
||||
if (this.$refs.newSeriesSelect) {
|
||||
this.$refs.newSeriesSelect.blur()
|
||||
}
|
||||
|
||||
if (this.selectedSeries.sequence !== this.originalSeriesSequence && this.selectedSeries.sequence.includes(' ')) {
|
||||
this.error = this.$strings.MessageSeriesSequenceCannotContainSpaces
|
||||
return
|
||||
}
|
||||
|
||||
this.$emit('submit')
|
||||
},
|
||||
clickClose() {
|
||||
@@ -100,6 +113,7 @@ export default {
|
||||
}
|
||||
},
|
||||
setShow() {
|
||||
this.error = null
|
||||
if (!this.el || !this.content) {
|
||||
this.init()
|
||||
}
|
||||
|
||||
@@ -74,19 +74,12 @@ export default {
|
||||
mediaTracks() {
|
||||
return this.media.tracks || []
|
||||
},
|
||||
isSingleM4b() {
|
||||
return this.mediaTracks.length === 1 && this.mediaTracks[0].metadata.ext.toLowerCase() === '.m4b'
|
||||
},
|
||||
chapters() {
|
||||
return this.media.chapters || []
|
||||
},
|
||||
showM4bDownload() {
|
||||
if (!this.mediaTracks.length) return false
|
||||
return !this.isSingleM4b
|
||||
},
|
||||
showMp3Split() {
|
||||
if (!this.mediaTracks.length) return false
|
||||
return this.isSingleM4b && this.chapters.length
|
||||
return true
|
||||
},
|
||||
queuedEmbedLIds() {
|
||||
return this.$store.state.tasks.queuedEmbedLIds || []
|
||||
|
||||
@@ -244,8 +244,8 @@ export default {
|
||||
const sizeInMb = payloadSize / 1024 / 1024
|
||||
const sizeInMbPretty = sizeInMb.toFixed(2) + 'MB'
|
||||
console.log('Request size', sizeInMb)
|
||||
if (sizeInMb > 4.99) {
|
||||
return this.$toast.error(`Request is too large (${sizeInMbPretty}) should be < 5Mb`)
|
||||
if (sizeInMb > 9.99) {
|
||||
return this.$toast.error(`Request is too large (${sizeInMbPretty}) should be < 10Mb`)
|
||||
}
|
||||
|
||||
this.processing = true
|
||||
|
||||
@@ -74,6 +74,9 @@ export default {
|
||||
currentChapterStart() {
|
||||
if (!this.currentChapter) return 0
|
||||
return this.currentChapter.start
|
||||
},
|
||||
isMobile() {
|
||||
return this.$store.state.globals.isMobile
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
@@ -145,6 +148,9 @@ export default {
|
||||
})
|
||||
},
|
||||
mousemoveTrack(e) {
|
||||
if (this.isMobile) {
|
||||
return
|
||||
}
|
||||
const offsetX = e.offsetX
|
||||
|
||||
const baseTime = this.useChapterTrack ? this.currentChapterStart : 0
|
||||
@@ -198,6 +204,7 @@ export default {
|
||||
setTrackWidth() {
|
||||
if (this.$refs.track) {
|
||||
this.trackWidth = this.$refs.track.clientWidth
|
||||
this.trackOffsetLeft = this.$refs.track.getBoundingClientRect().left
|
||||
} else {
|
||||
console.error('Track not loaded', this.$refs)
|
||||
}
|
||||
|
||||
@@ -164,14 +164,15 @@ export default {
|
||||
beforeMount() {
|
||||
this.yearInReviewYear = new Date().getFullYear()
|
||||
|
||||
// When not December show previous year
|
||||
if (new Date().getMonth() < 11) {
|
||||
this.availableYears = this.getAvailableYears()
|
||||
const availableYearValues = this.availableYears.map((y) => y.value)
|
||||
|
||||
// When not December show previous year if data is available
|
||||
if (new Date().getMonth() < 11 && availableYearValues.includes(this.yearInReviewYear - 1)) {
|
||||
this.yearInReviewYear--
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.availableYears = this.getAvailableYears()
|
||||
|
||||
if (typeof navigator.share !== 'undefined' && navigator.share) {
|
||||
this.showShareButton = true
|
||||
} else {
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
|
||||
<template>
|
||||
<div id="lazy-episodes-table" class="w-full py-6">
|
||||
<div class="flex flex-wrap flex-col md:flex-row md:items-center mb-4">
|
||||
@@ -176,6 +175,13 @@ export default {
|
||||
return episodeProgress && !episodeProgress.isFinished
|
||||
})
|
||||
.sort((a, b) => {
|
||||
// Swap values if sort descending
|
||||
if (this.sortDesc) {
|
||||
const temp = a
|
||||
a = b
|
||||
b = temp
|
||||
}
|
||||
|
||||
let aValue
|
||||
let bValue
|
||||
|
||||
@@ -194,10 +200,23 @@ export default {
|
||||
if (!bValue) bValue = Number.MAX_VALUE
|
||||
}
|
||||
|
||||
if (this.sortDesc) {
|
||||
return String(bValue).localeCompare(String(aValue), undefined, { numeric: true, sensitivity: 'base' })
|
||||
const primaryCompare = String(aValue).localeCompare(String(bValue), undefined, { numeric: true, sensitivity: 'base' })
|
||||
if (primaryCompare !== 0 || this.sortKey === 'publishedAt') return primaryCompare
|
||||
|
||||
// When sorting by season, secondary sort is by episode number
|
||||
if (this.sortKey === 'season') {
|
||||
const aEpisode = a.episode || ''
|
||||
const bEpisode = b.episode || ''
|
||||
|
||||
const secondaryCompare = String(aEpisode).localeCompare(String(bEpisode), undefined, { numeric: true, sensitivity: 'base' })
|
||||
if (secondaryCompare !== 0) return secondaryCompare
|
||||
}
|
||||
return String(aValue).localeCompare(String(bValue), undefined, { numeric: true, sensitivity: 'base' })
|
||||
|
||||
// Final sort by publishedAt
|
||||
let aPubDate = a.publishedAt || Number.MAX_VALUE
|
||||
let bPubDate = b.publishedAt || Number.MAX_VALUE
|
||||
|
||||
return String(aPubDate).localeCompare(String(bPubDate), undefined, { numeric: true, sensitivity: 'base' })
|
||||
})
|
||||
},
|
||||
episodesList() {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<div class="inline-flex toggle-btn-wrapper shadow-md">
|
||||
<button v-for="item in items" :key="item.value" type="button" class="toggle-btn outline-hidden relative border border-gray-600 px-4 py-1" :class="{ selected: item.value === value }" @click.stop="clickBtn(item.value)">
|
||||
<button v-for="item in items" :key="item.value" type="button" :disabled="disabled" class="toggle-btn outline-hidden relative border border-gray-600 px-4 py-1" :class="{ selected: item.value === value }" @click.stop="clickBtn(item.value)">
|
||||
{{ item.text }}
|
||||
</button>
|
||||
</div>
|
||||
@@ -9,13 +9,17 @@
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
value: String,
|
||||
value: [String, Number],
|
||||
/**
|
||||
* [{ "text", "", "value": "" }]
|
||||
*/
|
||||
items: {
|
||||
type: Array,
|
||||
default: Object
|
||||
},
|
||||
disabled: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -76,10 +80,19 @@ export default {
|
||||
.toggle-btn.selected {
|
||||
color: white;
|
||||
}
|
||||
.toggle-btn.selected:disabled {
|
||||
color: white;
|
||||
}
|
||||
.toggle-btn.selected::before {
|
||||
background-color: rgba(255, 255, 255, 0.1);
|
||||
}
|
||||
button.toggle-btn.selected:disabled::before {
|
||||
background-color: rgba(255, 255, 255, 0.05);
|
||||
}
|
||||
button.toggle-btn:disabled::before {
|
||||
background-color: rgba(0, 0, 0, 0.2);
|
||||
}
|
||||
button.toggle-btn:disabled {
|
||||
cursor: not-allowed;
|
||||
}
|
||||
</style>
|
||||
@@ -31,7 +31,7 @@
|
||||
</div>
|
||||
</div>
|
||||
</trix-toolbar>
|
||||
<trix-editor :toolbar="toolbarId" :contenteditable="!disabledEditor" :class="['trix-content']" ref="trix" :input="computedId" :placeholder="placeholder" @trix-change="handleContentChange" @trix-initialize="handleInitialize" @trix-focus="processTrixFocus" @trix-blur="processTrixBlur" />
|
||||
<trix-editor :toolbar="toolbarId" :contenteditable="!disabledEditor" :class="['trix-content']" ref="trix" :input="computedId" :placeholder="placeholder" @trix-change="handleContentChange" @trix-initialize="handleInitialize" @trix-focus="processTrixFocus" @trix-blur="processTrixBlur" @trix-attachment-add="handleAttachmentAdd" />
|
||||
<input type="hidden" :name="inputName" :id="computedId" :value="editorContent" />
|
||||
</div>
|
||||
</template>
|
||||
@@ -316,6 +316,12 @@ export default {
|
||||
if (this.$refs.trix && this.$refs.trix.blur) {
|
||||
this.$refs.trix.blur()
|
||||
}
|
||||
},
|
||||
handleAttachmentAdd(event) {
|
||||
// Prevent pasting in images from the browser
|
||||
if (!event.attachment.file) {
|
||||
event.attachment.remove()
|
||||
}
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
|
||||
211
client/components/widgets/EncoderOptionsCard.vue
Normal file
211
client/components/widgets/EncoderOptionsCard.vue
Normal file
@@ -0,0 +1,211 @@
|
||||
<template>
|
||||
<div class="w-full py-2">
|
||||
<div class="flex -mb-px">
|
||||
<button type="button" :disabled="disabled" class="w-1/2 h-8 rounded-tl-md relative border border-black-200 flex items-center justify-center disabled:cursor-not-allowed" :class="!showAdvancedView ? 'text-white bg-bg hover:bg-bg/60 border-b-bg' : 'text-gray-400 hover:text-gray-300 bg-primary/70 hover:bg-primary/60'" @click="showAdvancedView = false">
|
||||
<p class="text-sm">{{ $strings.HeaderPresets }}</p>
|
||||
</button>
|
||||
<button type="button" :disabled="disabled" class="w-1/2 h-8 rounded-tr-md relative border border-black-200 flex items-center justify-center -ml-px disabled:cursor-not-allowed" :class="showAdvancedView ? 'text-white bg-bg hover:bg-bg/60 border-b-bg' : 'text-gray-400 hover:text-gray-300 bg-primary/70 hover:bg-primary/60'" @click="showAdvancedView = true">
|
||||
<p class="text-sm">{{ $strings.HeaderAdvanced }}</p>
|
||||
</button>
|
||||
</div>
|
||||
<div class="p-4 md:p-8 border border-black-200 rounded-b-md mr-px bg-bg">
|
||||
<template v-if="!showAdvancedView">
|
||||
<div class="flex flex-wrap gap-4 sm:gap-8 justify-start sm:justify-center">
|
||||
<div class="flex flex-col items-start gap-2">
|
||||
<p class="text-sm w-40">{{ $strings.LabelCodec }}</p>
|
||||
<ui-toggle-btns v-model="selectedCodec" :items="codecItems" :disabled="disabled" />
|
||||
<p class="text-xs text-gray-300">
|
||||
{{ $strings.LabelCurrently }} <span class="text-white">{{ currentCodec }}</span> <span v-if="isCodecsDifferent" class="text-warning">(mixed)</span>
|
||||
</p>
|
||||
</div>
|
||||
<div class="flex flex-col items-start gap-2">
|
||||
<p class="text-sm w-40">{{ $strings.LabelBitrate }}</p>
|
||||
<ui-toggle-btns v-model="selectedBitrate" :items="bitrateItems" :disabled="disabled" />
|
||||
<p class="text-xs text-gray-300">
|
||||
{{ $strings.LabelCurrently }} <span class="text-white">{{ currentBitrate }} KB/s</span>
|
||||
</p>
|
||||
</div>
|
||||
<div class="flex flex-col items-start gap-2">
|
||||
<p class="text-sm w-40">{{ $strings.LabelChannels }}</p>
|
||||
<ui-toggle-btns v-model="selectedChannels" :items="channelsItems" :disabled="disabled" />
|
||||
<p class="text-xs text-gray-300">
|
||||
{{ $strings.LabelCurrently }} <span class="text-white">{{ currentChannels }} ({{ currentChanelLayout }})</span>
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
<template v-else>
|
||||
<div>
|
||||
<div class="flex flex-wrap gap-4 sm:gap-8 justify-start sm:justify-center mb-4">
|
||||
<div class="w-40">
|
||||
<ui-text-input-with-label v-model="customCodec" :label="$strings.LabelAudioCodec" :disabled="disabled" @input="customCodecChanged" />
|
||||
</div>
|
||||
<div class="w-40">
|
||||
<ui-text-input-with-label v-model="customBitrate" :label="$strings.LabelAudioBitrate" :disabled="disabled" @input="customBitrateChanged" />
|
||||
</div>
|
||||
<div class="w-40">
|
||||
<ui-text-input-with-label v-model="customChannels" :label="$strings.LabelAudioChannels" type="number" :disabled="disabled" @input="customChannelsChanged" />
|
||||
</div>
|
||||
</div>
|
||||
<p class="text-xs sm:text-sm text-warning sm:text-center">{{ $strings.LabelEncodingWarningAdvancedSettings }}</p>
|
||||
</div>
|
||||
</template>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
audioTracks: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
disabled: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
showAdvancedView: false,
|
||||
selectedCodec: 'aac',
|
||||
selectedBitrate: '128k',
|
||||
selectedChannels: 2,
|
||||
customCodec: 'aac',
|
||||
customBitrate: '128k',
|
||||
customChannels: 2,
|
||||
currentCodec: '',
|
||||
currentBitrate: '',
|
||||
currentChannels: '',
|
||||
currentChanelLayout: '',
|
||||
isCodecsDifferent: false
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
codecItems() {
|
||||
return [
|
||||
{
|
||||
text: 'Copy',
|
||||
value: 'copy'
|
||||
},
|
||||
{
|
||||
text: 'AAC',
|
||||
value: 'aac'
|
||||
},
|
||||
{
|
||||
text: 'OPUS',
|
||||
value: 'opus'
|
||||
}
|
||||
]
|
||||
},
|
||||
bitrateItems() {
|
||||
return [
|
||||
{
|
||||
text: '32k',
|
||||
value: '32k'
|
||||
},
|
||||
{
|
||||
text: '64k',
|
||||
value: '64k'
|
||||
},
|
||||
{
|
||||
text: '128k',
|
||||
value: '128k'
|
||||
},
|
||||
{
|
||||
text: '192k',
|
||||
value: '192k'
|
||||
}
|
||||
]
|
||||
},
|
||||
channelsItems() {
|
||||
return [
|
||||
{
|
||||
text: '1 (mono)',
|
||||
value: 1
|
||||
},
|
||||
{
|
||||
text: '2 (stereo)',
|
||||
value: 2
|
||||
}
|
||||
]
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
customBitrateChanged(val) {
|
||||
localStorage.setItem('embedMetadataBitrate', val)
|
||||
},
|
||||
customChannelsChanged(val) {
|
||||
localStorage.setItem('embedMetadataChannels', val)
|
||||
},
|
||||
customCodecChanged(val) {
|
||||
localStorage.setItem('embedMetadataCodec', val)
|
||||
},
|
||||
getEncodingOptions() {
|
||||
return {
|
||||
codec: this.selectedCodec || 'aac',
|
||||
bitrate: this.selectedBitrate || '128k',
|
||||
channels: this.selectedChannels || 2
|
||||
}
|
||||
},
|
||||
setPreset() {
|
||||
// If already AAC and not mixed, set copy
|
||||
if (this.currentCodec === 'aac' && !this.isCodecsDifferent) {
|
||||
this.selectedCodec = 'copy'
|
||||
} else {
|
||||
this.selectedCodec = 'aac'
|
||||
}
|
||||
|
||||
if (!this.currentBitrate) {
|
||||
this.selectedBitrate = '128k'
|
||||
} else {
|
||||
// Find closest bitrate rounding up
|
||||
const bitratesToMatch = [32, 64, 128, 192]
|
||||
const closestBitrate = bitratesToMatch.find((bitrate) => bitrate >= this.currentBitrate) || 192
|
||||
this.selectedBitrate = closestBitrate + 'k'
|
||||
}
|
||||
|
||||
if (!this.currentChannels || isNaN(this.currentChannels)) {
|
||||
this.selectedChannels = 2
|
||||
} else {
|
||||
// Either 1 or 2
|
||||
this.selectedChannels = Math.max(Math.min(Number(this.currentChannels), 2), 1)
|
||||
}
|
||||
},
|
||||
setCurrentValues() {
|
||||
if (this.audioTracks.length === 0) return
|
||||
|
||||
this.currentChannels = this.audioTracks[0].channels
|
||||
this.currentChanelLayout = this.audioTracks[0].channelLayout
|
||||
this.currentCodec = this.audioTracks[0].codec
|
||||
|
||||
let totalBitrate = 0
|
||||
for (const track of this.audioTracks) {
|
||||
const trackBitrate = !isNaN(track.bitRate) ? track.bitRate : 0
|
||||
totalBitrate += trackBitrate
|
||||
|
||||
if (track.channels > this.currentChannels) this.currentChannels = track.channels
|
||||
if (track.codec !== this.currentCodec) {
|
||||
console.warn('Audio track codec is different from the first track', track.codec)
|
||||
this.isCodecsDifferent = true
|
||||
}
|
||||
}
|
||||
|
||||
this.currentBitrate = Math.round(totalBitrate / this.audioTracks.length / 1000)
|
||||
},
|
||||
init() {
|
||||
this.customBitrate = localStorage.getItem('embedMetadataBitrate') || '128k'
|
||||
this.customChannels = localStorage.getItem('embedMetadataChannels') || 2
|
||||
this.customCodec = localStorage.getItem('embedMetadataCodec') || 'aac'
|
||||
|
||||
this.setCurrentValues()
|
||||
|
||||
this.setPreset()
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.init()
|
||||
}
|
||||
}
|
||||
</script>
|
||||
@@ -2,7 +2,7 @@
|
||||
<div>
|
||||
<ui-multi-select-query-input v-model="seriesItems" text-key="displayName" :label="$strings.LabelSeries" :disabled="disabled" readonly show-edit @edit="editSeriesItem" @add="addNewSeries" />
|
||||
|
||||
<modals-edit-series-input-inner-modal v-model="showSeriesForm" :selected-series="selectedSeries" :existing-series-names="existingSeriesNames" @submit="submitSeriesForm" />
|
||||
<modals-edit-series-input-inner-modal v-model="showSeriesForm" :selected-series="selectedSeries" :existing-series-names="existingSeriesNames" :original-series-sequence="originalSeriesSequence" @submit="submitSeriesForm" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -18,6 +18,7 @@ export default {
|
||||
data() {
|
||||
return {
|
||||
selectedSeries: null,
|
||||
originalSeriesSequence: null,
|
||||
showSeriesForm: false
|
||||
}
|
||||
},
|
||||
@@ -59,6 +60,7 @@ export default {
|
||||
..._series
|
||||
}
|
||||
|
||||
this.originalSeriesSequence = _series.sequence
|
||||
this.showSeriesForm = true
|
||||
},
|
||||
addNewSeries() {
|
||||
@@ -68,6 +70,7 @@ export default {
|
||||
sequence: ''
|
||||
}
|
||||
|
||||
this.originalSeriesSequence = null
|
||||
this.showSeriesForm = true
|
||||
},
|
||||
submitSeriesForm() {
|
||||
|
||||
@@ -3,7 +3,6 @@ import LazyBookCard from '@/components/cards/LazyBookCard'
|
||||
import LazySeriesCard from '@/components/cards/LazySeriesCard'
|
||||
import LazyCollectionCard from '@/components/cards/LazyCollectionCard'
|
||||
import LazyPlaylistCard from '@/components/cards/LazyPlaylistCard'
|
||||
import LazyAlbumCard from '@/components/cards/LazyAlbumCard'
|
||||
import AuthorCard from '@/components/cards/AuthorCard'
|
||||
|
||||
export default {
|
||||
@@ -20,7 +19,6 @@ export default {
|
||||
if (this.entityName === 'series') return Vue.extend(LazySeriesCard)
|
||||
if (this.entityName === 'collections') return Vue.extend(LazyCollectionCard)
|
||||
if (this.entityName === 'playlists') return Vue.extend(LazyPlaylistCard)
|
||||
if (this.entityName === 'albums') return Vue.extend(LazyAlbumCard)
|
||||
if (this.entityName === 'authors') return Vue.extend(AuthorCard)
|
||||
return Vue.extend(LazyBookCard)
|
||||
},
|
||||
@@ -28,7 +26,6 @@ export default {
|
||||
if (this.entityName === 'series') return 'cards-lazy-series-card'
|
||||
if (this.entityName === 'collections') return 'cards-lazy-collection-card'
|
||||
if (this.entityName === 'playlists') return 'cards-lazy-playlist-card'
|
||||
if (this.entityName === 'albums') return 'cards-lazy-album-card'
|
||||
if (this.entityName === 'authors') return 'cards-author-card'
|
||||
return 'cards-lazy-book-card'
|
||||
},
|
||||
|
||||
4
client/package-lock.json
generated
4
client/package-lock.json
generated
@@ -1,12 +1,12 @@
|
||||
{
|
||||
"name": "audiobookshelf-client",
|
||||
"version": "2.21.0",
|
||||
"version": "2.23.0",
|
||||
"lockfileVersion": 3,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "audiobookshelf-client",
|
||||
"version": "2.21.0",
|
||||
"version": "2.23.0",
|
||||
"license": "ISC",
|
||||
"dependencies": {
|
||||
"@nuxtjs/axios": "^5.13.6",
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "audiobookshelf-client",
|
||||
"version": "2.21.0",
|
||||
"version": "2.23.0",
|
||||
"buildNumber": 1,
|
||||
"description": "Self-hosted audiobook and podcast client",
|
||||
"main": "index.js",
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<div id="page-wrapper" class="bg-bg page overflow-y-auto relative" :class="streamLibraryItem ? 'streaming' : ''">
|
||||
<div class="flex items-center py-4 px-2 md:px-0 max-w-7xl mx-auto">
|
||||
<div class="flex items-center py-4 px-4 max-w-7xl mx-auto">
|
||||
<nuxt-link :to="`/item/${libraryItem.id}`" class="hover:underline">
|
||||
<h1 class="text-lg lg:text-xl">{{ title }}</h1>
|
||||
</nuxt-link>
|
||||
@@ -12,7 +12,7 @@
|
||||
<p class="text-base font-mono ml-4 hidden md:block">{{ $secondsToTimestamp(mediaDurationRounded) }}</p>
|
||||
</div>
|
||||
|
||||
<div class="flex flex-wrap-reverse justify-center py-4 px-2">
|
||||
<div class="flex flex-wrap-reverse lg:flex-nowrap justify-center py-4 px-4">
|
||||
<div class="w-full max-w-3xl py-4">
|
||||
<div class="flex items-center">
|
||||
<div class="w-12 hidden lg:block" />
|
||||
@@ -23,8 +23,8 @@
|
||||
</div>
|
||||
<div class="flex items-center mb-3 py-1 -mx-1">
|
||||
<div class="w-12 hidden lg:block" />
|
||||
<ui-btn v-if="chapters.length" color="bg-primary" small class="mx-1" @click.stop="removeAllChaptersClick">{{ $strings.ButtonRemoveAll }}</ui-btn>
|
||||
<ui-btn v-if="newChapters.length > 1" :color="showShiftTimes ? 'bg' : 'primary'" class="mx-1" small @click="showShiftTimes = !showShiftTimes">{{ $strings.ButtonShiftTimes }}</ui-btn>
|
||||
<ui-btn v-if="chapters.length" color="bg-primary" small class="mx-1 whitespace-nowrap" @click.stop="removeAllChaptersClick">{{ $strings.ButtonRemoveAll }}</ui-btn>
|
||||
<ui-btn v-if="newChapters.length > 1" :color="showShiftTimes ? 'bg-bg' : 'bg-primary'" class="mx-1 whitespace-nowrap" small @click="showShiftTimes = !showShiftTimes">{{ $strings.ButtonShiftTimes }}</ui-btn>
|
||||
<ui-btn color="bg-primary" small :class="{ 'mx-1': newChapters.length > 1 }" @click="showFindChaptersModal = true">{{ $strings.ButtonLookup }}</ui-btn>
|
||||
<div class="grow" />
|
||||
<ui-btn v-if="hasChanges" small class="mx-1" @click.stop="resetChapters">{{ $strings.ButtonReset }}</ui-btn>
|
||||
@@ -65,7 +65,7 @@
|
||||
<ui-time-picker v-else class="text-xs" v-model="chapter.start" :show-three-digit-hour="mediaDuration >= 360000" @change="checkChapters" />
|
||||
</div>
|
||||
<div class="grow px-1">
|
||||
<ui-text-input v-model="chapter.title" @change="checkChapters" class="text-xs" />
|
||||
<ui-text-input v-model="chapter.title" @change="checkChapters" class="text-xs min-w-52" />
|
||||
</div>
|
||||
<div class="w-32 min-w-32 px-2 py-1">
|
||||
<div class="flex items-center">
|
||||
@@ -144,17 +144,18 @@
|
||||
<div v-if="!chapterData" class="flex flex-col items-center justify-center p-20">
|
||||
<div class="relative">
|
||||
<div class="flex items-end space-x-2">
|
||||
<ui-text-input-with-label v-model.trim="asinInput" label="ASIN" />
|
||||
<ui-dropdown v-model="regionInput" :label="$strings.LabelRegion" small :items="audibleRegions" class="w-32" />
|
||||
<ui-btn small color="bg-primary" @click="findChapters">{{ $strings.ButtonSearch }}</ui-btn>
|
||||
<ui-text-input-with-label v-model.trim="asinInput" label="ASIN" class="flex-grow" />
|
||||
<ui-dropdown v-model="regionInput" :label="$strings.LabelRegion" small :items="audibleRegions" class="w-20 max-w-20" />
|
||||
<ui-btn color="bg-primary" @click="findChapters">{{ $strings.ButtonSearch }}</ui-btn>
|
||||
</div>
|
||||
<div class="mt-4">
|
||||
<ui-checkbox v-model="removeBranding" :label="$strings.LabelRemoveAudibleBranding" small checkbox-bg="bg" label-class="pl-2 text-base text-sm" @click="toggleRemoveBranding" />
|
||||
</div>
|
||||
|
||||
<div class="absolute left-0 mt-1.5 text-error text-s h-5">
|
||||
<p v-if="asinError">{{ asinError }}</p>
|
||||
<p v-if="asinError">{{ $strings.MessageAsinCheck }}</p>
|
||||
</div>
|
||||
|
||||
<div class="invisible h-5 mt-1 text-xs"></div>
|
||||
<div class="invisible mt-1 text-xs"></div>
|
||||
</div>
|
||||
</div>
|
||||
<div v-else class="w-full p-4">
|
||||
@@ -261,6 +262,7 @@ export default {
|
||||
showFindChaptersModal: false,
|
||||
chapterData: null,
|
||||
asinError: null,
|
||||
removeBranding: false,
|
||||
showSecondInputs: false,
|
||||
audibleRegions: ['US', 'CA', 'UK', 'AU', 'FR', 'DE', 'JP', 'IT', 'IN', 'ES'],
|
||||
hasChanges: false
|
||||
@@ -322,6 +324,9 @@ export default {
|
||||
|
||||
this.checkChapters()
|
||||
},
|
||||
toggleRemoveBranding() {
|
||||
this.removeBranding = !this.removeBranding
|
||||
},
|
||||
shiftChapterTimes() {
|
||||
if (!this.shiftAmount || isNaN(this.shiftAmount) || this.newChapters.length <= 1) {
|
||||
return
|
||||
@@ -331,12 +336,12 @@ export default {
|
||||
|
||||
const lastChapter = this.newChapters[this.newChapters.length - 1]
|
||||
if (lastChapter.start + amount > this.mediaDurationRounded) {
|
||||
this.$toast.error('Invalid shift amount. Last chapter start time would extend beyond the duration of this audiobook.')
|
||||
this.$toast.error(this.$strings.ToastChaptersInvalidShiftAmountLast)
|
||||
return
|
||||
}
|
||||
|
||||
if (this.newChapters[0].end + amount <= 0) {
|
||||
this.$toast.error('Invalid shift amount. First chapter would have zero or negative length.')
|
||||
if (this.newChapters[1].start + amount <= 0) {
|
||||
this.$toast.error(this.$strings.ToastChaptersInvalidShiftAmountStart)
|
||||
return
|
||||
}
|
||||
|
||||
@@ -568,7 +573,7 @@ export default {
|
||||
this.asinError = this.$getString(data.stringKey)
|
||||
} else {
|
||||
console.log('Chapter data', data)
|
||||
this.chapterData = data
|
||||
this.chapterData = this.removeBranding ? this.removeBrandingFromData(data) : data
|
||||
}
|
||||
})
|
||||
.catch((error) => {
|
||||
@@ -578,6 +583,37 @@ export default {
|
||||
this.showFindChaptersModal = false
|
||||
})
|
||||
},
|
||||
removeBrandingFromData(data) {
|
||||
if (!data) return data
|
||||
try {
|
||||
const introDuration = data.brandIntroDurationMs
|
||||
const outroDuration = data.brandOutroDurationMs
|
||||
|
||||
for (let i = 0; i < data.chapters.length; i++) {
|
||||
const chapter = data.chapters[i]
|
||||
if (chapter.startOffsetMs < introDuration) {
|
||||
// This should never happen, as the intro is not longer than the first chapter
|
||||
// If this happens set to the next second
|
||||
// Will be 0 for the first chapter anayways
|
||||
chapter.startOffsetMs = i * 1000
|
||||
chapter.startOffsetSec = i
|
||||
} else {
|
||||
chapter.startOffsetMs -= introDuration
|
||||
chapter.startOffsetSec = Math.floor(chapter.startOffsetMs / 1000)
|
||||
}
|
||||
}
|
||||
|
||||
const lastChapter = data.chapters[data.chapters.length - 1]
|
||||
// If there is an outro that's in the outro duration, remove it
|
||||
if (lastChapter && lastChapter.lengthMs <= outroDuration) {
|
||||
data.chapters.pop()
|
||||
}
|
||||
|
||||
return data
|
||||
} catch {
|
||||
return data
|
||||
}
|
||||
},
|
||||
resetChapters() {
|
||||
const payload = {
|
||||
message: this.$strings.MessageResetChaptersConfirm,
|
||||
|
||||
@@ -2,7 +2,14 @@
|
||||
<div id="page-wrapper" class="bg-bg page p-8 overflow-auto relative" :class="streamLibraryItem ? 'streaming' : ''">
|
||||
<div class="flex items-center justify-center mb-6">
|
||||
<div class="w-full max-w-2xl">
|
||||
<p class="text-2xl mb-2">{{ $strings.HeaderAudiobookTools }}</p>
|
||||
<div class="flex items-center mb-4">
|
||||
<nuxt-link :to="`/item/${libraryItem.id}`" class="hover:underline">
|
||||
<h1 class="text-lg lg:text-xl">{{ mediaMetadata.title }}</h1>
|
||||
</nuxt-link>
|
||||
<button class="w-7 h-7 flex items-center justify-center mx-4 hover:scale-110 duration-100 transform text-gray-200 hover:text-white" @click="editItem">
|
||||
<span class="material-symbols text-base">edit</span>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="w-full max-w-2xl">
|
||||
<div class="flex justify-end">
|
||||
@@ -13,13 +20,13 @@
|
||||
|
||||
<div class="flex justify-center mb-2">
|
||||
<div class="w-full max-w-2xl">
|
||||
<p class="text-xl">{{ $strings.HeaderMetadataToEmbed }}</p>
|
||||
<p class="text-lg">{{ $strings.HeaderMetadataToEmbed }}</p>
|
||||
</div>
|
||||
<div class="w-full max-w-2xl"></div>
|
||||
</div>
|
||||
|
||||
<div class="flex justify-center flex-wrap">
|
||||
<div class="w-full max-w-2xl border border-white/10 bg-bg mx-2">
|
||||
<div class="flex justify-center flex-wrap lg:flex-nowrap gap-4">
|
||||
<div class="w-full max-w-2xl border border-white/10 bg-bg">
|
||||
<div class="flex py-2 px-4">
|
||||
<div class="w-1/3 text-xs font-semibold uppercase text-gray-200">{{ $strings.LabelMetaTag }}</div>
|
||||
<div class="w-2/3 text-xs font-semibold uppercase text-gray-200">{{ $strings.LabelValue }}</div>
|
||||
@@ -35,7 +42,7 @@
|
||||
</template>
|
||||
</div>
|
||||
</div>
|
||||
<div class="w-full max-w-2xl border border-white/10 bg-bg mx-2">
|
||||
<div class="w-full max-w-2xl border border-white/10 bg-bg">
|
||||
<div class="flex py-2 px-4 bg-primary/25">
|
||||
<div class="grow text-xs font-semibold uppercase text-gray-200">{{ $strings.LabelChapterTitle }}</div>
|
||||
<div class="w-24 text-xs font-semibold uppercase text-gray-200">{{ $strings.LabelStart }}</div>
|
||||
@@ -77,10 +84,6 @@
|
||||
</div>
|
||||
<!-- m4b embed action buttons -->
|
||||
<div v-else class="w-full flex items-center mb-4">
|
||||
<button :disabled="processing" class="text-sm uppercase text-gray-200 flex items-center pt-px pl-1 pr-2 hover:bg-white/5 rounded-md" @click="showEncodeOptions = !showEncodeOptions">
|
||||
<span class="material-symbols text-xl">{{ showEncodeOptions || usingCustomEncodeOptions ? 'check_box' : 'check_box_outline_blank' }}</span> <span class="pl-1">{{ $strings.LabelUseAdvancedOptions }}</span>
|
||||
</button>
|
||||
|
||||
<div class="grow" />
|
||||
|
||||
<ui-btn v-if="!isTaskFinished && processing" color="bg-error" :loading="isCancelingEncode" class="mr-2" @click.stop="cancelEncodeClick">{{ $strings.ButtonCancelEncode }}</ui-btn>
|
||||
@@ -89,18 +92,16 @@
|
||||
<p v-else class="text-success text-lg font-semibold">{{ $strings.MessageM4BFinished }}</p>
|
||||
</div>
|
||||
|
||||
<!-- advanced encoding options -->
|
||||
<div v-if="isM4BTool" class="overflow-hidden">
|
||||
<transition name="slide">
|
||||
<div v-if="showEncodeOptions || usingCustomEncodeOptions" class="mb-4 pb-4 border-b border-white/10">
|
||||
<div class="flex flex-wrap -mx-2">
|
||||
<ui-text-input-with-label ref="bitrateInput" v-model="encodingOptions.bitrate" :disabled="processing || isTaskFinished" :label="$strings.LabelAudioBitrate" class="m-2 max-w-40" @input="bitrateChanged" />
|
||||
<ui-text-input-with-label ref="channelsInput" v-model="encodingOptions.channels" :disabled="processing || isTaskFinished" :label="$strings.LabelAudioChannels" class="m-2 max-w-40" @input="channelsChanged" />
|
||||
<ui-text-input-with-label ref="codecInput" v-model="encodingOptions.codec" :disabled="processing || isTaskFinished" :label="$strings.LabelAudioCodec" class="m-2 max-w-40" @input="codecChanged" />
|
||||
</div>
|
||||
<p class="text-sm text-warning">{{ $strings.LabelEncodingWarningAdvancedSettings }}</p>
|
||||
</div>
|
||||
</transition>
|
||||
<!-- show encoding options for running task -->
|
||||
<div v-if="encodeTaskHasEncodingOptions" class="mb-4 pb-4 border-b border-white/10">
|
||||
<div class="flex flex-wrap -mx-2">
|
||||
<ui-text-input-with-label ref="bitrateInput" v-model="encodingOptions.bitrate" readonly :label="$strings.LabelAudioBitrate" class="m-2 max-w-40" @input="bitrateChanged" />
|
||||
<ui-text-input-with-label ref="channelsInput" v-model="encodingOptions.channels" readonly :label="$strings.LabelAudioChannels" class="m-2 max-w-40" @input="channelsChanged" />
|
||||
<ui-text-input-with-label ref="codecInput" v-model="encodingOptions.codec" readonly :label="$strings.LabelAudioCodec" class="m-2 max-w-40" @input="codecChanged" />
|
||||
</div>
|
||||
</div>
|
||||
<div v-else-if="isM4BTool" class="mb-4">
|
||||
<widgets-encoder-options-card ref="encoderOptionsCard" :audio-tracks="audioFiles" :disabled="processing || isTaskFinished" />
|
||||
</div>
|
||||
|
||||
<div class="mb-4">
|
||||
@@ -146,19 +147,29 @@
|
||||
<div class="flex py-2 px-4 bg-primary/25">
|
||||
<div class="w-10 text-xs font-semibold text-gray-200">#</div>
|
||||
<div class="grow text-xs font-semibold uppercase text-gray-200">{{ $strings.LabelFilename }}</div>
|
||||
<div class="w-20 text-xs font-semibold uppercase text-gray-200 hidden lg:block">{{ $strings.LabelChannels }}</div>
|
||||
<div class="w-16 text-xs font-semibold uppercase text-gray-200 hidden md:block">{{ $strings.LabelCodec }}</div>
|
||||
<div class="w-16 text-xs font-semibold uppercase text-gray-200 hidden md:block">{{ $strings.LabelBitrate }}</div>
|
||||
<div class="w-16 text-xs font-semibold uppercase text-gray-200">{{ $strings.LabelSize }}</div>
|
||||
<div class="w-24"></div>
|
||||
</div>
|
||||
<template v-for="file in audioFiles">
|
||||
<div :key="file.index" class="flex py-2 px-4 text-sm" :class="file.index % 2 === 0 ? 'bg-primary/25' : ''">
|
||||
<div class="w-10">{{ file.index }}</div>
|
||||
<div :key="file.index" class="flex py-2 px-4 text-xs sm:text-sm" :class="file.index % 2 === 0 ? 'bg-primary/25' : ''">
|
||||
<div class="w-10 min-w-10">{{ file.index }}</div>
|
||||
<div class="grow">
|
||||
{{ file.metadata.filename }}
|
||||
</div>
|
||||
<div class="w-16 font-mono text-gray-200">
|
||||
<div class="w-20 min-w-20 text-gray-200 hidden lg:block">{{ file.channels || 'unknown' }} ({{ file.channelLayout || 'unknown' }})</div>
|
||||
<div class="w-16 min-w-16 text-gray-200 hidden md:block">
|
||||
{{ file.codec || 'unknown' }}
|
||||
</div>
|
||||
<div class="w-16 min-w-16 text-gray-200 hidden md:block">
|
||||
{{ $bytesPretty(file.bitRate || 0, 0) }}
|
||||
</div>
|
||||
<div class="w-16 min-w-16 text-gray-200">
|
||||
{{ $bytesPretty(file.metadata.size) }}
|
||||
</div>
|
||||
<div class="w-24">
|
||||
<div class="w-24 min-w-24">
|
||||
<div class="flex justify-center">
|
||||
<span v-if="audioFilesFinished[file.ino]" class="material-symbols text-xl text-success leading-none">check_circle</span>
|
||||
<div v-else-if="audioFilesEncoding[file.ino]">
|
||||
@@ -214,7 +225,6 @@ export default {
|
||||
metadataObject: null,
|
||||
selectedTool: 'embed',
|
||||
isCancelingEncode: false,
|
||||
showEncodeOptions: false,
|
||||
shouldBackupAudioFiles: true,
|
||||
encodingOptions: {
|
||||
bitrate: '128k',
|
||||
@@ -263,9 +273,6 @@ export default {
|
||||
audioFiles() {
|
||||
return (this.media.audioFiles || []).filter((af) => !af.exclude)
|
||||
},
|
||||
isSingleM4b() {
|
||||
return this.audioFiles.length === 1 && this.audioFiles[0].metadata.ext.toLowerCase() === '.m4b'
|
||||
},
|
||||
streamLibraryItem() {
|
||||
return this.$store.state.streamLibraryItem
|
||||
},
|
||||
@@ -273,14 +280,10 @@ export default {
|
||||
return this.media.chapters || []
|
||||
},
|
||||
availableTools() {
|
||||
if (this.isSingleM4b) {
|
||||
return [{ value: 'embed', text: this.$strings.LabelToolsEmbedMetadata }]
|
||||
} else {
|
||||
return [
|
||||
{ value: 'embed', text: this.$strings.LabelToolsEmbedMetadata },
|
||||
{ value: 'm4b', text: this.$strings.LabelToolsM4bEncoder }
|
||||
]
|
||||
}
|
||||
return [
|
||||
{ value: 'embed', text: this.$strings.LabelToolsEmbedMetadata },
|
||||
{ value: 'm4b', text: this.$strings.LabelToolsM4bEncoder }
|
||||
]
|
||||
},
|
||||
taskFailed() {
|
||||
return this.isTaskFinished && this.task.isFailed
|
||||
@@ -314,8 +317,8 @@ export default {
|
||||
isMetadataEmbedQueued() {
|
||||
return this.queuedEmbedLIds.some((lid) => lid === this.libraryItemId)
|
||||
},
|
||||
usingCustomEncodeOptions() {
|
||||
return this.isM4BTool && this.encodeTask && this.encodeTask.data.encodeOptions && Object.keys(this.encodeTask.data.encodeOptions).length > 0
|
||||
encodeTaskHasEncodingOptions() {
|
||||
return this.isM4BTool && !!this.encodeTask?.data.encodeOptions && Object.keys(this.encodeTask.data.encodeOptions).length > 0
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
@@ -351,19 +354,13 @@ export default {
|
||||
if (this.$refs.channelsInput) this.$refs.channelsInput.blur()
|
||||
if (this.$refs.codecInput) this.$refs.codecInput.blur()
|
||||
|
||||
let queryStr = ''
|
||||
if (this.showEncodeOptions) {
|
||||
const options = []
|
||||
if (this.encodingOptions.bitrate) options.push(`bitrate=${this.encodingOptions.bitrate}`)
|
||||
if (this.encodingOptions.channels) options.push(`channels=${this.encodingOptions.channels}`)
|
||||
if (this.encodingOptions.codec) options.push(`codec=${this.encodingOptions.codec}`)
|
||||
if (options.length) {
|
||||
queryStr = `?${options.join('&')}`
|
||||
}
|
||||
}
|
||||
const encodeOptions = this.$refs.encoderOptionsCard.getEncodingOptions()
|
||||
|
||||
const queryParams = new URLSearchParams(encodeOptions)
|
||||
|
||||
this.processing = true
|
||||
this.$axios
|
||||
.$post(`/api/tools/item/${this.libraryItemId}/encode-m4b${queryStr}`)
|
||||
.$post(`/api/tools/item/${this.libraryItemId}/encode-m4b?${queryParams.toString()}`)
|
||||
.then(() => {
|
||||
console.log('Ab m4b merge started')
|
||||
})
|
||||
@@ -416,14 +413,10 @@ export default {
|
||||
const shouldBackupAudioFiles = localStorage.getItem('embedMetadataShouldBackup')
|
||||
this.shouldBackupAudioFiles = shouldBackupAudioFiles != 0
|
||||
|
||||
if (this.usingCustomEncodeOptions) {
|
||||
if (this.encodeTaskHasEncodingOptions) {
|
||||
if (this.encodeTask.data.encodeOptions.bitrate) this.encodingOptions.bitrate = this.encodeTask.data.encodeOptions.bitrate
|
||||
if (this.encodeTask.data.encodeOptions.channels) this.encodingOptions.channels = this.encodeTask.data.encodeOptions.channels
|
||||
if (this.encodeTask.data.encodeOptions.codec) this.encodingOptions.codec = this.encodeTask.data.encodeOptions.codec
|
||||
} else {
|
||||
this.encodingOptions.bitrate = localStorage.getItem('embedMetadataBitrate') || '128k'
|
||||
this.encodingOptions.channels = localStorage.getItem('embedMetadataChannels') || '2'
|
||||
this.encodingOptions.codec = localStorage.getItem('embedMetadataCodec') || 'aac'
|
||||
}
|
||||
},
|
||||
fetchMetadataEmbedObject() {
|
||||
@@ -438,10 +431,24 @@ export default {
|
||||
},
|
||||
taskUpdated(task) {
|
||||
this.processing = !task.isFinished
|
||||
},
|
||||
editItem() {
|
||||
this.$store.commit('showEditModal', this.libraryItem)
|
||||
},
|
||||
libraryItemUpdated(libraryItem) {
|
||||
if (libraryItem.id === this.libraryItem.id) {
|
||||
this.libraryItem = libraryItem
|
||||
this.fetchMetadataEmbedObject()
|
||||
}
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.init()
|
||||
|
||||
this.$eventBus.$on(`${this.libraryItem.id}_updated`, this.libraryItemUpdated)
|
||||
},
|
||||
beforeDestroy() {
|
||||
this.$eventBus.$off(`${this.libraryItem.id}_updated`, this.libraryItemUpdated)
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
@@ -122,7 +122,8 @@
|
||||
</div>
|
||||
</transition>
|
||||
</div>
|
||||
<div class="w-full flex items-center justify-end p-4">
|
||||
<div class="w-full flex items-center justify-between p-4">
|
||||
<p v-if="enableOpenIDAuth" class="text-sm text-warning">{{ $strings.MessageAuthenticationOIDCChangesRestart }}</p>
|
||||
<ui-btn color="bg-success" :padding-x="8" small class="text-base" :loading="savingSettings" @click="saveSettings">{{ $strings.ButtonSave }}</ui-btn>
|
||||
</div>
|
||||
</app-settings-content>
|
||||
|
||||
@@ -819,6 +819,17 @@ export default {
|
||||
-webkit-line-clamp: 4;
|
||||
max-height: calc(6 * 1lh);
|
||||
}
|
||||
|
||||
/* Safari-specific fix for the description clamping */
|
||||
@supports (-webkit-touch-callout: none) {
|
||||
#item-description {
|
||||
position: relative;
|
||||
display: block;
|
||||
overflow: hidden;
|
||||
max-height: calc(6 * 1lh);
|
||||
}
|
||||
}
|
||||
|
||||
#item-description.show-full {
|
||||
-webkit-line-clamp: unset;
|
||||
max-height: 999rem;
|
||||
|
||||
@@ -22,6 +22,7 @@ export default {
|
||||
})
|
||||
results = {
|
||||
podcasts: results?.podcast || [],
|
||||
episodes: results?.episodes || [],
|
||||
books: results?.book || [],
|
||||
authors: results?.authors || [],
|
||||
series: results?.series || [],
|
||||
@@ -61,6 +62,7 @@ export default {
|
||||
})
|
||||
this.results = {
|
||||
podcasts: results?.podcast || [],
|
||||
episodes: results?.episodes || [],
|
||||
books: results?.book || [],
|
||||
authors: results?.authors || [],
|
||||
series: results?.series || [],
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
<template>
|
||||
<div class="w-full h-dvh max-h-dvh overflow-hidden" :style="{ backgroundColor: coverRgb }">
|
||||
<div class="w-full max-w-full h-dvh max-h-dvh overflow-hidden" :style="{ backgroundColor: coverRgb }">
|
||||
<div class="w-screen h-screen absolute inset-0 pointer-events-none" style="background: linear-gradient(180deg, rgba(0, 0, 0, 0) 0%, rgba(38, 38, 38, 1) 80%)"></div>
|
||||
<div class="absolute inset-0 w-screen h-dvh flex items-center justify-center z-10">
|
||||
<div class="w-full p-2 sm:p-4 md:p-8">
|
||||
@@ -335,8 +335,11 @@ export default {
|
||||
}
|
||||
},
|
||||
resize() {
|
||||
this.windowWidth = window.innerWidth
|
||||
this.windowHeight = window.innerHeight
|
||||
setTimeout(() => {
|
||||
this.windowWidth = window.innerWidth
|
||||
this.windowHeight = window.innerHeight
|
||||
this.$store.commit('globals/updateWindowSize', { width: window.innerWidth, height: window.innerHeight })
|
||||
}, 100)
|
||||
},
|
||||
playerError(error) {
|
||||
console.error('Player error', error)
|
||||
|
||||
@@ -359,15 +359,14 @@ export default {
|
||||
// Check if path already exists before starting upload
|
||||
// uploading fails if path already exists
|
||||
for (const item of items) {
|
||||
const filepath = Path.join(this.selectedFolder.fullPath, item.directory)
|
||||
const exists = await this.$axios
|
||||
.$post(`/api/filesystem/pathexists`, { filepath, directory: item.directory, folderPath: this.selectedFolder.fullPath })
|
||||
.$post(`/api/filesystem/pathexists`, { directory: item.directory, folderPath: this.selectedFolder.fullPath })
|
||||
.then((data) => {
|
||||
if (data.exists) {
|
||||
if (data.libraryItemTitle) {
|
||||
this.$toast.error(this.$getString('ToastUploaderItemExistsInSubdirectoryError', [data.libraryItemTitle]))
|
||||
} else {
|
||||
this.$toast.error(this.$getString('ToastUploaderFilepathExistsError', [filepath]))
|
||||
this.$toast.error(this.$getString('ToastUploaderFilepathExistsError', [Path.join(this.selectedFolder.fullPath, item.directory)]))
|
||||
}
|
||||
}
|
||||
return data.exists
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
export default class AudioTrack {
|
||||
constructor(track, userToken, routerBasePath) {
|
||||
constructor(track, sessionId, routerBasePath) {
|
||||
this.index = track.index || 0
|
||||
this.startOffset = track.startOffset || 0 // Total time of all previous tracks
|
||||
this.duration = track.duration || 0
|
||||
@@ -8,28 +8,29 @@ export default class AudioTrack {
|
||||
this.mimeType = track.mimeType
|
||||
this.metadata = track.metadata || {}
|
||||
|
||||
this.userToken = userToken
|
||||
this.sessionId = sessionId
|
||||
this.routerBasePath = routerBasePath || ''
|
||||
if (this.contentUrl?.startsWith('/hls')) {
|
||||
this.sessionTrackUrl = this.contentUrl
|
||||
} else {
|
||||
this.sessionTrackUrl = `/public/session/${sessionId}/track/${this.index}`
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Used for CastPlayer
|
||||
*/
|
||||
get fullContentUrl() {
|
||||
if (!this.contentUrl || this.contentUrl.startsWith('http')) return this.contentUrl
|
||||
|
||||
if (process.env.NODE_ENV === 'development') {
|
||||
return `${process.env.serverUrl}${this.contentUrl}?token=${this.userToken}`
|
||||
return `${process.env.serverUrl}${this.sessionTrackUrl}`
|
||||
}
|
||||
return `${window.location.origin}${this.routerBasePath}${this.contentUrl}?token=${this.userToken}`
|
||||
return `${window.location.origin}${this.routerBasePath}${this.sessionTrackUrl}`
|
||||
}
|
||||
|
||||
/**
|
||||
* Used for LocalPlayer
|
||||
*/
|
||||
get relativeContentUrl() {
|
||||
if (!this.contentUrl || this.contentUrl.startsWith('http')) return this.contentUrl
|
||||
|
||||
return `${this.routerBasePath}${this.contentUrl}?token=${this.userToken}`
|
||||
return `${this.routerBasePath}${this.sessionTrackUrl}`
|
||||
}
|
||||
}
|
||||
|
||||
@@ -37,9 +37,6 @@ export default class PlayerHandler {
|
||||
get isPlayingLocalItem() {
|
||||
return this.libraryItem && this.player instanceof LocalAudioPlayer
|
||||
}
|
||||
get userToken() {
|
||||
return this.ctx.$store.getters['user/getToken']
|
||||
}
|
||||
get playerPlaying() {
|
||||
return this.playerState === 'PLAYING'
|
||||
}
|
||||
@@ -226,7 +223,7 @@ export default class PlayerHandler {
|
||||
|
||||
console.log('[PlayerHandler] Preparing Session', session)
|
||||
|
||||
var audioTracks = session.audioTracks.map((at) => new AudioTrack(at, this.userToken, this.ctx.$config.routerBasePath))
|
||||
var audioTracks = session.audioTracks.map((at) => new AudioTrack(at, session.id, this.ctx.$config.routerBasePath))
|
||||
|
||||
this.ctx.playerLoading = true
|
||||
this.isHlsTranscode = true
|
||||
|
||||
@@ -5,6 +5,7 @@ import { supplant } from './utils'
|
||||
const defaultCode = 'en-us'
|
||||
|
||||
const languageCodeMap = {
|
||||
ar: { label: 'عربي', dateFnsLocale: 'ar' },
|
||||
bg: { label: 'Български', dateFnsLocale: 'bg' },
|
||||
bn: { label: 'বাংলা', dateFnsLocale: 'bn' },
|
||||
ca: { label: 'Català', dateFnsLocale: 'ca' },
|
||||
|
||||
@@ -10,6 +10,8 @@
|
||||
"ButtonApplyChapters": "حفظ الفصول",
|
||||
"ButtonAuthors": "المؤلفون",
|
||||
"ButtonBack": "الرجوع",
|
||||
"ButtonBatchEditPopulateFromExisting": "تعبئة من الموجود",
|
||||
"ButtonBatchEditPopulateMapDetails": "تعبئة تفاصيل الخريطة",
|
||||
"ButtonBrowseForFolder": "البحث عن المجلد",
|
||||
"ButtonCancel": "إلغاء",
|
||||
"ButtonCancelEncode": "إلغاء الترميز",
|
||||
@@ -30,8 +32,8 @@
|
||||
"ButtonEditChapters": "تعديل الفصول",
|
||||
"ButtonEditPodcast": "تعديل البودكاست",
|
||||
"ButtonEnable": "تفعيل",
|
||||
"ButtonFireAndFail": "النار والفشل",
|
||||
"ButtonFireOnTest": "حادثة إطلاق النار",
|
||||
"ButtonFireAndFail": "محاولة فاشلة",
|
||||
"ButtonFireOnTest": "تشغيل حدث الاختبار",
|
||||
"ButtonForceReScan": "فرض إعادة المسح",
|
||||
"ButtonFullPath": "المسار الكامل",
|
||||
"ButtonHide": "إخفاء",
|
||||
@@ -57,7 +59,7 @@
|
||||
"ButtonPause": "إيقاف مؤقت",
|
||||
"ButtonPlay": "تشغيل",
|
||||
"ButtonPlayAll": "تشغيل الكل",
|
||||
"ButtonPlaying": "مشغل الآن",
|
||||
"ButtonPlaying": "جاري التشغيل",
|
||||
"ButtonPlaylists": "قوائم التشغيل",
|
||||
"ButtonPrevious": "سابِق",
|
||||
"ButtonPreviousChapter": "الفصل السابق",
|
||||
@@ -88,13 +90,15 @@
|
||||
"ButtonSaveTracklist": "حفظ قائمة التشغيل",
|
||||
"ButtonScan": "تَحَقُق",
|
||||
"ButtonScanLibrary": "تَحَقُق من المكتبة",
|
||||
"ButtonScrollLeft": "تمرير لليسار",
|
||||
"ButtonScrollRight": "تمرير لليمين",
|
||||
"ButtonSearch": "بحث",
|
||||
"ButtonSelectFolderPath": "حدد مسار المجلد",
|
||||
"ButtonSeries": "سلسلة",
|
||||
"ButtonSetChaptersFromTracks": "تعيين الفصول من الملفات",
|
||||
"ButtonShare": "نشر",
|
||||
"ButtonShiftTimes": "أوقات العمل",
|
||||
"ButtonShow": "عرض",
|
||||
"ButtonShow": "أعرض",
|
||||
"ButtonStartM4BEncode": "ابدأ ترميز M4B",
|
||||
"ButtonStartMetadataEmbed": "ابدأ تضمين البيانات الوصفية",
|
||||
"ButtonStats": "الإحصائيات",
|
||||
@@ -153,64 +157,951 @@
|
||||
"HeaderLogs": "السجلات",
|
||||
"HeaderManageGenres": "إدارة الانواع",
|
||||
"HeaderManageTags": "إدارة العلامات",
|
||||
"HeaderOpenRSSFeed": "فتح تغذية RSS",
|
||||
"HeaderMapDetails": "تفاصيل الخريطة",
|
||||
"HeaderMatch": "مطابقة",
|
||||
"HeaderMetadataOrderOfPrecedence": "ترتيب أولوية البيانات الوصفية",
|
||||
"HeaderMetadataToEmbed": "البيانات الوصفية المراد تضمينها",
|
||||
"HeaderNewAccount": "حساب جديد",
|
||||
"HeaderNewLibrary": "مكتبة جديدة",
|
||||
"HeaderNotificationCreate": "إنشاء إشعار",
|
||||
"HeaderNotificationUpdate": "تحديث إشعار",
|
||||
"HeaderNotifications": "إشعارات",
|
||||
"HeaderOpenIDConnectAuthentication": "مصادقة OpenID Connect",
|
||||
"HeaderOpenListeningSessions": "جلسات الاستماع المفتوحة",
|
||||
"HeaderOpenRSSFeed": "عرض تغذية RSS",
|
||||
"HeaderOtherFiles": "ملفات أخرى",
|
||||
"HeaderPasswordAuthentication": "مصادقة كلمة المرور",
|
||||
"HeaderPermissions": "الصلاحيات",
|
||||
"HeaderPlayerQueue": "قائمة انتظار المشغل",
|
||||
"HeaderPlayerSettings": "إعدادات المشغل",
|
||||
"HeaderPlaylist": "قائمة تشغيل",
|
||||
"HeaderPlaylistItems": "عناصر قائمة التشغيل",
|
||||
"HeaderPodcastsToAdd": "بودكاست لإضافتها",
|
||||
"HeaderPresets": "إعدادات مسبقة",
|
||||
"HeaderPreviewCover": "معاينة الغلاف",
|
||||
"HeaderRSSFeedGeneral": "تفاصيل RSS",
|
||||
"HeaderRSSFeedIsOpen": "مغذي RSS مفتوح",
|
||||
"HeaderRSSFeeds": "خلاصات RSS",
|
||||
"HeaderRemoveEpisode": "إزالة حلقة",
|
||||
"HeaderRemoveEpisodes": "إزالة {0} حلقات",
|
||||
"HeaderSavedMediaProgress": "تقدم الوسائط المحفوظة",
|
||||
"HeaderSchedule": "جَدْوَل",
|
||||
"HeaderScheduleEpisodeDownloads": "جدولة التنزيلات التلقائية للحلقات",
|
||||
"HeaderScheduleLibraryScans": "جدولة عمليات المسح التلقائي للمكتبة",
|
||||
"HeaderSession": "الجلسة",
|
||||
"HeaderSetBackupSchedule": "تعيين جدول النسخ الاحتياطي",
|
||||
"HeaderSettings": "إعدادات",
|
||||
"HeaderSettingsDisplay": "عرض",
|
||||
"HeaderSettingsExperimental": "ميزات تجريبية",
|
||||
"HeaderSettingsGeneral": "عام",
|
||||
"HeaderSettingsScanner": "إعدادات المسح",
|
||||
"HeaderSettingsWebClient": "عميل الويب",
|
||||
"HeaderSleepTimer": "مؤقت النوم",
|
||||
"HeaderStatsLargestItems": "أكبر العناصر حجماً",
|
||||
"HeaderStatsLongestItems": "أطول العناصر (بالساعات)",
|
||||
"HeaderStatsMinutesListeningChart": "الدقائق المسموعة (آخر 7 أيام)",
|
||||
"HeaderStatsRecentSessions": "الجلسات الأخيرة",
|
||||
"HeaderStatsTop10Authors": "أفضل 10 مؤلفين",
|
||||
"HeaderStatsTop5Genres": "أفضل 5 أنواع",
|
||||
"HeaderTableOfContents": "جدول المحتويات",
|
||||
"HeaderTools": "أدوات",
|
||||
"HeaderUpdateAccount": "تحديث الحساب",
|
||||
"HeaderUpdateAuthor": "تحديث المؤلف",
|
||||
"HeaderUpdateDetails": "تحديث التفاصيل",
|
||||
"HeaderUpdateLibrary": "تحديث المكتبة",
|
||||
"HeaderUsers": "المستخدمون",
|
||||
"HeaderYearReview": "ملخص عام {0}",
|
||||
"HeaderYourStats": "إحصائياتك",
|
||||
"LabelAbridged": "مختصر",
|
||||
"LabelAbridgedChecked": "مختصر (محدد)",
|
||||
"LabelAbridgedUnchecked": "غير مختصر (غير محدد)",
|
||||
"LabelAccessibleBy": "يمكن الوصول إليه بواسطة",
|
||||
"LabelAccountType": "نوع الحساب",
|
||||
"LabelAccountTypeAdmin": "مدير",
|
||||
"LabelAccountTypeGuest": "ضيف",
|
||||
"LabelAccountTypeUser": "مستخدم",
|
||||
"LabelActivities": "النشاطات",
|
||||
"LabelActivity": "نشاط",
|
||||
"LabelAddToCollection": "إضافة إلى المجموعة",
|
||||
"LabelAddToCollectionBatch": "إضافة {0} كتابًا إلى المجموعة",
|
||||
"LabelAddToPlaylist": "أضف إلى قائمة التشغيل",
|
||||
"LabelAddToPlaylistBatch": "إضافة {0} عناصر إلى قائمة التشغيل",
|
||||
"LabelAddedAt": "أضيفت على",
|
||||
"LabelAddedDate": "تمت الإضافة",
|
||||
"LabelAdminUsersOnly": "للمستخدمين المديرين فقط",
|
||||
"LabelAll": "الكل",
|
||||
"LabelAllEpisodesDownloaded": "تم تنزيل جميع الحلقات",
|
||||
"LabelAllUsers": "جميع المستخدمين",
|
||||
"LabelAllUsersExcludingGuests": "جميع المستخدمين باستثناء الضيوف",
|
||||
"LabelAllUsersIncludingGuests": "جميع المستخدمين بما في ذلك الضيوف",
|
||||
"LabelAlreadyInYourLibrary": "موجود بالفعل في مكتبتك",
|
||||
"LabelApiToken": "رمز API",
|
||||
"LabelAppend": "إلحاق",
|
||||
"LabelAudioBitrate": "معدل بت الصوت (على سبيل المثال 128 كيلو بايت)",
|
||||
"LabelAudioChannels": "قنوات الصوت (1 أو 2)",
|
||||
"LabelAudioCodec": "برنامج ترميز الصوت",
|
||||
"LabelAuthor": "المؤلف",
|
||||
"LabelAuthorFirstLast": "المؤلف (الاسم الأول الأخير)",
|
||||
"LabelAuthorLastFirst": "المؤلف (الاسم الأخير، الأول)",
|
||||
"LabelAuthors": "المؤلفون",
|
||||
"LabelAutoDownloadEpisodes": "تنزيل الحلقات تلقائيًا",
|
||||
"LabelAutoFetchMetadata": "جلب البيانات الوصفية تلقائيًا",
|
||||
"LabelAutoFetchMetadataHelp": "يجلب البيانات الوصفية للعنوان والمؤلف والسلسلة لتسهيل عملية الرفع. قد يلزم مطابقة بيانات وصفية إضافية بعد الرفع.",
|
||||
"LabelAutoLaunch": "تشغيل تلقائي",
|
||||
"LabelAutoLaunchDescription": "إعادة التوجيه إلى مزود المصادقة تلقائيًا عند الانتقال إلى صفحة تسجيل الدخول (مسار التجاوز اليدوي <code>/login?autoLaunch=0</code>)",
|
||||
"LabelAutoRegister": "تسجيل تلقائي",
|
||||
"LabelAutoRegisterDescription": "إنشاء مستخدمين جدد تلقائيًا بعد تسجيل الدخول",
|
||||
"LabelBackToUser": "العودة إلى المستخدم",
|
||||
"LabelBackupAudioFiles": "نسخ ملفات الصوت احتياطيًا",
|
||||
"LabelBackupLocation": "موقع النسخ الاحتياطي",
|
||||
"LabelBackupsEnableAutomaticBackups": "نسخ احتياطية تلقائية",
|
||||
"LabelBackupsEnableAutomaticBackupsHelp": "النسخ الاحتياطية المحفوظة في / البيانات الوصفية / النسخ الاحتياطية",
|
||||
"LabelBackupsMaxBackupSize": "الحد الأقصى لحجم النسخ الاحتياطي (بالجيجابايت) (0 لغير محدود)",
|
||||
"LabelBackupsMaxBackupSizeHelp": "كإجراء وقائي ضد سوء التكوين، ستفشل عمليات النسخ الاحتياطي إذا تجاوزت الحجم المحدد.",
|
||||
"LabelBackupsNumberToKeep": "عدد النسخ الاحتياطية التي يجب الاحتفاظ بها",
|
||||
"LabelBackupsNumberToKeepHelp": "ستتم إزالة نسخة احتياطية واحدة فقط في كل مرة، لذا إذا كان لديك بالفعل عدد نسخ احتياطية أكبر من هذا، فيجب عليك إزالتها يدويًا.",
|
||||
"LabelBitrate": "معدل البت",
|
||||
"LabelBonus": "مكافأة",
|
||||
"LabelBooks": "الكتب",
|
||||
"LabelButtonText": "نص الزر",
|
||||
"LabelByAuthor": "بواسطة {0}",
|
||||
"LabelChangePassword": "تغيير كلمة المرور",
|
||||
"LabelChannels": "قنوات",
|
||||
"LabelChapterCount": "{0} فصول",
|
||||
"LabelChapterTitle": "عنوان الفصل",
|
||||
"LabelChapters": "الفصول",
|
||||
"LabelChaptersFound": "تم العثور على فصول",
|
||||
"LabelClickForMoreInfo": "انقر لمزيد من المعلومات",
|
||||
"LabelClickToUseCurrentValue": "انقر لاستخدام القيمة الحالية",
|
||||
"LabelClosePlayer": "إغلاق المشغل",
|
||||
"LabelCodec": "برنامج الترميز",
|
||||
"LabelCollapseSeries": "إخفاء المسلسلات",
|
||||
"LabelCollapseSubSeries": "إخفاء المسلسلات الفرعية",
|
||||
"LabelCollection": "مجموعة",
|
||||
"LabelCollections": "مجموعات",
|
||||
"LabelComplete": "مكتمل",
|
||||
"LabelConfirmPassword": "تأكيد كلمة المرور",
|
||||
"LabelContinueListening": "استمرار الاستماع",
|
||||
"LabelContinueReading": "استمرار القراءة",
|
||||
"LabelContinueSeries": "استمرار المسلسلات",
|
||||
"LabelCover": "الغلاف",
|
||||
"LabelCoverImageURL": "رابط صورة الغلاف",
|
||||
"LabelCoverProvider": "مزود الغلاف",
|
||||
"LabelCreatedAt": "تاريخ الإنشاء",
|
||||
"LabelCronExpression": "تعبير Cron",
|
||||
"LabelCurrent": "الحالي",
|
||||
"LabelCurrently": "حاليًا:",
|
||||
"LabelCustomCronExpression": "تعبير Cron مخصص:",
|
||||
"LabelDatetime": "التاريخ والوقت",
|
||||
"LabelDays": "أيام",
|
||||
"LabelDeleteFromFileSystemCheckbox": "حذف من نظام الملفات (إلغاء التحديد للإزالة من قاعدة البيانات فقط)",
|
||||
"LabelDescription": "الوصف",
|
||||
"LabelDeselectAll": "إلغاء تحديد الكل",
|
||||
"LabelDevice": "الجهاز",
|
||||
"LabelDeviceInfo": "معلومات الجهاز",
|
||||
"LabelDeviceIsAvailableTo": "الجهاز متاح لـ...",
|
||||
"LabelDirectory": "مجلد / دليل",
|
||||
"LabelDiscFromFilename": "القرص من اسم الملف",
|
||||
"LabelDiscFromMetadata": "القرص من البيانات الوصفية",
|
||||
"LabelDiscover": "استكشف",
|
||||
"LabelDownload": "تنزيل",
|
||||
"LabelDownloadNEpisodes": "تنزيل {0} حلقات",
|
||||
"LabelDownloadable": "قابل للتنزيل",
|
||||
"LabelDuration": "المدة",
|
||||
"LabelDurationComparisonExactMatch": "(تطابق تام)",
|
||||
"LabelDurationComparisonLonger": "(أطول بـ {0})",
|
||||
"LabelDurationComparisonShorter": "({0} أقصر)",
|
||||
"LabelDurationFound": "المدة الموجودة:",
|
||||
"LabelEbook": "الكتاب الإلكتروني",
|
||||
"LabelEbooks": "الكتب الإلكترونية",
|
||||
"LabelEdit": "تعديل",
|
||||
"LabelEmail": "البريد الإلكتروني",
|
||||
"LabelEmailSettingsFromAddress": "عنوان المرسل",
|
||||
"LabelEmailSettingsRejectUnauthorized": "رفض الشهادات غير المصرح بها",
|
||||
"LabelEmailSettingsRejectUnauthorizedHelp": "قد يؤدي تعطيل التحقق من شهادة SSL إلى تعريض اتصالك لمخاطر أمنية، مثل هجمات الوسيط. لا تقم بتعطيل هذا الخيار إلا إذا كنت تفهم الآثار المترتبة عليه وتثق في خادم البريد الذي تتصل به.",
|
||||
"LabelEmailSettingsSecure": "آمن",
|
||||
"LabelEmailSettingsSecureHelp": "إذا كانت القيمة true، فسيستخدم الاتصال TLS عند الاتصال بالخادم. وإذا كانت false، فسيتم استخدام TLS إذا كان الخادم يدعم امتداد STARTTLS. في معظم الحالات، اضبط هذه القيمة على true إذا كنت تتصل بالمنفذ 465. أما بالنسبة للمنفذ 587 أو 25، فاحتفظ بها على false. (من nodemailer.com/smtp/#authentication)",
|
||||
"LabelEmailSettingsTestAddress": "عنوان الاختبار",
|
||||
"LabelEmbeddedCover": "غلاف مضمن",
|
||||
"LabelEnable": "تمكين",
|
||||
"LabelEncodingBackupLocation": "سيتم تخزين نسخة احتياطية من ملفاتك الصوتية الأصلية في:",
|
||||
"LabelEncodingChaptersNotEmbedded": "الفصول غير مضمنة في الكتب الصوتية متعددة المسارات.",
|
||||
"LabelEncodingClearItemCache": "تأكد من مسح ذاكرة التخزين المؤقت للعناصر بشكل دوري.",
|
||||
"LabelEncodingFinishedM4B": "سيتم وضع ملف M4B النهائي في مجلد الكتب الصوتية الخاص بك في:",
|
||||
"LabelEncodingInfoEmbedded": "سيتم تضمين البيانات الوصفية في المسارات الصوتية داخل مجلد الكتب الصوتية الخاص بك.",
|
||||
"LabelEncodingStartedNavigation": "بمجرد بدء المهمة، يمكنك الانتقال من هذه الصفحة.",
|
||||
"LabelEncodingTimeWarning": "قد تستغرق عملية الترميز ما يصل إلى 30 دقيقة.",
|
||||
"LabelEncodingWarningAdvancedSettings": "تحذير: لا تقم بتحديث هذه الإعدادات إلا إذا كنت على دراية بخيارات ترميز ffmpeg.",
|
||||
"LabelEncodingWatcherDisabled": "إذا قمت بتعطيل المراقب، فستحتاج إلى إعادة فحص هذا الكتاب الصوتي بعد ذلك.",
|
||||
"LabelEnd": "انهاء",
|
||||
"LabelEndOfChapter": "نهاية الفصل",
|
||||
"LabelEpisode": "الحلقة",
|
||||
"LabelEpisodeNotLinkedToRssFeed": "الحلقة غير مرتبطة بخلاصة RSS",
|
||||
"LabelEpisodeNumber": "الحلقة #{0}",
|
||||
"LabelEpisodeTitle": "عنوان الحلقة",
|
||||
"LabelEpisodeType": "نوع الحلقة",
|
||||
"LabelEpisodeUrlFromRssFeed": "رابط الحلقة من خلاصة RSS",
|
||||
"LabelEpisodes": "حلقات",
|
||||
"LabelEpisodic": "عرضي / حلقي",
|
||||
"LabelExample": "مثال",
|
||||
"LabelExpandSeries": "توسيع السلاسل",
|
||||
"LabelExpandSubSeries": "توسيع السلاسل الفرعية",
|
||||
"LabelExplicit": "صريح",
|
||||
"LabelExplicitChecked": "صريح (محدد)",
|
||||
"LabelExplicitUnchecked": "غير صريح (غير محدد)",
|
||||
"LabelExportOPML": "تصدير OPML",
|
||||
"LabelFeedURL": "عنوان التغذية",
|
||||
"LabelFetchingMetadata": "جلب البيانات الوصفية",
|
||||
"LabelFile": "الملف",
|
||||
"LabelFileBirthtime": "وقت انشاء الملف",
|
||||
"LabelFileBornDate": "تاريخ الإنشاء {0}",
|
||||
"LabelFileModified": "تم تعديل الملف",
|
||||
"LabelFileModifiedDate": "تم التعديل في {0}",
|
||||
"LabelFilename": "اسم الملف",
|
||||
"LabelFilterByUser": "تصفية حسب المستخدم",
|
||||
"LabelFindEpisodes": "البحث عن حلقات",
|
||||
"LabelFinished": "المنجزة",
|
||||
"LabelFolder": "المجلد",
|
||||
"LabelFolders": "مجلدات",
|
||||
"LabelFontBold": "عريض",
|
||||
"LabelFontBoldness": "تعريض الخط",
|
||||
"LabelFontFamily": "عائلة الخط",
|
||||
"LabelFontItalic": "مائل",
|
||||
"LabelFontScale": "نطاق الخط",
|
||||
"LabelFontStrikethrough": "يتوسطه خط",
|
||||
"LabelFormat": "تنسيق",
|
||||
"LabelFull": "كامل",
|
||||
"LabelGenre": "التصنيف",
|
||||
"LabelGenres": "التصانيف",
|
||||
"LabelHardDeleteFile": "حذف الملف نهائيًا",
|
||||
"LabelHasEbook": "يحتوي كتاب إلكتروني",
|
||||
"LabelHasSupplementaryEbook": "يحتوي كتاب إلكتروني تكميلي",
|
||||
"LabelHideSubtitles": "إخفاء الترجمة",
|
||||
"LabelHighestPriority": "الأولوية القصوى",
|
||||
"LabelHost": "المضيف",
|
||||
"LabelHour": "ساعة",
|
||||
"LabelHours": "ساعات",
|
||||
"LabelIcon": "أيقونة",
|
||||
"LabelImageURLFromTheWeb": "رابط الصورة من الويب",
|
||||
"LabelInProgress": "تحت التنفيذ",
|
||||
"LabelIncludeInTracklist": "تضمين في قائمة المسارات",
|
||||
"LabelIncomplete": "غير مكتمل",
|
||||
"LabelInterval": "فاصل زمني",
|
||||
"LabelIntervalCustomDailyWeekly": "يومي/أسبوعي مخصص",
|
||||
"LabelIntervalEvery12Hours": "كل 12 ساعة",
|
||||
"LabelIntervalEvery15Minutes": "كل 15 دقيقة",
|
||||
"LabelIntervalEvery2Hours": "كل ساعتين",
|
||||
"LabelIntervalEvery30Minutes": "كل 30 دقيقة",
|
||||
"LabelIntervalEvery6Hours": "كل 6 ساعات",
|
||||
"LabelIntervalEveryDay": "كل يوم",
|
||||
"LabelIntervalEveryHour": "كل ساعة",
|
||||
"LabelIntervalEveryMinute": "كل دقيقة",
|
||||
"LabelInvert": "عكس",
|
||||
"LabelItem": "عنصر",
|
||||
"LabelJumpBackwardAmount": "مقدار الرجوع للخلف",
|
||||
"LabelJumpForwardAmount": "مقدار التقدم للأمام",
|
||||
"LabelLanguage": "اللغة",
|
||||
"LabelLanguageDefaultServer": "لغة الخادم الافتراضية",
|
||||
"LabelLanguages": "اللغات",
|
||||
"LabelLastBookAdded": "آخر كتاب تمت إضافته",
|
||||
"LabelLastBookUpdated": "آخر كتاب تم تحديثه",
|
||||
"LabelLastSeen": "آخر ظهور",
|
||||
"LabelLastTime": "آخر مرة",
|
||||
"LabelLastUpdate": "آخر تحديث",
|
||||
"LabelLayout": "التنسيق",
|
||||
"LabelLayoutSinglePage": "صفحة واحدة",
|
||||
"LabelLayoutSplitPage": "صفحتان متقابلتان",
|
||||
"LabelLess": "أقل",
|
||||
"LabelLibrariesAccessibleToUser": "المكتبات المتاحة للمستخدم",
|
||||
"LabelLibrary": "مكتبة",
|
||||
"LabelLibraryFilterSublistEmpty": "لا يوجد {0}",
|
||||
"LabelLibraryItem": "عنصر المكتبة",
|
||||
"LabelLibraryName": "اسم المكتبة",
|
||||
"LabelLimit": "حد",
|
||||
"LabelLineSpacing": "تباعد الأسطر",
|
||||
"LabelListenAgain": "الاستماع مجدداً",
|
||||
"LabelMediaType": "نوع الوسائط"
|
||||
"LabelLogLevelDebug": "تصحيح الأخطاء",
|
||||
"LabelLogLevelInfo": "معلومات",
|
||||
"LabelLogLevelWarn": "تحذير",
|
||||
"LabelLookForNewEpisodesAfterDate": "البحث عن حلقات جديدة بعد هذا التاريخ",
|
||||
"LabelLowestPriority": "الأولوية الأدنى",
|
||||
"LabelMatchExistingUsersBy": "مطابقة المستخدمين الحاليين بواسطة",
|
||||
"LabelMatchExistingUsersByDescription": "يستخدم لربط المستخدمين الحاليين. بمجرد الاتصال، سيتم مطابقة المستخدمين بواسطة معرف فريد من مزود SSO الخاص بك",
|
||||
"LabelMaxEpisodesToDownload": "الحد الأقصى لعدد الحلقات التي سيتم تنزيلها. استخدم 0 لغير محدود.",
|
||||
"LabelMaxEpisodesToDownloadPerCheck": "الحد الأقصى لعدد الحلقات الجديدة التي سيتم تنزيلها في كل فحص",
|
||||
"LabelMaxEpisodesToKeep": "الحد الأقصى لعدد الحلقات التي سيتم الاحتفاظ بها",
|
||||
"LabelMaxEpisodesToKeepHelp": "القيمة 0 لا تضع حدًا أقصى. بعد تنزيل حلقة جديدة تلقائيًا، سيؤدي هذا إلى حذف أقدم حلقة إذا كان لديك أكثر من X حلقة. سيؤدي هذا إلى حذف حلقة واحدة فقط لكل تنزيل جديد.",
|
||||
"LabelMediaPlayer": "مشغل الوسائط",
|
||||
"LabelMediaType": "نوع الوسائط",
|
||||
"LabelMetaTag": "علامة بيانات وصفية",
|
||||
"LabelMetaTags": "علامات البيانات الوصفية",
|
||||
"LabelMetadataOrderOfPrecedenceDescription": "ستتجاوز مصادر البيانات الوصفية ذات الأولوية الأعلى مصادر البيانات الوصفية ذات الأولوية الأقل",
|
||||
"LabelMetadataProvider": "مزود البيانات الوصفية",
|
||||
"LabelMinute": "دقيقة",
|
||||
"LabelMinutes": "دقائق",
|
||||
"LabelMissing": "مفقود",
|
||||
"LabelMissingEbook": "لا يوجد كتاب إلكتروني",
|
||||
"LabelMissingSupplementaryEbook": "لا يوجد كتاب إلكتروني تكميلي",
|
||||
"LabelMobileRedirectURIs": "معرفات URI لإعادة التوجيه المسموح بها لتطبيقات الجوال",
|
||||
"LabelMobileRedirectURIsDescription": "هذه قائمة بيضاء لمعرفات URI لإعادة التوجيه الصالحة لتطبيقات الجوال. المعرف الافتراضي هو <code>audiobookshelf://oauth</code>، والذي يمكنك إزالته أو استكماله بمعرفات URI إضافية لتكامل تطبيقات الطرف الثالث. استخدام علامة النجمة (<code>*</code>) كإدخال وحيد يسمح بأي معرف URI.",
|
||||
"LabelMore": "أكثر",
|
||||
"LabelMoreInfo": "معلومات أكثر",
|
||||
"LabelName": "الاسم",
|
||||
"LabelNarrator": "الراوي",
|
||||
"LabelNarrators": "الرواة",
|
||||
"LabelNew": "جديد",
|
||||
"LabelNewPassword": "كلمة سر جديدة",
|
||||
"LabelNewestAuthors": "أجدد المؤلفين",
|
||||
"LabelNewestEpisodes": "أجدد الحلقات",
|
||||
"LabelNextBackupDate": "تاريخ النسخ الاحتياطي التالي",
|
||||
"LabelNextScheduledRun": "التشغيل المجدول التالي",
|
||||
"LabelNoCustomMetadataProviders": "لا يوجد مزودو بيانات وصفية مخصصون",
|
||||
"LabelNoEpisodesSelected": "لم يتم تحديد أي حلقات",
|
||||
"LabelNotFinished": "لم يتم الانتهاء",
|
||||
"LabelNotStarted": "لم يتم البدء",
|
||||
"LabelNotes": "ملاحظات",
|
||||
"LabelNotificationAppriseURL": "رابط (روابط) Apprise",
|
||||
"LabelNotificationAvailableVariables": "المتغيرات المتاحة",
|
||||
"LabelNotificationBodyTemplate": "قالب النص",
|
||||
"LabelNotificationEvent": "حدث الإشعار",
|
||||
"LabelNotificationTitleTemplate": "قالب العنوان",
|
||||
"LabelNotificationsMaxFailedAttempts": "الحد الأقصى لعدد المحاولات الفاشلة",
|
||||
"LabelNotificationsMaxFailedAttemptsHelp": "يتم تعطيل الإشعارات بمجرد فشل إرسالها لهذا العدد من المرات",
|
||||
"LabelNotificationsMaxQueueSize": "الحد الأقصى لحجم قائمة انتظار أحداث الإشعارات",
|
||||
"LabelNotificationsMaxQueueSizeHelp": "تقتصر الأحداث على التشغيل مرة واحدة في الثانية. سيتم تجاهل الأحداث إذا كانت قائمة الانتظار في الحد الأقصى لحجمها. هذا يمنع إرسال الإشعارات بشكل متكرر.",
|
||||
"LabelNumberOfBooks": "عدد الكتب",
|
||||
"LabelNumberOfEpisodes": "# من الحلقات",
|
||||
"LabelOpenIDAdvancedPermsClaimDescription": "اسم مطالبة OpenID التي تحتوي على أذونات متقدمة لإجراءات المستخدم داخل التطبيق والتي ستطبق على الأدوار غير الإدارية (<b>إذا تم تكوينها</b>). إذا كانت المطالبة مفقودة من الاستجابة، فسيتم رفض الوصول إلى ABS. إذا كان هناك خيار واحد مفقودًا، فسيتم التعامل معه على أنه <code>false</code>. تأكد من أن مطالبة موفر الهوية تطابق البنية المتوقعة:",
|
||||
"LabelOpenIDClaims": "اترك الخيارات التالية فارغة لتعطيل تعيين المجموعة والأذونات المتقدمة، وسيتم تعيين مجموعة \"مستخدم\" تلقائيًا بعد ذلك.",
|
||||
"LabelOpenIDGroupClaimDescription": "اسم مطالبة OpenID التي تحتوي على قائمة بمجموعات المستخدم. يشار إليها عادةً باسم <code>groups</code>.<b>إذا تم تكوينها</b>، فسيقوم التطبيق تلقائيًا بتعيين الأدوار بناءً على عضويات مجموعة المستخدم، بشرط أن تسمى هذه المجموعات بشكل غير حساس لحالة الأحرف \"admin\" أو \"user\" أو \"guest\" في المطالبة. يجب أن تحتوي المطالبة على قائمة، وإذا كان المستخدم ينتمي إلى مجموعات متعددة، فسيقوم التطبيق بتعيين الدور المقابل لأعلى مستوى من الوصول. إذا لم تتطابق أي مجموعة، فسيتم رفض الوصول.",
|
||||
"LabelOpenRSSFeed": "تغذية RSS مفتوحة",
|
||||
"LabelOverwrite": "استبدال",
|
||||
"LabelPaginationPageXOfY": "صفحة {0} من {1}",
|
||||
"LabelPassword": "كلمة المرور",
|
||||
"LabelPath": "مسار",
|
||||
"LabelPermanent": "دائم",
|
||||
"LabelPermissionsAccessAllLibraries": "يمكنه الوصول إلى جميع المكتبات",
|
||||
"LabelPermissionsAccessAllTags": "يمكنه الوصول إلى جميع العلامات",
|
||||
"LabelPermissionsAccessExplicitContent": "يمكنه الوصول إلى المحتوى الصريح",
|
||||
"LabelPermissionsCreateEreader": "يمكنه إنشاء قارئ إلكتروني",
|
||||
"LabelPermissionsDelete": "يمكنه الحذف",
|
||||
"LabelPermissionsDownload": "يمكنه التنزيل",
|
||||
"LabelPermissionsUpdate": "يمكنه التحديث",
|
||||
"LabelPermissionsUpload": "يمكنه الرفع",
|
||||
"LabelPersonalYearReview": "ملخص عامك ({0})",
|
||||
"LabelPhotoPathURL": "مسار/رابط الصورة",
|
||||
"LabelPlayMethod": "طريقة التشغيل",
|
||||
"LabelPlaybackRateIncrementDecrement": "مقدار زيادة/نقصان سرعة التشغيل",
|
||||
"LabelPlayerChapterNumberMarker": "{0} من {1}",
|
||||
"LabelPlaylists": "قوائم التشغيل",
|
||||
"LabelPodcast": "مدونة صوتية",
|
||||
"LabelPodcastSearchRegion": "منطقة البحث عن البودكاست",
|
||||
"LabelPodcastType": "نوع البودكاست",
|
||||
"LabelPodcasts": "بودكاست",
|
||||
"LabelPort": "منفذ",
|
||||
"LabelPrefixesToIgnore": "البادئات التي يجب تجاهلها (غير حساسة لحالة الأحرف)",
|
||||
"LabelPreventIndexing": "منع فهرسة تغذيتك بواسطة دليل آيتونز وقوقل بودكاست",
|
||||
"LabelPrimaryEbook": "الكتاب الإلكتروني الأساسي",
|
||||
"LabelProgress": "تقدم",
|
||||
"LabelProvider": "مزود",
|
||||
"LabelProviderAuthorizationValue": "قيمة رأس التفويض",
|
||||
"LabelPubDate": "تاريخ النشر",
|
||||
"LabelPublishYear": "سنة النشر",
|
||||
"LabelPublishedDate": "منشور {0}",
|
||||
"LabelPublishedDecade": "عقد النشر",
|
||||
"LabelPublishedDecades": "عقود النشر",
|
||||
"LabelPublisher": "الناشر",
|
||||
"LabelPublishers": "الناشرون",
|
||||
"LabelRSSFeedCustomOwnerEmail": "البريد الالكتروني المخصص للمالك",
|
||||
"LabelRSSFeedCustomOwnerName": "الاسم المخصص للمالك",
|
||||
"LabelRSSFeedOpen": "فتح تغذية RSS",
|
||||
"LabelRSSFeedPreventIndexing": "منع الفهرسة",
|
||||
"LabelRSSFeedSlug": "اسم تعريف تغذية RSS",
|
||||
"LabelRSSFeedURL": "رابط تغذية RSS",
|
||||
"LabelRandomly": "عشوائياً",
|
||||
"LabelReAddSeriesToContinueListening": "إعادة إضافة السلسلة إلى \"متابعة الاستماع\"",
|
||||
"LabelRead": "اقرأ",
|
||||
"LabelReadAgain": "اقرأ مرة أخرى",
|
||||
"LabelReadEbookWithoutProgress": "قراءة الكتاب الإلكتروني دون حفظ التقدم",
|
||||
"LabelRecentSeries": "المسلسلات الحديثة",
|
||||
"LabelRecentlyAdded": "المضافة حديثاً",
|
||||
"LabelRecommended": "موصى به",
|
||||
"LabelRedo": "إعادة",
|
||||
"LabelRegion": "المنطقة",
|
||||
"LabelReleaseDate": "تاريخ الإصدار",
|
||||
"LabelRemoveAllMetadataAbs": "إزالة جميع ملفات metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "إزالة جميع ملفات metadata.json",
|
||||
"LabelRemoveAudibleBranding": "إزالة مقدمة وخاتمة Audible من الفصول",
|
||||
"LabelRemoveCover": "إزالة الغلاف",
|
||||
"LabelRemoveMetadataFile": "إزالة ملفات البيانات الوصفية في مجلدات عناصر المكتبة",
|
||||
"LabelRemoveMetadataFileHelp": "إزالة جميع ملفات metadata.json و metadata.abs في مجلدات {0} الخاصة بك.",
|
||||
"LabelRowsPerPage": "عدد الصفوف في الصفحة",
|
||||
"LabelSearchTerm": "مصطلح البحث",
|
||||
"LabelSearchTitle": "بحث بالعنوان",
|
||||
"LabelSearchTitleOrASIN": "بحث بالعنوان أو ASIN",
|
||||
"LabelSeason": "الموسم",
|
||||
"LabelSeasonNumber": "الموسم #{0}",
|
||||
"LabelSelectAll": "تحديد الكل",
|
||||
"LabelSelectAllEpisodes": "تحديد جميع الحلقات",
|
||||
"LabelSelectEpisodesShowing": "تحديد {0} حلقة معروضة",
|
||||
"LabelSelectUsers": "تحديد المستخدمين",
|
||||
"LabelSendEbookToDevice": "إرسال الكتاب الإلكتروني إلى...",
|
||||
"LabelSequence": "تسلسل",
|
||||
"LabelSerial": "مسلسل",
|
||||
"LabelSeries": "المسلسلات",
|
||||
"LabelSeriesName": "اسم السلسلة",
|
||||
"LabelSeriesProgress": "تقدم السلسلة",
|
||||
"LabelServerLogLevel": "مستوى سجل الخادم",
|
||||
"LabelServerYearReview": "ملخص عام الخادم ({0})",
|
||||
"LabelSetEbookAsPrimary": "تعيين كرئيسي",
|
||||
"LabelSetEbookAsSupplementary": "تعيين كتكميلي",
|
||||
"LabelSettingsAllowIframe": "السماح بالتضمين في إطار iframe",
|
||||
"LabelSettingsAudiobooksOnly": "كتب صوتية فقط",
|
||||
"LabelSettingsAudiobooksOnlyHelp": "سيؤدي تمكين هذا الإعداد إلى تجاهل ملفات الكتب الإلكترونية ما لم تكن داخل مجلد كتاب صوتي، وفي هذه الحالة سيتم تعيينها ككتب إلكترونية تكميلية",
|
||||
"LabelSettingsBookshelfViewHelp": "تصميم يحاكي الواقع مع رفوف خشبية",
|
||||
"LabelSettingsChromecastSupport": "دعم Chromecast",
|
||||
"LabelSettingsDateFormat": "تنسيق التاريخ",
|
||||
"LabelSettingsEnableWatcher": "فحص المكتبات تلقائيًا بحثًا عن تغييرات",
|
||||
"LabelSettingsEnableWatcherForLibrary": "فحص المكتبة تلقائيًا بحثًا عن تغييرات",
|
||||
"LabelSettingsEnableWatcherHelp": "يمكّن الإضافة/التحديث التلقائي للعناصر عند اكتشاف تغييرات في الملفات. *يتطلب إعادة تشغيل الخادم",
|
||||
"LabelSettingsEpubsAllowScriptedContent": "السماح بالمحتوى النصي في ملفات epub",
|
||||
"LabelSettingsEpubsAllowScriptedContentHelp": "السماح لملفات epub بتنفيذ النصوص البرمجية. يوصى بإبقاء هذا الإعداد معطلاً ما لم تثق في مصدر ملفات epub.",
|
||||
"LabelSettingsExperimentalFeatures": "ميزات تجريبية",
|
||||
"LabelSettingsExperimentalFeaturesHelp": "ميزات قيد التطوير يمكنها استخدام ملاحظاتك والمساعدة في اختبارها. انقر لفتح مناقشة على GitHub.",
|
||||
"LabelSettingsFindCovers": "البحث عن الأغلفة",
|
||||
"LabelSettingsFindCoversHelp": "إذا لم يكن لدى كتابك الصوتي غلاف مضمن أو صورة غلاف داخل المجلد، فسيحاول الماسح الضوئي العثور على غلاف.<br> ملاحظة: سيؤدي هذا إلى إطالة وقت الفحص",
|
||||
"LabelSettingsHideSingleBookSeries": "إخفاء السلاسل ذات الكتاب الواحد",
|
||||
"LabelSettingsHideSingleBookSeriesHelp": "سيتم إخفاء السلاسل التي تحتوي على كتاب واحد من صفحة السلاسل وأرفف الصفحة الرئيسية.",
|
||||
"LabelSettingsHomePageBookshelfView": "استخدام عرض الرفوف في الصفحة الرئيسية",
|
||||
"LabelSettingsLibraryBookshelfView": "استخدام عرض الرفوف في المكتبة",
|
||||
"LabelSettingsLibraryMarkAsFinishedPercentComplete": "النسبة المئوية المكتملة أكبر من",
|
||||
"LabelSettingsLibraryMarkAsFinishedTimeRemaining": "الوقت المتبقي أقل من (ثواني)",
|
||||
"LabelSettingsLibraryMarkAsFinishedWhen": "تعليم عنصر الوسائط على أنه منتهٍ عند",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeries": "تخطي الكتب السابقة في \"متابعة السلسلة\"",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeriesHelp": "يعرض رف \"متابعة السلسلة\" في الصفحة الرئيسية أول كتاب لم يبدأ في السلاسل التي تحتوي على كتاب واحد على الأقل منتهي ولا يوجد كتب قيد التقدم. سيؤدي تمكين هذا الإعداد إلى متابعة السلاسل من أبعد كتاب مكتمل بدلاً من أول كتاب لم يبدأ.",
|
||||
"LabelSettingsParseSubtitles": "تحليل الترجمة",
|
||||
"LabelSettingsParseSubtitlesHelp": "استخراج الترجمة من أسماء مجلدات الكتب الصوتية.<br>يجب فصل الترجمة بـ \" - \"<br>مثال: \"عنوان الكتاب - ترجمة هنا\" تحتوي على الترجمة \"ترجمة هنا\"",
|
||||
"LabelSettingsPreferMatchedMetadata": "تفضيل البيانات الوصفية المطابقة",
|
||||
"LabelSettingsPreferMatchedMetadataHelp": "ستتجاوز البيانات المطابقة تفاصيل العنصر عند استخدام المطابقة السريعة. بشكل افتراضي، ستملأ المطابقة السريعة التفاصيل المفقودة فقط.",
|
||||
"LabelSettingsSkipMatchingBooksWithASIN": "تخطي مطابقة الكتب التي لديها ASIN بالفعل",
|
||||
"LabelSettingsSkipMatchingBooksWithISBN": "تخطي مطابقة الكتب التي لديها ISBN بالفعل",
|
||||
"LabelSettingsSortingIgnorePrefixes": "تجاهل البادئات عند الفرز",
|
||||
"LabelSettingsSortingIgnorePrefixesHelp": "مثال: بالنسبة للبادئة \"the\"، سيتم فرز عنوان الكتاب \"The Book Title\" كـ \"Book Title, The\"",
|
||||
"LabelSettingsSquareBookCovers": "استخدام أغلفة كتب مربعة",
|
||||
"LabelSettingsSquareBookCoversHelp": "تفضيل استخدام الأغلفة المربعة على أغلفة الكتب القياسية بنسبة 1.6:1",
|
||||
"LabelSettingsStoreCoversWithItem": "تخزين الأغلفة مع العنصر",
|
||||
"LabelSettingsStoreCoversWithItemHelp": "بشكل افتراضي، يتم تخزين الأغلفة في /metadata/items، وسيؤدي تمكين هذا الإعداد إلى تخزين الأغلفة في مجلد عنصر المكتبة الخاص بك. سيتم الاحتفاظ بملف واحد فقط باسم \"cover\"",
|
||||
"LabelSettingsStoreMetadataWithItem": "تخزين البيانات الوصفية مع العنصر",
|
||||
"LabelSettingsStoreMetadataWithItemHelp": "بشكل افتراضي، يتم تخزين ملفات البيانات الوصفية في /metadata/items، وسيؤدي تمكين هذا الإعداد إلى تخزين ملفات البيانات الوصفية في مجلدات عناصر المكتبة الخاصة بك",
|
||||
"LabelSettingsTimeFormat": "تنسيق الوقت",
|
||||
"LabelShare": "مشاركة",
|
||||
"LabelShareDownloadableHelp": "يسمح للمستخدمين الذين لديهم رابط المشاركة بتنزيل ملف مضغوط لعنصر المكتبة.",
|
||||
"LabelShareOpen": "فتح المشاركة",
|
||||
"LabelShareURL": "رابط المشاركة",
|
||||
"LabelShowAll": "إظهار الكل",
|
||||
"LabelShowSeconds": "إظهار الثواني",
|
||||
"LabelShowSubtitles": "إظهار الترجمة",
|
||||
"LabelSize": "الحجم",
|
||||
"LabelSleepTimer": "مؤقت النوم",
|
||||
"LabelSlug": "اسم تعريفي سهل القراءة",
|
||||
"LabelSortAscending": "تصاعدي",
|
||||
"LabelSortDescending": "تنازلي",
|
||||
"LabelSortPubDate": "فرز حسب تاريخ النشر",
|
||||
"LabelStart": "ابدأ",
|
||||
"LabelStartTime": "وقت البدء",
|
||||
"LabelStarted": "بدأ",
|
||||
"LabelStartedAt": "بدأ في",
|
||||
"LabelStatsAudioTracks": "مسارات الصوت",
|
||||
"LabelStatsAuthors": "المؤلفون",
|
||||
"LabelStatsBestDay": "أفضل يوم",
|
||||
"LabelStatsDailyAverage": "المتوسط اليومي",
|
||||
"LabelStatsDays": "أيام",
|
||||
"LabelStatsDaysListened": "أيام الاستماع",
|
||||
"LabelStatsHours": "ساعات",
|
||||
"LabelStatsInARow": "على التوالي",
|
||||
"LabelStatsItemsFinished": "العناصر المنتهية",
|
||||
"LabelStatsItemsInLibrary": "العناصر في المكتبة",
|
||||
"LabelStatsMinutes": "دقائق",
|
||||
"LabelStatsMinutesListening": "دقائق الاستماع",
|
||||
"LabelStatsOverallDays": "إجمالي الأيام",
|
||||
"LabelStatsOverallHours": "إجمالي الساعات",
|
||||
"LabelStatsWeekListening": "استماع هذا الأسبوع",
|
||||
"LabelSubtitle": "عنوان فرعي / ترجمة",
|
||||
"LabelSupportedFileTypes": "أنواع الملفات المدعومة",
|
||||
"LabelTag": "علامة",
|
||||
"LabelTags": "علامات",
|
||||
"LabelTagsAccessibleToUser": "العلامات المتاحة للمستخدم",
|
||||
"LabelTagsNotAccessibleToUser": "العلامات غير المتاحة للمستخدم",
|
||||
"LabelTasks": "المهام قيد التشغيل",
|
||||
"LabelTextEditorBulletedList": "قائمة نقطية",
|
||||
"LabelTextEditorLink": "رابط",
|
||||
"LabelTextEditorNumberedList": "قائمة مرقمة",
|
||||
"LabelTextEditorUnlink": "إزالة الرابط",
|
||||
"LabelTheme": "النمط",
|
||||
"LabelThemeDark": "غامق",
|
||||
"LabelThemeLight": "فاتح",
|
||||
"LabelTimeBase": "قاعدة الوقت",
|
||||
"LabelTimeDurationXHours": "{0} ساعات",
|
||||
"LabelTimeDurationXMinutes": "{0} دقائق",
|
||||
"LabelTimeDurationXSeconds": "{0} ثواني",
|
||||
"LabelTimeInMinutes": "الوقت بالدقائق",
|
||||
"LabelTimeLeft": "باقي {0}",
|
||||
"LabelTimeListened": "الوقت المستمع إليه",
|
||||
"LabelTimeListenedToday": "الوقت المستمع إليه اليوم",
|
||||
"LabelTimeRemaining": "{0} متبقية",
|
||||
"LabelTimeToShift": "الوقت المراد إزاحته بالثواني",
|
||||
"LabelTitle": "عنوان",
|
||||
"LabelToolsEmbedMetadata": "تضمين البيانات الوصفية",
|
||||
"LabelToolsEmbedMetadataDescription": "تضمين البيانات الوصفية في ملفات الصوت بما في ذلك صورة الغلاف والفصول.",
|
||||
"LabelToolsM4bEncoder": "ترميز M4B",
|
||||
"LabelToolsMakeM4b": "إنشاء ملف كتاب صوتي M4B",
|
||||
"LabelToolsMakeM4bDescription": "إنشاء ملف كتاب صوتي .M4B مع بيانات وصفية مضمنة وصورة غلاف وفصول.",
|
||||
"LabelToolsSplitM4b": "تقسيم M4B إلى ملفات MP3",
|
||||
"LabelToolsSplitM4bDescription": "إنشاء ملفات MP3 من ملف M4B مقسم حسب الفصول مع بيانات وصفية مضمنة وصورة غلاف وفصول.",
|
||||
"LabelTotalDuration": "المدة الكلية",
|
||||
"LabelTotalTimeListened": "إجمالي وقت الاستماع",
|
||||
"LabelTrackFromFilename": "المسار من اسم الملف",
|
||||
"LabelTrackFromMetadata": "المسار من البيانات الوصفية",
|
||||
"LabelTracks": "المسارات",
|
||||
"LabelTracksMultiTrack": "متعدد المسارات",
|
||||
"LabelTracksNone": "لا توجد مسارات",
|
||||
"LabelTracksSingleTrack": "مسار واحد",
|
||||
"LabelTrailer": "مقطع دعائي",
|
||||
"LabelType": "نوع",
|
||||
"LabelUnabridged": "غير مختصر",
|
||||
"LabelUndo": "تراجع",
|
||||
"LabelUnknown": "مجهول",
|
||||
"LabelUnknownPublishDate": "تاريخ النشر مجهول",
|
||||
"LabelUpdateCover": "تحديث الغلاف",
|
||||
"LabelUpdateCoverHelp": "السماح باستبدال الأغلفة الموجودة للكتب المحددة عند العثور على تطابق",
|
||||
"LabelUpdateDetails": "تحديث التفاصيل",
|
||||
"LabelUpdateDetailsHelp": "السماح باستبدال التفاصيل الموجودة للكتب المحددة عند العثور على تطابق",
|
||||
"LabelUpdatedAt": "تاريخ التحديث",
|
||||
"LabelUploaderDragAndDrop": "اسحب وأفلت الملفات أو المجلدات",
|
||||
"LabelUploaderDragAndDropFilesOnly": "اسحب وأفلت الملفات",
|
||||
"LabelUploaderDropFiles": "إفلات الملفات",
|
||||
"LabelUploaderItemFetchMetadataHelp": "جلب العنوان والمؤلف والسلسلة تلقائيًا",
|
||||
"LabelUseAdvancedOptions": "استخدام الخيارات المتقدمة",
|
||||
"LabelUseChapterTrack": "استخدام مسار الفصل",
|
||||
"LabelUseFullTrack": "استخدام المسار الكامل",
|
||||
"LabelUseZeroForUnlimited": "استخدم 0 لغير محدود",
|
||||
"LabelUser": "مستخدم",
|
||||
"LabelUsername": "اسم المستخدم",
|
||||
"LabelValue": "القيمة",
|
||||
"LabelVersion": "الإصدار",
|
||||
"LabelViewBookmarks": "عرض الإشارات المرجعية",
|
||||
"LabelViewChapters": "عرض الفصول",
|
||||
"LabelViewPlayerSettings": "عرض إعدادات المشغل",
|
||||
"LabelViewQueue": "عرض قائمة انتظار المشغل",
|
||||
"LabelVolume": "مستوى الصوت",
|
||||
"LabelWebRedirectURLsDescription": "قم بتخويل عناوين URL هذه في موفر OAuth الخاص بك للسماح بإعادة التوجيه إلى تطبيق الويب بعد تسجيل الدخول:",
|
||||
"LabelWebRedirectURLsSubfolder": "مجلد فرعي لعناوين URL لإعادة التوجيه",
|
||||
"LabelWeekdaysToRun": "أيام الأسبوع المراد التشغيل فيها",
|
||||
"LabelXBooks": "{0} كتب",
|
||||
"LabelXItems": "{0} عناصر",
|
||||
"LabelYearReviewHide": "إخفاء ملخص العام",
|
||||
"LabelYearReviewShow": "عرض ملخص العام",
|
||||
"LabelYourAudiobookDuration": "مدة كتابك الصوتي",
|
||||
"LabelYourBookmarks": "علاماتك المرجعية",
|
||||
"LabelYourPlaylists": "قوائم التشغيل الخاصة بك",
|
||||
"LabelYourProgress": "تقدمك",
|
||||
"MessageAddToPlayerQueue": "إضافة إلى قائمة انتظار المشغل",
|
||||
"MessageAppriseDescription": "لاستخدام هذه الميزة، ستحتاج إلى تشغيل مثيل <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> أو واجهة برمجة تطبيقات تتعامل مع نفس الطلبات. <br />يجب أن يكون عنوان URL الخاص بـ Apprise API هو مسار URL الكامل لإرسال الإشعار، على سبيل المثال، إذا كان مثيل واجهة برمجة التطبيقات الخاصة بك يعمل على <code>http://192.168.1.1:8337</code>، فستضع <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "تأكد من أنك تستخدم ASIN من منطقة Audible الصحيحة، وليس Amazon.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "أعد تشغيل الخادم بعد الحفظ لتطبيق تغييرات OIDC.",
|
||||
"MessageBackupsDescription": "تتضمن النسخ الاحتياطية المستخدمين وتقدم المستخدم وتفاصيل عنصر المكتبة وإعدادات الخادم والصور المخزنة في <code>/metadata/items</code> و <code>/metadata/authors</code>. <strong>لا</strong> تتضمن النسخ الاحتياطية أي ملفات مخزنة في مجلدات مكتبتك.",
|
||||
"MessageBackupsLocationEditNote": "ملاحظة: لن يؤدي تحديث موقع النسخ الاحتياطي إلى نقل أو تعديل النسخ الاحتياطية الموجودة",
|
||||
"MessageBackupsLocationNoEditNote": "ملاحظة: يتم تعيين موقع النسخ الاحتياطي من خلال متغير بيئة ولا يمكن تغييره هنا.",
|
||||
"MessageBackupsLocationPathEmpty": "لا يمكن أن يكون مسار موقع النسخ الاحتياطي فارغًا",
|
||||
"MessageBatchEditPopulateMapDetailsAllHelp": "املأ الحقول الممكّنة ببيانات من جميع العناصر. سيتم دمج الحقول ذات القيم المتعددة",
|
||||
"MessageBatchEditPopulateMapDetailsItemHelp": "املأ حقول تفاصيل الخريطة الممكّنة ببيانات من هذا العنصر",
|
||||
"MessageBatchQuickMatchDescription": "ستحاول المطابقة السريعة إضافة الأغلفة والبيانات الوصفية المفقودة للعناصر المحددة. قم بتمكين الخيارات أدناه للسماح للمطابقة السريعة بالكتابة فوق الأغلفة و/أو البيانات الوصفية الموجودة.",
|
||||
"MessageBookshelfNoCollections": "لم تنشئ أي مجموعات حتى الآن",
|
||||
"MessageBookshelfNoCollectionsHelp": "المجموعات عامة. يمكن لجميع المستخدمين الذين لديهم حق الوصول إلى المكتبة رؤيتها.",
|
||||
"MessageBookshelfNoRSSFeeds": "لا توجد خلاصات RSS مفتوحة",
|
||||
"MessageBookshelfNoResultsForFilter": "لا توجد نتائج للفلتر \"{0}: {1}\"",
|
||||
"MessageBookshelfNoResultsForQuery": "لا توجد نتائج للاستعلام",
|
||||
"MessageBookshelfNoSeries": "ليس لديك أي مسلسلات",
|
||||
"MessageChapterEndIsAfter": "نهاية الفصل بعد نهاية كتابك الصوتي",
|
||||
"MessageChapterErrorFirstNotZero": "يجب أن يبدأ الفصل الأول عند 0",
|
||||
"MessageChapterErrorStartGteDuration": "يجب أن يكون وقت البدء غير الصالح أقل من مدة الكتاب الصوتي",
|
||||
"MessageChapterErrorStartLtPrev": "يجب أن يكون وقت البدء غير الصالح أكبر من أو يساوي وقت بدء الفصل السابق",
|
||||
"MessageChapterStartIsAfter": "بداية الفصل بعد نهاية كتابك الصوتي",
|
||||
"MessageChaptersNotFound": "لم يتم العثور على فصول",
|
||||
"MessageCheckingCron": "جارٍ فحص cron...",
|
||||
"MessageConfirmCloseFeed": "هل أنت متأكد أنك تريد إغلاق هذه التغذية؟",
|
||||
"MessageConfirmDeleteBackup": "هل أنت متأكد أنك تريد حذف النسخ الاحتياطي لـ {0}؟",
|
||||
"MessageConfirmDeleteDevice": "هل أنت متأكد أنك تريد حذف جهاز القارئ الإلكتروني \"{0}\"؟",
|
||||
"MessageConfirmDeleteFile": "سيؤدي هذا إلى حذف الملف من نظام الملفات الخاص بك. هل أنت متأكد؟",
|
||||
"MessageConfirmDeleteLibrary": "هل أنت متأكد أنك تريد حذف المكتبة \"{0}\" نهائيًا؟",
|
||||
"MessageConfirmDeleteLibraryItem": "سيؤدي هذا إلى حذف عنصر المكتبة من قاعدة البيانات ونظام الملفات الخاص بك. هل أنت متأكد؟",
|
||||
"MessageConfirmDeleteLibraryItems": "سيؤدي هذا إلى حذف {0} عنصرًا من عناصر المكتبة من قاعدة البيانات ونظام الملفات الخاص بك. هل أنت متأكد؟",
|
||||
"MessageConfirmDeleteMetadataProvider": "هل أنت متأكد أنك تريد حذف مزود البيانات الوصفية المخصص \"{0}\"؟",
|
||||
"MessageConfirmDeleteNotification": "هل أنت متأكد أنك تريد حذف هذا الإشعار؟",
|
||||
"MessageConfirmDeleteSession": "هل أنت متأكد أنك تريد حذف هذه الجلسة؟",
|
||||
"MessageConfirmEmbedMetadataInAudioFiles": "هل أنت متأكد أنك تريد تضمين البيانات الوصفية في {0} ملفًا صوتيًا؟",
|
||||
"MessageConfirmForceReScan": "هل أنت متأكد أنك تريد فرض إعادة الفحص؟",
|
||||
"MessageConfirmMarkAllEpisodesFinished": "هل أنت متأكد أنك تريد تعليم جميع الحلقات على أنها منتهية؟",
|
||||
"MessageConfirmMarkAllEpisodesNotFinished": "هل أنت متأكد أنك تريد تعليم جميع الحلقات على أنها غير منتهية؟",
|
||||
"MessageConfirmMarkItemFinished": "هل أنت متأكد أنك تريد تعليم \"{0}\" على أنه منتهٍ؟",
|
||||
"MessageConfirmMarkItemNotFinished": "هل أنت متأكد أنك تريد تعليم \"{0}\" على أنه غير منتهٍ؟",
|
||||
"MessageConfirmMarkSeriesFinished": "هل أنت متأكد أنك تريد تعليم جميع الكتب في هذه السلسلة على أنها منتهية؟",
|
||||
"MessageConfirmMarkSeriesNotFinished": "هل أنت متأكد أنك تريد تعليم جميع الكتب في هذه السلسلة على أنها غير منتهية؟",
|
||||
"MessageConfirmNotificationTestTrigger": "هل تريد تشغيل هذا الإشعار ببيانات اختبار؟",
|
||||
"MessageConfirmPurgeCache": "سيؤدي مسح ذاكرة التخزين المؤقت إلى حذف الدليل بأكمله في <code>/metadata/cache</code>. <br /><br />هل أنت متأكد أنك تريد إزالة دليل ذاكرة التخزين المؤقت؟",
|
||||
"MessageConfirmPurgeItemsCache": "سيؤدي مسح ذاكرة التخزين المؤقت للعناصر إلى حذف الدليل بأكمله في <code>/metadata/cache/items</code> <br />هل أنت متأكد؟",
|
||||
"MessageConfirmQuickEmbed": "تحذير! لن يقوم التضمين السريع بنسخ ملفاتك الصوتية احتياطيًا. تأكد من أن لديك نسخة احتياطية من ملفاتك الصوتية. <br><br>هل ترغب في المتابعة؟",
|
||||
"MessageConfirmQuickMatchEpisodes": "ستؤدي المطابقة السريعة للحلقات إلى الكتابة فوق التفاصيل إذا تم العثور على تطابق. سيتم تحديث الحلقات غير المتطابقة فقط. هل أنت متأكد؟",
|
||||
"MessageConfirmReScanLibraryItems": "هل أنت متأكد أنك تريد إعادة فحص {0} عنصرًا؟",
|
||||
"MessageConfirmRemoveAllChapters": "هل أنت متأكد أنك تريد إزالة جميع الفصول؟",
|
||||
"MessageConfirmRemoveAuthor": "هل أنت متأكد أنك تريد إزالة المؤلف \"{0}\"؟",
|
||||
"MessageConfirmRemoveCollection": "هل أنت متأكد أنك تريد إزالة المجموعة \"{0}\"؟",
|
||||
"MessageConfirmRemoveEpisode": "هل أنت متأكد أنك تريد إزالة الحلقة \"{0}\"؟",
|
||||
"MessageConfirmRemoveEpisodes": "هل أنت متأكد أنك تريد إزالة {0} حلقة؟",
|
||||
"MessageConfirmRemoveListeningSessions": "هل أنت متأكد أنك تريد إزالة {0} جلسة استماع؟",
|
||||
"MessageConfirmRemoveMetadataFiles": "هل أنت متأكد أنك تريد إزالة جميع ملفات البيانات الوصفية {0} الموجودة في مجلدات عناصر مكتبتك؟",
|
||||
"MessageConfirmRemoveNarrator": "هل أنت متأكد أنك تريد إزالة الراوي \"{0}\"؟",
|
||||
"MessageConfirmRemovePlaylist": "هل أنت متأكد أنك تريد إزالة قائمة التشغيل الخاصة بك \"{0}\"؟",
|
||||
"MessageConfirmRenameGenre": "هل أنت متأكد أنك تريد إعادة تسمية النوع \"{0}\" إلى \"{1}\" لجميع العناصر؟",
|
||||
"MessageConfirmRenameGenreMergeNote": "ملاحظة: هذا النوع موجود بالفعل لذا سيتم دمجهما.",
|
||||
"MessageConfirmRenameGenreWarning": "تحذير! يوجد نوع مشابه بحالة أحرف مختلفة بالفعل \"{0}\".",
|
||||
"MessageConfirmRenameTag": "هل أنت متأكد أنك تريد إعادة تسمية العلامة \"{0}\" إلى \"{1}\" لجميع العناصر؟",
|
||||
"MessageConfirmRenameTagMergeNote": "ملاحظة: هذه العلامة موجودة بالفعل لذا سيتم دمجهما.",
|
||||
"MessageConfirmRenameTagWarning": "تحذير! توجد علامة مشابهة بحالة أحرف مختلفة بالفعل \"{0}\".",
|
||||
"MessageConfirmResetProgress": "هل أنت متأكد أنك تريد إعادة تعيين تقدمك؟",
|
||||
"MessageConfirmSendEbookToDevice": "هل أنت متأكد أنك تريد إرسال الكتاب الإلكتروني \"{1}\" ({0}) إلى الجهاز \"{2}\"؟",
|
||||
"MessageConfirmUnlinkOpenId": "هل أنت متأكد أنك تريد فصل هذا المستخدم عن OpenID؟",
|
||||
"MessageDaysListenedInTheLastYear": "تم الاستماع لمدة {0} يومًا في العام الماضي",
|
||||
"MessageDownloadingEpisode": "جاري تنزيل الحلقة",
|
||||
"MessageDragFilesIntoTrackOrder": "اسحب الملفات إلى ترتيب المسارات الصحيح",
|
||||
"MessageEmbedFailed": "فشل التضمين!",
|
||||
"MessageEmbedFinished": "تم الانتهاء من التضمين!",
|
||||
"MessageEmbedQueue": "تمت إضافته إلى قائمة انتظار تضمين البيانات الوصفية ({0} في قائمة الانتظار)",
|
||||
"MessageEpisodesQueuedForDownload": "تمت إضافة {0} حلقة (حلقات) إلى قائمة انتظار التنزيل",
|
||||
"MessageEreaderDevices": "لضمان تسليم الكتب الإلكترونية، قد تحتاج إلى إضافة عنوان البريد الإلكتروني أعلاه كمرسل صالح لكل جهاز مدرج أدناه.",
|
||||
"MessageFeedURLWillBe": "سيكون رابط التغذية هو {0}",
|
||||
"MessageFetching": "جاري الجلب...",
|
||||
"MessageForceReScanDescription": "سيقوم بفحص جميع الملفات مرة أخرى كفحص جديد. سيتم فحص علامات ID3 لملفات الصوت وملفات OPF والملفات النصية كأنها جديدة.",
|
||||
"MessageImportantNotice": "إشعار هام!",
|
||||
"MessageInsertChapterBelow": "إدراج فصل أدناه",
|
||||
"MessageInvalidAsin": "ASIN غير صالح",
|
||||
"MessageItemsSelected": "تم تحديد {0} عنصرًا",
|
||||
"MessageItemsUpdated": "تم تحديث {0} عنصرًا",
|
||||
"MessageJoinUsOn": "انضم إلينا على",
|
||||
"MessageLoading": "جاري التحميل...",
|
||||
"MessageLoadingFolders": "جاري تحميل المجلدات...",
|
||||
"MessageLogsDescription": "يتم تخزين السجلات في <code>/metadata/logs</code> كملفات JSON. يتم تخزين سجلات الأعطال في <code>/metadata/logs/crash_logs.txt</code>.",
|
||||
"MessageM4BFailed": "فشل M4B!",
|
||||
"MessageM4BFinished": "تم الانتهاء من M4B!",
|
||||
"MessageMapChapterTitles": "ربط عناوين الفصول بفصول كتابك الصوتي الحالي دون تعديل الطوابع الزمنية",
|
||||
"MessageMarkAllEpisodesFinished": "تعليم جميع الحلقات على أنها منتهية",
|
||||
"MessageMarkAllEpisodesNotFinished": "تعليم جميع الحلقات على أنها غير منتهية",
|
||||
"MessageMarkAsFinished": "وضع علامة \"تم الإنتهاء\"",
|
||||
"MessageMarkAsNotFinished": "وضع علامة \"غير منته\"",
|
||||
"MessageMatchBooksDescription": "سيحاول مطابقة الكتب في المكتبة مع كتاب من مزود البحث المحدد وملء التفاصيل الفارغة وصورة الغلاف. لا يستبدل التفاصيل الموجودة.",
|
||||
"MessageNoAudioTracks": "لا توجد مسارات صوتية",
|
||||
"MessageNoAuthors": "لا يوجد مؤلفون",
|
||||
"MessageNoBackups": "لا توجد نسخ احتياطية",
|
||||
"MessageNoBookmarks": "لا توجد علامات مرجعية",
|
||||
"MessageNoChapters": "لا توجد فصول",
|
||||
"MessageNoCollections": "لا توجد مجموعات",
|
||||
"MessageNoCoversFound": "لم يتم العثور على أغلفة",
|
||||
"MessageNoDescription": "لا يوجد وصف",
|
||||
"MessageNoDevices": "لا توجد أجهزة",
|
||||
"MessageNoDownloadsInProgress": "لا توجد تنزيلات قيد التقدم حاليًا",
|
||||
"MessageNoDownloadsQueued": "لا توجد تنزيلات في قائمة الانتظار",
|
||||
"MessageNoEpisodeMatchesFound": "لم يتم العثور على أي تطابقات للحلقات",
|
||||
"MessageNoEpisodes": "لا توجد حلقات",
|
||||
"MessageNoFoldersAvailable": "لا توجد مجلدات متاحة",
|
||||
"MessageNoGenres": "لا توجد تصانيف",
|
||||
"MessageNoIssues": "لا توجد مشاكل",
|
||||
"MessageNoItems": "لا توجد عناصر",
|
||||
"MessageNoItemsFound": "لم يتم العثور على عناصر",
|
||||
"MessageNoListeningSessions": "لا توجد جلسات استماع",
|
||||
"MessageNoLogs": "لا توجد سجلات",
|
||||
"MessageNoMediaProgress": "لا يوجد تقدم للوسائط",
|
||||
"MessageNoNotifications": "لا توجد إشعارات",
|
||||
"MessageNoPodcastFeed": "بودكاست غير صالح: لا يوجد تغذية",
|
||||
"MessageNoPodcastsFound": "لم يتم العثور على أي بودكاست",
|
||||
"MessageNoResults": "لا توجد نتائج",
|
||||
"MessageNoSearchResultsFor": "لا توجد نتائج بحث عن \"{0}\"",
|
||||
"MessageNoSeries": "لا توجد مسلسلات",
|
||||
"MessageNoTags": "لا توجد علامات",
|
||||
"MessageNoTasksRunning": "لا توجد مهام قيد التشغيل",
|
||||
"MessageNoUpdatesWereNecessary": "لا حاجة لأي تحديثات",
|
||||
"MessageNoUserPlaylists": "ليس لديك أي قوائم تشغيل",
|
||||
"MessageNoUserPlaylistsHelp": "قوائم التشغيل خاصة. لا يمكن إلا للمستخدم الذي ينشئها رؤيتها.",
|
||||
"MessageNotYetImplemented": "لم يتم تنفيذه بعد",
|
||||
"MessageOpmlPreviewNote": "ملاحظة: هذه معاينة لملف OPML الذي تم تحليله. سيتم أخذ عنوان البودكاست الفعلي من خلاصة RSS.",
|
||||
"MessageOr": "أو",
|
||||
"MessagePauseChapter": "إيقاف تشغيل الفصل مؤقتًا",
|
||||
"MessagePlayChapter": "الاستماع إلى بداية الفصل",
|
||||
"MessagePlaylistCreateFromCollection": "إنشاء قائمة تشغيل من المجموعة",
|
||||
"MessagePleaseWait": "الرجاء الانتظار...",
|
||||
"MessagePodcastHasNoRSSFeedForMatching": "لا يحتوي البودكاست على عنوان URL لخلاصة RSS لاستخدامه في المطابقة",
|
||||
"MessagePodcastSearchField": "أدخل مصطلح البحث أو عنوان URL الخاص بتغذية RSS",
|
||||
"MessageQuickEmbedInProgress": "التضمين السريع قيد التقدم",
|
||||
"MessageQuickEmbedQueue": "تمت إضافته إلى قائمة انتظار التضمين السريع ({0} في قائمة الانتظار)",
|
||||
"MessageQuickMatchAllEpisodes": "مطابقة سريعة لجميع الحلقات",
|
||||
"MessageQuickMatchDescription": "املأ تفاصيل العنصر الفارغة والغلاف بأول نتيجة مطابقة من '{0}'. لا يستبدل التفاصيل إلا إذا تم تمكين إعداد الخادم 'تفضيل البيانات الوصفية المطابقة'.",
|
||||
"MessageRemoveChapter": "إزالة الفصل",
|
||||
"MessageRemoveEpisodes": "إزالة {0} حلقة (حلقات)",
|
||||
"MessageRemoveFromPlayerQueue": "إزالة من قائمة انتظار المشغل",
|
||||
"MessageRemoveUserWarning": "هل أنت متأكد أنك تريد حذف المستخدم \"{0}\" نهائيًا؟",
|
||||
"MessageReportBugsAndContribute": "أبلغ عن الأخطاء، واطلب الميزات، وساهم في",
|
||||
"MessageResetChaptersConfirm": "هل أنت متأكد أنك تريد إعادة تعيين الفصول والتراجع عن التغييرات التي أجريتها؟",
|
||||
"MessageRestoreBackupConfirm": "هل أنت متأكد أنك تريد استعادة النسخ الاحتياطي الذي تم إنشاؤه في",
|
||||
"MessageRestoreBackupWarning": "ستؤدي استعادة النسخ الاحتياطي إلى الكتابة فوق قاعدة البيانات بأكملها الموجودة في /config وصور الأغلفة في /metadata/items و /metadata/authors.<br /><br /> لا تعدل النسخ الاحتياطية أي ملفات في مجلدات مكتبتك. إذا قمت بتمكين إعدادات الخادم لتخزين صور الأغلفة والبيانات الوصفية في مجلدات مكتبتك، فلن يتم نسخها احتياطيًا أو الكتابة فوقها.<br /><br /> سيتم تحديث جميع العملاء الذين يستخدمون الخادم الخاص بك تلقائيًا.",
|
||||
"MessageScheduleLibraryScanNote": "بالنسبة لمعظم المستخدمين، يوصى بترك هذه الميزة معطلة وإبقاء إعداد مراقب المجلدات ممكّنًا. سيكتشف مراقب المجلدات تلقائيًا التغييرات في مجلدات مكتبتك. لا يعمل مراقب المجلدات مع كل نظام ملفات (مثل NFS)، لذا يمكن استخدام عمليات فحص المكتبة المجدولة بدلاً من ذلك.",
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "تشغيل كل {0} في الساعة {1}",
|
||||
"MessageSearchResultsFor": "نتائج البحث عن",
|
||||
"MessageSelected": "تم تحديد {0}",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "السلسلة المتعاقبة لا يمكن أن تحتوي على مسافات",
|
||||
"MessageServerCouldNotBeReached": "تعذر الوصول إلى الخادم",
|
||||
"MessageSetChaptersFromTracksDescription": "تعيين الفصول باستخدام كل ملف صوتي كفصل وعنوان الفصل كاسم الملف الصوتي",
|
||||
"MessageShareExpirationWillBe": "سيكون تاريخ الانتهاء <strong>{0}</strong>",
|
||||
"MessageShareExpiresIn": "ينتهي خلال {0}",
|
||||
"MessageShareURLWillBe": "سيكون رابط المشاركة هو <strong>{0}</strong>",
|
||||
"MessageStartPlaybackAtTime": "هل تريد بدء التشغيل لـ \"{0}\" في الساعة {1}؟",
|
||||
"MessageTaskAudioFileNotWritable": "الملف الصوتي \"{0}\" غير قابل للكتابة",
|
||||
"MessageTaskCanceledByUser": "تم إلغاء المهمة بواسطة المستخدم",
|
||||
"MessageTaskDownloadingEpisodeDescription": "جاري تنزيل الحلقة \"{0}\"",
|
||||
"MessageTaskEmbeddingMetadata": "جاري تضمين البيانات الوصفية",
|
||||
"MessageTaskEmbeddingMetadataDescription": "جاري تضمين البيانات الوصفية في الكتاب الصوتي \"{0}\"",
|
||||
"MessageTaskEncodingM4b": "جاري ترميز M4B",
|
||||
"MessageTaskEncodingM4bDescription": "جاري ترميز الكتاب الصوتي \"{0}\" في ملف m4b واحد",
|
||||
"MessageTaskFailed": "فشل",
|
||||
"MessageTaskFailedToBackupAudioFile": "فشل في نسخ الملف الصوتي \"{0}\" احتياطيًا",
|
||||
"MessageTaskFailedToCreateCacheDirectory": "فشل في إنشاء دليل ذاكرة التخزين المؤقت",
|
||||
"MessageTaskFailedToEmbedMetadataInFile": "فشل في تضمين البيانات الوصفية في الملف \"{0}\"",
|
||||
"MessageTaskFailedToMergeAudioFiles": "فشل في دمج الملفات الصوتية",
|
||||
"MessageTaskFailedToMoveM4bFile": "فشل في نقل ملف m4b",
|
||||
"MessageTaskFailedToWriteMetadataFile": "فشل في كتابة ملف البيانات الوصفية",
|
||||
"MessageTaskMatchingBooksInLibrary": "جارٍ مطابقة الكتب في المكتبة \"{0}\"",
|
||||
"MessageTaskNoFilesToScan": "لا توجد ملفات للفحص",
|
||||
"MessageTaskOpmlImport": "استيراد OPML",
|
||||
"MessageTaskOpmlImportDescription": "جارٍ إنشاء بودكاست من {0} خلاصة RSS",
|
||||
"MessageTaskOpmlImportFeed": "استيراد تغذية OPML",
|
||||
"MessageTaskOpmlImportFeedDescription": "جارٍ استيراد خلاصة RSS \"{0}\"",
|
||||
"MessageTaskOpmlImportFeedFailed": "فشل في الحصول على تغذية البودكاست",
|
||||
"MessageTaskOpmlImportFeedPodcastDescription": "جارٍ إنشاء بودكاست \"{0}\"",
|
||||
"MessageTaskOpmlImportFeedPodcastExists": "البودكاست موجود بالفعل في المسار",
|
||||
"MessageTaskOpmlImportFeedPodcastFailed": "فشل في إنشاء البودكاست",
|
||||
"MessageTaskOpmlImportFinished": "تمت إضافة {0} بودكاست",
|
||||
"MessageTaskOpmlParseFailed": "فشل في تحليل ملف OPML",
|
||||
"MessageTaskOpmlParseFastFail": "ملف OPML غير صالح، لم يتم العثور على علامة <opml> أو لم يتم العثور على علامة <outline>",
|
||||
"MessageTaskOpmlParseNoneFound": "لم يتم العثور على أي خلاصات في ملف OPML",
|
||||
"MessageTaskScanItemsAdded": "تمت إضافة {0}",
|
||||
"MessageTaskScanItemsMissing": "{0} مفقود",
|
||||
"MessageTaskScanItemsUpdated": "{0} تم تحديثه",
|
||||
"MessageTaskScanNoChangesNeeded": "لا توجد تغييرات مطلوبة",
|
||||
"MessageTaskScanningFileChanges": "جاري فحص تغييرات الملفات في \"{0}\"",
|
||||
"MessageTaskScanningLibrary": "جاري فحص مكتبة \"{0}\"",
|
||||
"MessageTaskTargetDirectoryNotWritable": "الدليل الهدف غير قابل للكتابة",
|
||||
"MessageThinking": "جاري التفكير...",
|
||||
"MessageUploaderItemFailed": "فشل الرفع",
|
||||
"MessageUploaderItemSuccess": "تم الرفع بنجاح!",
|
||||
"MessageUploading": "جاري الرفع...",
|
||||
"MessageValidCronExpression": "تعبير Cron صالح",
|
||||
"MessageWatcherIsDisabledGlobally": "المراقب معطل عالميًا في إعدادات الخادم",
|
||||
"MessageXLibraryIsEmpty": "مكتبة {0} فارغة!",
|
||||
"MessageYourAudiobookDurationIsLonger": "مدة كتابك الصوتي أطول من المدة التي تم العثور عليها",
|
||||
"MessageYourAudiobookDurationIsShorter": "مدة كتابك الصوتي أقصر من المدة التي تم العثور عليها",
|
||||
"NoteChangeRootPassword": "مستخدم الجذر هو المستخدم الوحيد الذي يمكن أن يكون لديه كلمة مرور فارغة",
|
||||
"NoteChapterEditorTimes": "ملاحظة: يجب أن يظل وقت بدء الفصل الأول عند 0:00 ولا يمكن أن يتجاوز وقت بدء الفصل الأخير مدة هذا الكتاب الصوتي.",
|
||||
"NoteFolderPicker": "ملاحظة: لن يتم عرض المجلدات التي تم تعيينها بالفعل",
|
||||
"NoteRSSFeedPodcastAppsHttps": "تحذير: تتطلب معظم تطبيقات البث الصوتي أن يكون عنوان URL الخاص بتغذية RSS يستخدم HTTPS",
|
||||
"NoteRSSFeedPodcastAppsPubDate": "تحذير: حلقة واحدة أو أكثر من حلقاتك ليس لها تاريخ نشر. بعض تطبيقات البودكاست تتطلب هذا.",
|
||||
"NoteUploaderFoldersWithMediaFiles": "سيتم التعامل مع المجلدات التي تحتوي على ملفات وسائط كعناصر مكتبة منفصلة.",
|
||||
"NoteUploaderOnlyAudioFiles": "في حالة رفع ملفات صوتية فقط، سيتم التعامل مع كل ملف صوتي ككتاب صوتي منفصل.",
|
||||
"NoteUploaderUnsupportedFiles": "يتم تجاهل الملفات غير المدعومة. عند اختيار مجلد أو إسقاطه، يتم تجاهل الملفات الأخرى التي ليست في مجلد عنصر.",
|
||||
"NotificationOnBackupCompletedDescription": "يتم تشغيله عند اكتمال النسخ الاحتياطي",
|
||||
"NotificationOnBackupFailedDescription": "يتم تشغيله عند فشل النسخ الاحتياطي",
|
||||
"NotificationOnEpisodeDownloadedDescription": "يتم تشغيله عند تنزيل حلقة بودكاست تلقائيًا",
|
||||
"NotificationOnTestDescription": "حدث لاختبار نظام الإشعارات",
|
||||
"PlaceholderNewCollection": "اسم المجموعة الجديدة",
|
||||
"PlaceholderNewFolderPath": "مسار المجلد الجديد",
|
||||
"PlaceholderNewPlaylist": "اسم قائمة التشغيل الجديدة",
|
||||
"PlaceholderSearch": "بحث..",
|
||||
"PlaceholderSearchEpisode": "بحث عن حلقة..",
|
||||
"StatsAuthorsAdded": "تمت إضافة مؤلفين",
|
||||
"StatsBooksAdded": "تمت إضافة كتب",
|
||||
"StatsBooksAdditional": "تتضمن بعض الإضافات…",
|
||||
"StatsBooksFinished": "كتب تم الانتهاء منها",
|
||||
"StatsBooksFinishedThisYear": "بعض الكتب التي تم الانتهاء منها هذا العام…",
|
||||
"StatsBooksListenedTo": "كتب تم الاستماع إليها",
|
||||
"StatsCollectionGrewTo": "نما مجموعتك من الكتب لتصبح…",
|
||||
"StatsSessions": "جلسات",
|
||||
"StatsSpentListening": "تم قضاء وقت في الاستماع",
|
||||
"StatsTopAuthor": "أفضل مؤلف",
|
||||
"StatsTopAuthors": "أفضل المؤلفين",
|
||||
"StatsTopGenre": "أفضل تصنيف",
|
||||
"StatsTopGenres": "أفضل التصنيفات",
|
||||
"StatsTopMonth": "أفضل شهر",
|
||||
"StatsTopNarrator": "أفضل راوي",
|
||||
"StatsTopNarrators": "أفضل الرواة",
|
||||
"StatsTotalDuration": "بإجمالي مدة…",
|
||||
"StatsYearInReview": "ملخص العام",
|
||||
"ToastAccountUpdateSuccess": "تم تحديث الحساب",
|
||||
"ToastAppriseUrlRequired": "يجب إدخال عنوان URL لـ Apprise",
|
||||
"ToastAsinRequired": "ASIN مطلوب",
|
||||
"ToastAuthorImageRemoveSuccess": "تمت إزالة صورة المؤلف",
|
||||
"ToastAuthorNotFound": "لم يتم العثور على المؤلف \"{0}\"",
|
||||
"ToastAuthorRemoveSuccess": "تمت إزالة المؤلف",
|
||||
"ToastAuthorSearchNotFound": "لم يتم العثور على المؤلف",
|
||||
"ToastAuthorUpdateMerged": "تم دمج المؤلف",
|
||||
"ToastAuthorUpdateSuccess": "تم تحديث المؤلف",
|
||||
"ToastAuthorUpdateSuccessNoImageFound": "تم تحديث المؤلف (لم يتم العثور على صورة)",
|
||||
"ToastBackupAppliedSuccess": "تم تطبيق النسخ الاحتياطي",
|
||||
"ToastBackupCreateFailed": "فشل إنشاء النسخ الاحتياطي",
|
||||
"ToastBackupCreateSuccess": "تم إنشاء النسخ الاحتياطي",
|
||||
"ToastBackupDeleteFailed": "فشل حذف النسخ الاحتياطي",
|
||||
"ToastBackupDeleteSuccess": "تم حذف النسخ الاحتياطي",
|
||||
"ToastBackupInvalidMaxKeep": "عدد غير صالح للنسخ الاحتياطية التي يجب الاحتفاظ بها",
|
||||
"ToastBackupInvalidMaxSize": "حجم أقصى غير صالح للنسخ الاحتياطي",
|
||||
"ToastBackupRestoreFailed": "فشل استعادة النسخ الاحتياطي",
|
||||
"ToastBackupUploadFailed": "فشل رفع النسخ الاحتياطي",
|
||||
"ToastBackupUploadSuccess": "تم رفع النسخ الاحتياطي",
|
||||
"ToastBatchApplyDetailsToItemsSuccess": "تم تطبيق التفاصيل على العناصر",
|
||||
"ToastBatchDeleteFailed": "فشل الحذف المجمّع",
|
||||
"ToastBatchDeleteSuccess": "نجاح الحذف المجمّع",
|
||||
"ToastBatchQuickMatchFailed": "فشلت المطابقة السريعة المجمّعة!",
|
||||
"ToastBatchQuickMatchStarted": "بدأت المطابقة السريعة المجمّعة لـ {0} كتابًا!",
|
||||
"ToastBatchUpdateFailed": "فشل التحديث المجمّع",
|
||||
"ToastBatchUpdateSuccess": "نجاح التحديث المجمّع",
|
||||
"ToastBookmarkCreateFailed": "فشل في إنشاء الإشارة المرجعية",
|
||||
"ToastBookmarkCreateSuccess": "تمت إضافة الإشارة المرجعية",
|
||||
"ToastBookmarkRemoveSuccess": "تمت إزالة الإشارة المرجعية",
|
||||
"ToastCachePurgeFailed": "فشل مسح ذاكرة التخزين المؤقت",
|
||||
"ToastCachePurgeSuccess": "تم مسح ذاكرة التخزين المؤقت بنجاح",
|
||||
"ToastChaptersHaveErrors": "الفصول تحتوي على أخطاء",
|
||||
"ToastChaptersInvalidShiftAmountLast": "مقدار إزاحة غير صالح. سيمتد وقت بدء الفصل الأخير إلى ما بعد مدة هذا الكتاب الصوتي.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "مقدار إزاحة غير صالح. سيكون للفصل الأول طول صفري أو سالب وسيتم الكتابة فوقه بواسطة الفصل الثاني. قم بزيادة مدة بدء الفصل الثاني.",
|
||||
"ToastChaptersMustHaveTitles": "يجب أن تحتوي الفصول على عناوين",
|
||||
"ToastChaptersRemoved": "تمت إزالة الفصول",
|
||||
"ToastChaptersUpdated": "تم تحديث الفصول",
|
||||
"ToastCollectionItemsAddFailed": "فشل إضافة عنصر (عناصر) إلى المجموعة",
|
||||
"ToastCollectionRemoveSuccess": "تمت إزالة المجموعة",
|
||||
"ToastCollectionUpdateSuccess": "تم تحديث المجموعة",
|
||||
"ToastCoverUpdateFailed": "فشل تحديث الغلاف",
|
||||
"ToastDateTimeInvalidOrIncomplete": "التاريخ والوقت غير صالحين أو غير مكتملين",
|
||||
"ToastDeleteFileFailed": "فشل حذف الملف",
|
||||
"ToastDeleteFileSuccess": "تم حذف الملف",
|
||||
"ToastDeviceAddFailed": "فشل إضافة الجهاز",
|
||||
"ToastDeviceNameAlreadyExists": "جهاز قارئ إلكتروني بهذا الاسم موجود بالفعل",
|
||||
"ToastDeviceTestEmailFailed": "فشل إرسال البريد الإلكتروني التجريبي",
|
||||
"ToastDeviceTestEmailSuccess": "تم إرسال البريد الإلكتروني التجريبي",
|
||||
"ToastEmailSettingsUpdateSuccess": "تم تحديث إعدادات البريد الإلكتروني",
|
||||
"ToastEncodeCancelFailed": "فشل إلغاء الترميز",
|
||||
"ToastEncodeCancelSucces": "تم إلغاء الترميز",
|
||||
"ToastEpisodeDownloadQueueClearFailed": "فشل مسح قائمة انتظار تنزيل الحلقات",
|
||||
"ToastEpisodeDownloadQueueClearSuccess": "تم مسح قائمة انتظار تنزيل الحلقات",
|
||||
"ToastEpisodeUpdateSuccess": "تم تحديث {0} حلقة",
|
||||
"ToastErrorCannotShare": "لا يمكن المشاركة محليًا على هذا الجهاز",
|
||||
"ToastFailedToLoadData": "فشل تحميل البيانات",
|
||||
"ToastFailedToMatch": "فشل التطابق",
|
||||
"ToastFailedToShare": "فشل المشاركة",
|
||||
"ToastFailedToUpdate": "فشل التحديث",
|
||||
"ToastInvalidImageUrl": "رابط صورة غير صالح",
|
||||
"ToastInvalidMaxEpisodesToDownload": "الحد الأقصى غير صالح لعدد الحلقات المراد تنزيلها",
|
||||
"ToastInvalidUrl": "رابط غير صالح",
|
||||
"ToastItemCoverUpdateSuccess": "تم تحديث غلاف العنصر",
|
||||
"ToastItemDeletedFailed": "فشل حذف العنصر",
|
||||
"ToastItemDeletedSuccess": "تم حذف العنصر",
|
||||
"ToastItemDetailsUpdateSuccess": "تم تحديث تفاصيل العنصر",
|
||||
"ToastItemMarkedAsFinishedFailed": "فشل وضع علامة \"مكتمل\"",
|
||||
"ToastItemMarkedAsFinishedSuccess": "تم وضع علامة \"تم الانتهاء\" على العنصر",
|
||||
"ToastItemMarkedAsNotFinishedFailed": "فشل وضع علامة \"غير مكتمل\"",
|
||||
"ToastItemMarkedAsNotFinishedSuccess": "تم وضع علامة \"غير مكتمل\" على العنصر",
|
||||
"ToastItemUpdateSuccess": "تم تحديث العنصر",
|
||||
"ToastLibraryCreateFailed": "فشل إنشاء المكتبة",
|
||||
"ToastLibraryCreateSuccess": "تم إنشاء المكتبة \"{0}\"",
|
||||
"ToastLibraryDeleteFailed": "فشل حذف المكتبة",
|
||||
"ToastLibraryDeleteSuccess": "تم حذف المكتبة",
|
||||
"ToastLibraryScanFailedToStart": "فشل بدء الفحص",
|
||||
"ToastLibraryScanStarted": "بدأ فحص المكتبة",
|
||||
"ToastLibraryUpdateSuccess": "تم تحديث المكتبة \"{0}\"",
|
||||
"ToastMatchAllAuthorsFailed": "فشل مطابقة جميع المؤلفين",
|
||||
"ToastMetadataFilesRemovedError": "حدث خطأ أثناء إزالة ملفات البيانات الوصفية. {0}",
|
||||
"ToastMetadataFilesRemovedNoneFound": "لا توجد بيانات وصفية. تم العثور على {0} ملف في المكتبة",
|
||||
"ToastMetadataFilesRemovedNoneRemoved": "لا توجد بيانات وصفية. تمت إزالة {0} ملفًا",
|
||||
"ToastMetadataFilesRemovedSuccess": "{0} بيانات وصفية. تم إزالة {1} ملف",
|
||||
"ToastMustHaveAtLeastOnePath": "يجب أن يكون هناك مسار واحد على الأقل",
|
||||
"ToastNameEmailRequired": "الاسم والبريد الإلكتروني مطلوبان",
|
||||
"ToastNameRequired": "الاسم مطلوب",
|
||||
"ToastNewEpisodesFound": "تم العثور على {0} حلقة جديدة",
|
||||
"ToastNewUserCreatedFailed": "فشل إنشاء الحساب: \"{0}\"",
|
||||
"ToastNewUserCreatedSuccess": "تم إنشاء حساب جديد",
|
||||
"ToastNewUserLibraryError": "يجب تحديد مكتبة واحدة على الأقل",
|
||||
"ToastNewUserPasswordError": "يجب أن يكون لديك كلمة مرور، يمكن لمستخدم الجذر فقط أن يكون لديه كلمة مرور فارغة",
|
||||
"ToastNewUserTagError": "يجب تحديد علامة واحدة على الأقل",
|
||||
"ToastNewUserUsernameError": "أدخل اسم مستخدم",
|
||||
"ToastNoNewEpisodesFound": "لم يتم العثور على حلقات جديدة",
|
||||
"ToastNoRSSFeed": "لا يحتوي البودكاست على خلاصة RSS",
|
||||
"ToastNoUpdatesNecessary": "لا توجد تحديثات ضرورية",
|
||||
"ToastNotificationCreateFailed": "فشل إنشاء الإشعار",
|
||||
"ToastNotificationDeleteFailed": "فشل حذف الإشعار",
|
||||
"ToastNotificationFailedMaximum": "يجب أن يكون الحد الأقصى للمحاولات الفاشلة >= 0",
|
||||
"ToastNotificationQueueMaximum": "يجب أن يكون الحد الأقصى لقائمة انتظار الإشعارات >= 0",
|
||||
"ToastNotificationSettingsUpdateSuccess": "تم تحديث إعدادات الإشعارات",
|
||||
"ToastNotificationTestTriggerFailed": "فشل تشغيل إشعار الاختبار",
|
||||
"ToastNotificationTestTriggerSuccess": "تم تشغيل إشعار الاختبار",
|
||||
"ToastNotificationUpdateSuccess": "تم تحديث الإشعار",
|
||||
"ToastPlaylistCreateFailed": "فشل إنشاء قائمة التشغيل",
|
||||
"ToastPlaylistCreateSuccess": "تم إنشاء قائمة التشغيل",
|
||||
"ToastPlaylistRemoveSuccess": "تمت إزالة قائمة التشغيل",
|
||||
"ToastPlaylistUpdateSuccess": "تم تحديث قائمة التشغيل",
|
||||
"ToastPodcastCreateFailed": "فشل إنشاء البودكاست",
|
||||
"ToastPodcastCreateSuccess": "تم إنشاء البودكاست بنجاح",
|
||||
"ToastPodcastGetFeedFailed": "فشل في الحصول على تغذية البودكاست",
|
||||
"ToastPodcastNoEpisodesInFeed": "لم يتم العثور على حلقات في خلاصة RSS",
|
||||
"ToastPodcastNoRssFeed": "لا يحتوي البودكاست على خلاصة RSS",
|
||||
"ToastProgressIsNotBeingSynced": "لا تتم مزامنة التقدم، أعد تشغيل التشغيل",
|
||||
"ToastProviderCreatedFailed": "فشل إضافة المزود",
|
||||
"ToastProviderCreatedSuccess": "تمت إضافة مزود جديد",
|
||||
"ToastProviderNameAndUrlRequired": "الاسم والرابط مطلوبان",
|
||||
"ToastProviderRemoveSuccess": "تمت إزالة المزود",
|
||||
"ToastRSSFeedCloseFailed": "فشل إغلاق مغذّي RSS",
|
||||
"ToastRSSFeedCloseSuccess": "تم إغلاق مغذّي RSS",
|
||||
"ToastRemoveFailed": "فشل الإزالة",
|
||||
"ToastRemoveItemFromCollectionFailed": "فشل إزالة العنصر من المجموعة",
|
||||
"ToastRemoveItemFromCollectionSuccess": "تمت إزالة العنصر من المجموعة",
|
||||
"ToastRemoveItemsWithIssuesFailed": "فشل إزالة عناصر المكتبة التي بها مشاكل",
|
||||
"ToastRemoveItemsWithIssuesSuccess": "تمت إزالة عناصر المكتبة التي بها مشاكل",
|
||||
"ToastRenameFailed": "فشل إعادة التسمية",
|
||||
"ToastRescanFailed": "فشل إعادة الفحص لـ {0}",
|
||||
"ToastRescanRemoved": "اكتملت إعادة الفحص، وتمت إزالة العنصر",
|
||||
"ToastRescanUpToDate": "اكتملت إعادة الفحص، العنصر كان محدثًا",
|
||||
"ToastRescanUpdated": "اكتملت إعادة الفحص، وتم تحديث العنصر",
|
||||
"ToastScanFailed": "فشل فحص عنصر المكتبة",
|
||||
"ToastSelectAtLeastOneUser": "حدد مستخدمًا واحدًا على الأقل",
|
||||
"ToastSendEbookToDeviceFailed": "فشل إرسال الكتاب الإلكتروني إلى الجهاز",
|
||||
"ToastSendEbookToDeviceSuccess": "تم إرسال الكتاب الإلكتروني إلى الجهاز \"{0}\"",
|
||||
"ToastSeriesSubmitFailedSameName": "لا يمكن إضافة سلسلتين بنفس الاسم",
|
||||
"ToastSeriesUpdateFailed": "فشل تحديث السلسلة",
|
||||
"ToastSeriesUpdateSuccess": "نجاح تحديث السلسلة",
|
||||
"ToastServerSettingsUpdateSuccess": "تم تحديث إعدادات الخادم",
|
||||
"ToastSessionCloseFailed": "فشل إغلاق الجلسة",
|
||||
"ToastSessionDeleteFailed": "فشل حذف الجلسة",
|
||||
"ToastSessionDeleteSuccess": "تم حذف الجلسة",
|
||||
"ToastSleepTimerDone": "انتهى مؤقت النوم... ششششش",
|
||||
"ToastSlugMustChange": "يحتوي الاسم التعريفي على أحرف غير صالحة",
|
||||
"ToastSlugRequired": "الاسم التعريفي مطلوب",
|
||||
"ToastSocketConnected": "تم الاتصال بالمقبس",
|
||||
"ToastSocketDisconnected": "تم قطع الاتصال بالمقبس",
|
||||
"ToastSocketFailedToConnect": "فشل الاتصال بالمقبس",
|
||||
"ToastSortingPrefixesEmptyError": "يجب أن يكون هناك بادئة فرز واحدة على الأقل",
|
||||
"ToastSortingPrefixesUpdateSuccess": "تم تحديث بادئات الفرز ({0} عنصرًا)",
|
||||
"ToastTitleRequired": "العنوان مطلوب",
|
||||
"ToastUnknownError": "خطأ غير معروف",
|
||||
"ToastUnlinkOpenIdFailed": "فشل فصل المستخدم عن OpenID",
|
||||
"ToastUnlinkOpenIdSuccess": "تم فصل المستخدم عن OpenID",
|
||||
"ToastUploaderFilepathExistsError": "مسار الملف \"{0}\" موجود بالفعل على الخادم",
|
||||
"ToastUploaderItemExistsInSubdirectoryError": "يستخدم العنصر \"{0}\" دليلًا فرعيًا لمسار الرفع.",
|
||||
"ToastUserDeleteFailed": "فشل حذف المستخدم",
|
||||
"ToastUserDeleteSuccess": "تم حذف المستخدم",
|
||||
"ToastUserPasswordChangeSuccess": "تم تغيير كلمة المرور بنجاح",
|
||||
"ToastUserPasswordMismatch": "كلمات المرور غير متطابقة",
|
||||
"ToastUserPasswordMustChange": "يجب ألا تطابق كلمة المرور الجديدة كلمة المرور القديمة",
|
||||
"ToastUserRootRequireName": "يجب إدخال اسم مستخدم الجذر"
|
||||
}
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Плейлист",
|
||||
"HeaderPlaylistItems": "Елементи от плейлист",
|
||||
"HeaderPodcastsToAdd": "Подкасти за Добавяне",
|
||||
"HeaderPresets": "Настройки по подразбиране",
|
||||
"HeaderPreviewCover": "Преглед на Корица",
|
||||
"HeaderRSSFeedGeneral": "RSS подробности",
|
||||
"HeaderRSSFeedIsOpen": "RSS емисията е отворена",
|
||||
@@ -219,6 +220,7 @@
|
||||
"LabelAccountTypeAdmin": "Администратор",
|
||||
"LabelAccountTypeGuest": "Гост",
|
||||
"LabelAccountTypeUser": "Потребител",
|
||||
"LabelActivities": "Дейности",
|
||||
"LabelActivity": "Дейност",
|
||||
"LabelAddToCollection": "Добави в Колекция",
|
||||
"LabelAddToCollectionBatch": "Добави {0} Книги в Колекция",
|
||||
@@ -253,7 +255,7 @@
|
||||
"LabelBackupLocation": "Местоположение на Архив",
|
||||
"LabelBackupsEnableAutomaticBackups": "Включи автоматично архивиране",
|
||||
"LabelBackupsEnableAutomaticBackupsHelp": "Архиви запазени в /metadata/backups",
|
||||
"LabelBackupsMaxBackupSize": "Максимален размер на архива (в GB)",
|
||||
"LabelBackupsMaxBackupSize": "Максимален размер на архива (в GB) (0 за неограничен)",
|
||||
"LabelBackupsMaxBackupSizeHelp": "За защита срещу грешки в конфигурацията, архивите ще се провалят ако надхвърлят конфигурирания размер.",
|
||||
"LabelBackupsNumberToKeep": "Брой архиви за запазване",
|
||||
"LabelBackupsNumberToKeepHelp": "Само 1 архив ще бъде премахнат на веднъж, така че ако вече имате повече архиви от това трябва да ги премахнете ръчно.",
|
||||
@@ -283,6 +285,7 @@
|
||||
"LabelContinueSeries": "Продължи серии",
|
||||
"LabelCover": "Корица",
|
||||
"LabelCoverImageURL": "URL на Корица",
|
||||
"LabelCoverProvider": "Източник за обложки",
|
||||
"LabelCreatedAt": "Създадено на",
|
||||
"LabelCronExpression": "Cron израз",
|
||||
"LabelCurrent": "Текущо",
|
||||
@@ -325,11 +328,20 @@
|
||||
"LabelEncodingClearItemCache": "Уверете се, че периодично изчиствате кеша на елементите.",
|
||||
"LabelEncodingFinishedM4B": "Завършеният M4B файл ще бъде поставен в папката на вашите аудиокниги на:",
|
||||
"LabelEncodingInfoEmbedded": "Метаданните ще бъдат вградени в аудио траковете в папката на вашите аудиокниги.",
|
||||
"LabelEncodingStartedNavigation": "Когато задачата е стартирана, можете да смените тази страница.",
|
||||
"LabelEncodingTimeWarning": "Кодирането може да отнеме до 30 минути.",
|
||||
"LabelEncodingWarningAdvancedSettings": "Внимание: Не променяйте тези настройки, ако не сте запознати с ffmpeg настройките за кодиране.",
|
||||
"LabelEncodingWatcherDisabled": "Ако сте изключили наблюдението на папки, ще е нужно да сканирате повторно аудио книгата.",
|
||||
"LabelEnd": "Край",
|
||||
"LabelEndOfChapter": "Край на глава",
|
||||
"LabelEpisode": "Епизод",
|
||||
"LabelEpisodeNotLinkedToRssFeed": "Епизодът не е свързан с RSS канал",
|
||||
"LabelEpisodeNumber": "Епизод #{0}",
|
||||
"LabelEpisodeTitle": "Заглавие на Епизод",
|
||||
"LabelEpisodeType": "Тип на Епизод",
|
||||
"LabelEpisodeUrlFromRssFeed": "URL адрес на епизод от RSS канал",
|
||||
"LabelEpisodes": "Епизоди",
|
||||
"LabelEpisodic": "Епизодичен",
|
||||
"LabelExample": "Пример",
|
||||
"LabelExpandSeries": "Покажи сериите",
|
||||
"LabelExpandSubSeries": "Покажи съб сериите",
|
||||
@@ -341,7 +353,9 @@
|
||||
"LabelFetchingMetadata": "Взимане на Метаданни",
|
||||
"LabelFile": "Файл",
|
||||
"LabelFileBirthtime": "Дата на създаване на файла",
|
||||
"LabelFileBornDate": "Роден {0}",
|
||||
"LabelFileModified": "Дата на модификация на файла",
|
||||
"LabelFileModifiedDate": "Променен {0}",
|
||||
"LabelFilename": "Име на файла",
|
||||
"LabelFilterByUser": "Филтриране по Потребител",
|
||||
"LabelFindEpisodes": "Намери Епизоди",
|
||||
@@ -355,14 +369,17 @@
|
||||
"LabelFontScale": "Мащаб на шрифта",
|
||||
"LabelFontStrikethrough": "Зачертан",
|
||||
"LabelFormat": "Формат",
|
||||
"LabelFull": "Пълен",
|
||||
"LabelGenre": "Жанр",
|
||||
"LabelGenres": "Жанрове",
|
||||
"LabelHardDeleteFile": "Пълно Изтриване на Файл",
|
||||
"LabelHasEbook": "Има е-книга",
|
||||
"LabelHasSupplementaryEbook": "Има допълнителна е-книга",
|
||||
"LabelHideSubtitles": "Скрий субтитри",
|
||||
"LabelHighestPriority": "Най-висок Приоритет",
|
||||
"LabelHost": "Хост",
|
||||
"LabelHour": "Час",
|
||||
"LabelHours": "Часа",
|
||||
"LabelIcon": "Икона",
|
||||
"LabelImageURLFromTheWeb": "URL на Изображение от Интернет",
|
||||
"LabelInProgress": "В процес на изпълнение",
|
||||
@@ -377,8 +394,11 @@
|
||||
"LabelIntervalEvery6Hours": "Всеки 6 часа",
|
||||
"LabelIntervalEveryDay": "Всеки ден",
|
||||
"LabelIntervalEveryHour": "Всеки час",
|
||||
"LabelIntervalEveryMinute": "Всяка минута",
|
||||
"LabelInvert": "Обърни",
|
||||
"LabelItem": "Елемент",
|
||||
"LabelJumpBackwardAmount": "Количество за прескачане назад",
|
||||
"LabelJumpForwardAmount": "Количество за прескачане напред",
|
||||
"LabelLanguage": "Език",
|
||||
"LabelLanguageDefaultServer": "Език по подразбиране на сървъра",
|
||||
"LabelLanguages": "Езици",
|
||||
@@ -393,6 +413,7 @@
|
||||
"LabelLess": "По-малко",
|
||||
"LabelLibrariesAccessibleToUser": "Библиотеки Достъпни за Потребителя",
|
||||
"LabelLibrary": "Библиотека",
|
||||
"LabelLibraryFilterSublistEmpty": "Не {0}",
|
||||
"LabelLibraryItem": "Елемент на Библиотека",
|
||||
"LabelLibraryName": "Име на Библиотека",
|
||||
"LabelLimit": "Лимит",
|
||||
@@ -405,6 +426,10 @@
|
||||
"LabelLowestPriority": "Най-нисък Приоритет",
|
||||
"LabelMatchExistingUsersBy": "Съпостави съществуващи потребители по",
|
||||
"LabelMatchExistingUsersByDescription": "Използва се за свързване на съществуващи потребители. След свързване потребителите ще бъдат съпоставени по уникален идентификатор от вашия доставчик на SSO",
|
||||
"LabelMaxEpisodesToDownload": "Максимален брой епизоди за сваляне. Използвай 0 за неограничен.",
|
||||
"LabelMaxEpisodesToDownloadPerCheck": "Максимален брой нови епизоди за сваляне за проверка",
|
||||
"LabelMaxEpisodesToKeep": "Максимален брой епизоди за запазване",
|
||||
"LabelMaxEpisodesToKeepHelp": "Стойност 0 указва без максимален лимит. След като нов епизод е автоматично свален, най-старият епизод ще бъде изтрит, ако имате повече от X епизода. Само по един епизод ще бъде изтриван за всеки нов свален такъв.",
|
||||
"LabelMediaPlayer": "Медия Плейър",
|
||||
"LabelMediaType": "Тип медия",
|
||||
"LabelMetaTag": "Мета Таг",
|
||||
@@ -412,6 +437,7 @@
|
||||
"LabelMetadataOrderOfPrecedenceDescription": "По-високите източници на метаданни ще заменят по-ниските",
|
||||
"LabelMetadataProvider": "Доставчик на Метаданни",
|
||||
"LabelMinute": "Минута",
|
||||
"LabelMinutes": "Минути",
|
||||
"LabelMissing": "Липсващо",
|
||||
"LabelMissingEbook": "Няма електронна книга",
|
||||
"LabelMissingSupplementaryEbook": "Няма допълнителна електронна книга",
|
||||
@@ -449,11 +475,14 @@
|
||||
"LabelOpenIDGroupClaimDescription": "Име на OpenID твърдението, което съдържа списък с групите на потребителя. Обикновено се нарича <code>groups</code>. <b>Ако е конфигурирано</b>, приложението автоматично ще присвоява роли въз основа на членството на потребителя в групи, при условие че тези групи са наименувани без чувствителност към регистъра като 'admin', 'user' или 'guest' в твърдението. Твърдението трябва да съдържа списък и ако потребителят принадлежи към множество групи, приложението ще присвои ролята, съответстваща на най-високото ниво на достъп. Ако няма съвпадение с група, достъпът ще бъде отказан.",
|
||||
"LabelOpenRSSFeed": "Отвори RSS Feed",
|
||||
"LabelOverwrite": "Презапиши",
|
||||
"LabelPaginationPageXOfY": "Страница {0} от {1}",
|
||||
"LabelPassword": "Парола",
|
||||
"LabelPath": "Път",
|
||||
"LabelPermanent": "Постоянен",
|
||||
"LabelPermissionsAccessAllLibraries": "Може да достъпи до всички библиотеки",
|
||||
"LabelPermissionsAccessAllTags": "Може да достъпи всички тагове",
|
||||
"LabelPermissionsAccessExplicitContent": "Може да достъпи експлицитно съдържание",
|
||||
"LabelPermissionsCreateEreader": "Може да създава електронен четец",
|
||||
"LabelPermissionsDelete": "Може да трие",
|
||||
"LabelPermissionsDownload": "Може да сваля",
|
||||
"LabelPermissionsUpdate": "Може да обновява",
|
||||
@@ -461,6 +490,8 @@
|
||||
"LabelPersonalYearReview": "Преглед на годината Ви ({0})",
|
||||
"LabelPhotoPathURL": "Път/URL на Снимка",
|
||||
"LabelPlayMethod": "Метод на Пускане",
|
||||
"LabelPlaybackRateIncrementDecrement": "Размер на увеличаване/намаляне при скоростта на възпроизвеждане",
|
||||
"LabelPlayerChapterNumberMarker": "{0} от {1}",
|
||||
"LabelPlaylists": "Плейлисти",
|
||||
"LabelPodcast": "Подкаст",
|
||||
"LabelPodcastSearchRegion": "Регион за Търсене на Подкасти",
|
||||
@@ -472,9 +503,12 @@
|
||||
"LabelPrimaryEbook": "Основна Електронна Книга",
|
||||
"LabelProgress": "Прогрес",
|
||||
"LabelProvider": "Доставчик",
|
||||
"LabelProviderAuthorizationValue": "Стойност на Authorization Header",
|
||||
"LabelPubDate": "Дата на публикуване",
|
||||
"LabelPublishYear": "Година на публикуване",
|
||||
"LabelPublishedDate": "Публикувани {0}",
|
||||
"LabelPublishedDecade": "Десетилетие на публикуване",
|
||||
"LabelPublishedDecades": "Десетилетия на публикуване",
|
||||
"LabelPublisher": "Издател",
|
||||
"LabelPublishers": "Издателство",
|
||||
"LabelRSSFeedCustomOwnerEmail": "Персонализиран имейл на собственика",
|
||||
@@ -484,6 +518,7 @@
|
||||
"LabelRSSFeedSlug": "идентификатор на RSS емисия",
|
||||
"LabelRSSFeedURL": "URL на RSS емисия",
|
||||
"LabelRandomly": "Случайно",
|
||||
"LabelReAddSeriesToContinueListening": "Добави отново в \"Продължете да слушате\"",
|
||||
"LabelRead": "Прочети",
|
||||
"LabelReadAgain": "Прочети отново",
|
||||
"LabelReadEbookWithoutProgress": "Прочети електронната книга без записване прогрес",
|
||||
@@ -493,29 +528,40 @@
|
||||
"LabelRedo": "Повтори",
|
||||
"LabelRegion": "Регион",
|
||||
"LabelReleaseDate": "Дата на Издаване",
|
||||
"LabelRemoveAllMetadataAbs": "Премахни всички metadata.abs файлове",
|
||||
"LabelRemoveAllMetadataJson": "Премахни всички metadata.json файлове",
|
||||
"LabelRemoveAudibleBranding": "Премахни въведението и заключението на Audible от главите",
|
||||
"LabelRemoveCover": "Премахни Корица",
|
||||
"LabelRemoveMetadataFile": "Премахни файловете с метаданни от папката на библиотеката",
|
||||
"LabelRemoveMetadataFileHelp": "Премахни всички metadata.json и metadata.abs файлове от вашата {0} папка.",
|
||||
"LabelRowsPerPage": "Редове на Страница",
|
||||
"LabelSearchTerm": "Търси Термин",
|
||||
"LabelSearchTitle": "Търси Заглавие",
|
||||
"LabelSearchTitleOrASIN": "Търси Заглавие или ASIN",
|
||||
"LabelSeason": "Сезон",
|
||||
"LabelSeasonNumber": "Сезон #{0}",
|
||||
"LabelSelectAll": "Избери всичко",
|
||||
"LabelSelectAllEpisodes": "Избери всички епизоди",
|
||||
"LabelSelectEpisodesShowing": "Избери {0} епизоди показани",
|
||||
"LabelSelectUsers": "Избери Потребители",
|
||||
"LabelSendEbookToDevice": "Изпрати електронна книга до ...",
|
||||
"LabelSequence": "Последователност",
|
||||
"LabelSerial": "Сериал",
|
||||
"LabelSeries": "От сериите",
|
||||
"LabelSeriesName": "Име на Серия",
|
||||
"LabelSeriesProgress": "Прогрес на Серия",
|
||||
"LabelServerLogLevel": "Ниво на сървърен журнал",
|
||||
"LabelServerYearReview": "Преглед на годината на сървъра ({0})",
|
||||
"LabelSetEbookAsPrimary": "Направи главен",
|
||||
"LabelSetEbookAsSupplementary": "Направи второстепенен",
|
||||
"LabelSettingsAllowIframe": "Разреши вграждане в iframe",
|
||||
"LabelSettingsAudiobooksOnly": "Само аудиокниги",
|
||||
"LabelSettingsAudiobooksOnlyHelp": "Активирането на тази настройка ще игнорира файловете на електронни книги, освен ако не са в папка с аудиокниги, в което случай ще бъдат зададени като допълнителни електронни книги",
|
||||
"LabelSettingsBookshelfViewHelp": "Скеуморфен дизайн с дървени рафтове",
|
||||
"LabelSettingsChromecastSupport": "Chromecast поддръжка",
|
||||
"LabelSettingsDateFormat": "Формат на Дата",
|
||||
"LabelSettingsEnableWatcher": "Автоматично сканиране на библиотеките за промени",
|
||||
"LabelSettingsEnableWatcherForLibrary": "Автоматично сканиране на библиотеката за промени",
|
||||
"LabelSettingsEnableWatcherHelp": "Включва автоматичното добавяне/обновяване на елементи, когато се открият промени във файловете. *Изисква рестарт на сървъра",
|
||||
"LabelSettingsEpubsAllowScriptedContent": "Позволи скриптово съдържание в epub-и",
|
||||
"LabelSettingsEpubsAllowScriptedContentHelp": "Позволи epub файловете да изпълняват скриптове. Препоръчително е да бъде изключено освен ако не се доверявате на източника на epub файловете.",
|
||||
@@ -527,10 +573,13 @@
|
||||
"LabelSettingsHideSingleBookSeriesHelp": "Сериите с една книга ще бъдат скрити от страницата на серията и рафтовете на началната страница.",
|
||||
"LabelSettingsHomePageBookshelfView": "Начална страница изглед на рафт",
|
||||
"LabelSettingsLibraryBookshelfView": "Библиотека изглед на рафт",
|
||||
"LabelSettingsLibraryMarkAsFinishedPercentComplete": "Процент завършеност е по-голям от",
|
||||
"LabelSettingsLibraryMarkAsFinishedTimeRemaining": "Оставащо време е по-малко от (секунди)",
|
||||
"LabelSettingsLibraryMarkAsFinishedWhen": "Отбелязване на мултимедиен елемент като завършен когато",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeries": "Пропусни предишни книги в Продължи Поредица",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeriesHelp": "Рафтът на началната страница 'Продължи поредицата' показва първата книга, която не е започната в поредици, в които има поне една завършена книга и няма книги в процес на четене. Активирането на тази настройка ще продължи поредицата от най-далечната завършена книга вместо от първата незапочната книга.",
|
||||
"LabelSettingsParseSubtitles": "Извлечи подзаглавия",
|
||||
"LabelSettingsParseSubtitlesHelp": "Извлича подзаглавия от имената на папките на аудиокнигите.<br>Подзаглавията трябва да бъдат разделени с \" - \"<br>например \"Заглавие на Книга - Тук е Подзаглавито\" има подзаглавие \"Тук е Подзаглавито\"",
|
||||
"LabelSettingsParseSubtitlesHelp": "Извлича подзаглавия от имената на папките на аудио книгите.<br>Подзаглавията трябва да бъдат разделени с \" - \"<br>например \"Заглавие на Книга - Тук е подзаглавието\" има подзаглавие \"Тук е подзаглавието\"",
|
||||
"LabelSettingsPreferMatchedMetadata": "Предпочети съвпадащи метаданни",
|
||||
"LabelSettingsPreferMatchedMetadataHelp": "Съвпадащите данни ще заменят детайлите на елемента при използване на Бързо Съпоставяне. По подразбиране Бързото Съпоставяне ще попълни само липсващите детайли.",
|
||||
"LabelSettingsSkipMatchingBooksWithASIN": "Пропусни съвпадащи книги, които вече имат ASIN",
|
||||
@@ -544,11 +593,19 @@
|
||||
"LabelSettingsStoreMetadataWithItem": "Запази метаданните с елемента",
|
||||
"LabelSettingsStoreMetadataWithItemHelp": "По подразбиране метаданните се съхраняват в /metadata/items, като активирате тази настройка метаданните ще се съхраняват в папката на елемента на вашата библиотека",
|
||||
"LabelSettingsTimeFormat": "Формат на Време",
|
||||
"LabelShare": "Сподели",
|
||||
"LabelShareDownloadableHelp": "Разреши на потребителите през връзка за споделяне да свалят zip файл с мултимедийния елемент.",
|
||||
"LabelShareOpen": "Общодостъпно",
|
||||
"LabelShareURL": "URL за споделяне",
|
||||
"LabelShowAll": "Покажи всички",
|
||||
"LabelShowSeconds": "Покажи секунди",
|
||||
"LabelShowSubtitles": "Показвай подзаглавия",
|
||||
"LabelSize": "Размер",
|
||||
"LabelSleepTimer": "Таймер за изключване",
|
||||
"LabelSlug": "Слъг",
|
||||
"LabelSortAscending": "Възходящ",
|
||||
"LabelSortDescending": "Низходящ",
|
||||
"LabelSortPubDate": "Подреди по дата на публикуване",
|
||||
"LabelStart": "Старт",
|
||||
"LabelStartTime": "Начално Време",
|
||||
"LabelStarted": "Стартирано",
|
||||
@@ -583,6 +640,11 @@
|
||||
"LabelThemeDark": "Тъмна",
|
||||
"LabelThemeLight": "Светла",
|
||||
"LabelTimeBase": "Времева Основа",
|
||||
"LabelTimeDurationXHours": "{0} часа",
|
||||
"LabelTimeDurationXMinutes": "{0} минути",
|
||||
"LabelTimeDurationXSeconds": "{0} секунди",
|
||||
"LabelTimeInMinutes": "Време в минути",
|
||||
"LabelTimeLeft": "остава {0}",
|
||||
"LabelTimeListened": "Време Слушано",
|
||||
"LabelTimeListenedToday": "Време Слушано Днес",
|
||||
"LabelTimeRemaining": "{0} оставащи",
|
||||
@@ -590,6 +652,7 @@
|
||||
"LabelTitle": "Заглавие",
|
||||
"LabelToolsEmbedMetadata": "Вграждане на Метаданни",
|
||||
"LabelToolsEmbedMetadataDescription": "Вграждане на метаданни в аудио файлове, включително корица и глави.",
|
||||
"LabelToolsM4bEncoder": "M4B кодировчик",
|
||||
"LabelToolsMakeM4b": "Направи M4B Аудиокнига Файл",
|
||||
"LabelToolsMakeM4bDescription": "Генериране на .M4B аудиокнига файл с вградени метаданни, корица и глави.",
|
||||
"LabelToolsSplitM4b": "Раздели M4B на MP3-ки",
|
||||
@@ -602,26 +665,32 @@
|
||||
"LabelTracksMultiTrack": "Многоканален",
|
||||
"LabelTracksNone": "Няма канали",
|
||||
"LabelTracksSingleTrack": "Единичен канал",
|
||||
"LabelTrailer": "Трейлър",
|
||||
"LabelType": "Тип",
|
||||
"LabelUnabridged": "Несъкратен",
|
||||
"LabelUndo": "Отмени",
|
||||
"LabelUnknown": "Неизвестен",
|
||||
"LabelUnknownPublishDate": "Неизвестна дата на публикуване",
|
||||
"LabelUpdateCover": "Обнови Корица",
|
||||
"LabelUpdateCoverHelp": "Позволи презаписване на съществуващите корици за избраните книги, когато се намери съвпадение",
|
||||
"LabelUpdateDetails": "Обнови Детайли",
|
||||
"LabelUpdateDetailsHelp": "Позволи презаписване на съществуващите детайли за избраните книги, когато се намери съвпадение",
|
||||
"LabelUpdatedAt": "Обновено на",
|
||||
"LabelUploaderDragAndDrop": "Плъзни и Пусни Файлове или Папки",
|
||||
"LabelUploaderDragAndDropFilesOnly": "Извлачване на файлове",
|
||||
"LabelUploaderDropFiles": "Пусни Файлове",
|
||||
"LabelUploaderItemFetchMetadataHelp": "Автоматично вземи заглавие, автор и серия",
|
||||
"LabelUseAdvancedOptions": "Използвай разширени опции",
|
||||
"LabelUseChapterTrack": "Използвай канал за глава",
|
||||
"LabelUseFullTrack": "Използвай пълен канал",
|
||||
"LabelUseZeroForUnlimited": "Използвай 0 за неограничен",
|
||||
"LabelUser": "Потребител",
|
||||
"LabelUsername": "Потребителско име",
|
||||
"LabelValue": "Стойност",
|
||||
"LabelVersion": "Версия",
|
||||
"LabelViewBookmarks": "Виж Отметки",
|
||||
"LabelViewChapters": "Виж Глави",
|
||||
"LabelViewPlayerSettings": "Виж настройки на плеъра",
|
||||
"LabelViewQueue": "Виж Опашка",
|
||||
"LabelVolume": "Сила на Звука",
|
||||
"LabelWeekdaysToRun": "Делници за изпълнение",
|
||||
|
||||
@@ -1,35 +1,35 @@
|
||||
{
|
||||
"ButtonAdd": "Afegeix",
|
||||
"ButtonAddChapters": "Afegeix",
|
||||
"ButtonAddDevice": "Afegeix Dispositiu",
|
||||
"ButtonAddLibrary": "Crea Biblioteca",
|
||||
"ButtonAddChapters": "Afegeix capítols",
|
||||
"ButtonAddDevice": "Afegeix un aparell",
|
||||
"ButtonAddLibrary": "Afegeix una biblioteca",
|
||||
"ButtonAddPodcasts": "Afegeix pòdcasts",
|
||||
"ButtonAddUser": "Crea Usuari",
|
||||
"ButtonAddYourFirstLibrary": "Crea la teva Primera Biblioteca",
|
||||
"ButtonAddUser": "Afegeix un usuari",
|
||||
"ButtonAddYourFirstLibrary": "Afegiu la vostra primera biblioteca",
|
||||
"ButtonApply": "Aplica",
|
||||
"ButtonApplyChapters": "Aplica Capítols",
|
||||
"ButtonApplyChapters": "Aplica capítols",
|
||||
"ButtonAuthors": "Autors",
|
||||
"ButtonBack": "Enrere",
|
||||
"ButtonBatchEditPopulateFromExisting": "Omplir des d'existent",
|
||||
"ButtonBatchEditPopulateMapDetails": "Omplir detalls del mapa",
|
||||
"ButtonBrowseForFolder": "Cerca Carpeta",
|
||||
"ButtonBatchEditPopulateMapDetails": "Omple els detalls del mapa",
|
||||
"ButtonBrowseForFolder": "Cerca una carpeta",
|
||||
"ButtonCancel": "Cancel·la",
|
||||
"ButtonCancelEncode": "Cancel·la Codificador",
|
||||
"ButtonCancelEncode": "Cancel·la la codificació",
|
||||
"ButtonChangeRootPassword": "Canvia Contrasenya Root",
|
||||
"ButtonCheckAndDownloadNewEpisodes": "Verifica i Descarrega Nous Episodis",
|
||||
"ButtonChooseAFolder": "Tria una Carpeta",
|
||||
"ButtonChooseFiles": "Tria un Fitxer",
|
||||
"ButtonClearFilter": "Elimina Filtres",
|
||||
"ButtonCloseFeed": "Tanca Font",
|
||||
"ButtonChooseAFolder": "Trieu una carpeta",
|
||||
"ButtonChooseFiles": "Trieu fitxers",
|
||||
"ButtonClearFilter": "Neteja el filtre",
|
||||
"ButtonCloseFeed": "Tanca el canal",
|
||||
"ButtonCloseSession": "Tanca la sessió oberta",
|
||||
"ButtonCollections": "Col·leccions",
|
||||
"ButtonConfigureScanner": "Configura Escàner",
|
||||
"ButtonCreate": "Crea",
|
||||
"ButtonCreateBackup": "Crea Còpia de Seguretat",
|
||||
"ButtonDelete": "Elimina",
|
||||
"ButtonDelete": "Suprimeix",
|
||||
"ButtonDownloadQueue": "Cua",
|
||||
"ButtonEdit": "Edita",
|
||||
"ButtonEditChapters": "Edita Capítol",
|
||||
"ButtonEditChapters": "Edita capítols",
|
||||
"ButtonEditPodcast": "Edita el pòdcast",
|
||||
"ButtonEnable": "Habilita",
|
||||
"ButtonFireAndFail": "Executat i fallat",
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Llista de Reproducció",
|
||||
"HeaderPlaylistItems": "Elements de la Llista de Reproducció",
|
||||
"HeaderPodcastsToAdd": "Pòdcasts a afegir",
|
||||
"HeaderPresets": "Valors predefinits",
|
||||
"HeaderPreviewCover": "Previsualització de la Portada",
|
||||
"HeaderRSSFeedGeneral": "Detalls RSS",
|
||||
"HeaderRSSFeedIsOpen": "La Font RSS està oberta",
|
||||
@@ -192,7 +193,7 @@
|
||||
"HeaderSettings": "Paràmetres",
|
||||
"HeaderSettingsDisplay": "Interfície",
|
||||
"HeaderSettingsExperimental": "Funcionalitats experimentals",
|
||||
"HeaderSettingsGeneral": "General",
|
||||
"HeaderSettingsGeneral": "Generals",
|
||||
"HeaderSettingsScanner": "Escàner",
|
||||
"HeaderSettingsWebClient": "Client web",
|
||||
"HeaderSleepTimer": "Temporitzador de son",
|
||||
@@ -221,10 +222,10 @@
|
||||
"LabelAccountTypeUser": "Usuari",
|
||||
"LabelActivities": "Activitats",
|
||||
"LabelActivity": "Activitat",
|
||||
"LabelAddToCollection": "Afegit a la Col·lecció",
|
||||
"LabelAddToCollectionBatch": "S'han Afegit {0} Llibres a la Col·lecció",
|
||||
"LabelAddToPlaylist": "Afegit a la llista de reproducció",
|
||||
"LabelAddToPlaylistBatch": "S'han Afegit {0} Elements a la Llista de Reproducció",
|
||||
"LabelAddToCollection": "Afegeix a la col·lecció",
|
||||
"LabelAddToCollectionBatch": "Afegeix {0} llibres a la col·lecció",
|
||||
"LabelAddToPlaylist": "Afegeix a la llista de reproducció",
|
||||
"LabelAddToPlaylistBatch": "Afegeix {0} elements a la llista de reproducció",
|
||||
"LabelAddedAt": "Afegit",
|
||||
"LabelAddedDate": "{0} Afegit",
|
||||
"LabelAdminUsersOnly": "Només usuaris administradors",
|
||||
@@ -233,7 +234,7 @@
|
||||
"LabelAllUsers": "Tots els usuaris",
|
||||
"LabelAllUsersExcludingGuests": "Tots els usuaris excepte convidats",
|
||||
"LabelAllUsersIncludingGuests": "Tots els usuaris i convidats",
|
||||
"LabelAlreadyInYourLibrary": "Ja existeix a la Biblioteca",
|
||||
"LabelAlreadyInYourLibrary": "Ja existeix a la biblioteca",
|
||||
"LabelApiToken": "Testimoni de l'API",
|
||||
"LabelAppend": "Adjuntar",
|
||||
"LabelAudioBitrate": "Taxa de bits d'àudio (per exemple, 128k)",
|
||||
@@ -290,14 +291,14 @@
|
||||
"LabelCronExpression": "Expressió de Cron",
|
||||
"LabelCurrent": "Actual",
|
||||
"LabelCurrently": "En aquest moment:",
|
||||
"LabelCustomCronExpression": "Expressió de Cron Personalitzada:",
|
||||
"LabelDatetime": "Hora i Data",
|
||||
"LabelCustomCronExpression": "Expressió del Cron personalitzada:",
|
||||
"LabelDatetime": "Data i hora",
|
||||
"LabelDays": "Dies",
|
||||
"LabelDeleteFromFileSystemCheckbox": "Suprimeix del sistema de fitxers (desmarqueu per a eliminar de la base de dades només)",
|
||||
"LabelDescription": "Descripció",
|
||||
"LabelDeselectAll": "Desseleccionar Tots",
|
||||
"LabelDevice": "Dispositiu",
|
||||
"LabelDeviceInfo": "Informació del Dispositiu",
|
||||
"LabelDeviceInfo": "Informació de l'aparell",
|
||||
"LabelDeviceIsAvailableTo": "El dispositiu està disponible per a...",
|
||||
"LabelDirectory": "Directori",
|
||||
"LabelDiscFromFilename": "Disc a partir del nom de fitxer",
|
||||
@@ -335,11 +336,11 @@
|
||||
"LabelEnd": "Fi",
|
||||
"LabelEndOfChapter": "Fi del capítol",
|
||||
"LabelEpisode": "Episodi",
|
||||
"LabelEpisodeNotLinkedToRssFeed": "Episodi no enllaçat al feed RSS",
|
||||
"LabelEpisodeNotLinkedToRssFeed": "Episodi no enllaçat al canal RSS",
|
||||
"LabelEpisodeNumber": "Episodi #{0}",
|
||||
"LabelEpisodeTitle": "Títol de l'Episodi",
|
||||
"LabelEpisodeType": "Tipus d'Episodi",
|
||||
"LabelEpisodeUrlFromRssFeed": "URL de l'episodi del feed RSS",
|
||||
"LabelEpisodeUrlFromRssFeed": "URL de l'episodi del canal RSS",
|
||||
"LabelEpisodes": "Episodis",
|
||||
"LabelEpisodic": "Episodis",
|
||||
"LabelExample": "Exemple",
|
||||
@@ -352,7 +353,7 @@
|
||||
"LabelFeedURL": "Font de URL",
|
||||
"LabelFetchingMetadata": "Obtenció de metadades",
|
||||
"LabelFile": "Fitxer",
|
||||
"LabelFileBirthtime": "Arxiu creat a",
|
||||
"LabelFileBirthtime": "Fitxer creat a",
|
||||
"LabelFileBornDate": "Creat {0}",
|
||||
"LabelFileModified": "Fitxer modificat",
|
||||
"LabelFileModifiedDate": "Modificat {0}",
|
||||
@@ -439,7 +440,7 @@
|
||||
"LabelMinute": "Minut",
|
||||
"LabelMinutes": "Minuts",
|
||||
"LabelMissing": "Absent",
|
||||
"LabelMissingEbook": "No té ebook",
|
||||
"LabelMissingEbook": "No té llibre electrònic",
|
||||
"LabelMissingSupplementaryEbook": "No té ebook complementari",
|
||||
"LabelMobileRedirectURIs": "URI de redirecció mòbil permeses",
|
||||
"LabelMobileRedirectURIsDescription": "Aquesta és una llista blanca d'URI de redirecció vàlides per a aplicacions mòbils. El predeterminat és <code> audiobookshelf</code>, que pots eliminar o complementar amb URI addicionals per a la integració d'aplicacions de tercers. Usant un asterisc (<code> *</code>) com a única entrada que permet qualsevol URI.",
|
||||
@@ -473,6 +474,7 @@
|
||||
"LabelOpenIDAdvancedPermsClaimDescription": "Nom de la notificació de OpenID que conté permisos avançats per accions d'usuari dins l'aplicació que s'aplicaran a rols que no siguin d'administrador (<b>si estan configurats</b>). Si el reclam no apareix en la resposta, es denegarà l'accés a ABS. Si manca una sola opció, es tractarà com a <code>falsa</code>. Assegura't que la notificació del proveïdor d'identitats coincideixi amb l'estructura esperada:",
|
||||
"LabelOpenIDClaims": "Deixa les següents opcions buides per desactivar l'assignació avançada de grups i permisos, el que assignaria automàticament al grup 'Usuari'.",
|
||||
"LabelOpenIDGroupClaimDescription": "Nom de la declaració OpenID que conté una llista de grups de l'usuari. Comunament coneguts com <code>grups</code>. <b>Si es configura</b>, l'aplicació assignarà automàticament rols basats en la pertinença a grups de l'usuari, sempre que aquests grups es denominen 'admin', 'user' o 'guest' en la notificació. La sol·licitud ha de contenir una llista, i si un usuari pertany a diversos grups, l'aplicació assignarà el rol corresponent al major nivell d'accés. Si cap grup coincideix, es denegarà l'accés.",
|
||||
"LabelOpenRSSFeed": "Obre el canal RSS",
|
||||
"LabelOverwrite": "Sobreescriure",
|
||||
"LabelPaginationPageXOfY": "Pàgina {0} de {1}",
|
||||
"LabelPassword": "Contrasenya",
|
||||
@@ -496,25 +498,25 @@
|
||||
"LabelPodcastType": "Tipus de pòdcast",
|
||||
"LabelPodcasts": "Pòdcasts",
|
||||
"LabelPort": "Port",
|
||||
"LabelPrefixesToIgnore": "Prefixos per Ignorar (no distingeix entre majúscules i minúscules.)",
|
||||
"LabelPreventIndexing": "Evita que la teva font sigui indexada pels directoris de podcasts d'iTunes i Google",
|
||||
"LabelPrimaryEbook": "Ebook Principal",
|
||||
"LabelPrefixesToIgnore": "Prefixos a ignorar (no distingeix entre majúscules i minúscules)",
|
||||
"LabelPreventIndexing": "Evita que el vostre canal l'indexin els directoris de pòdcasts de l'iTunes i Google",
|
||||
"LabelPrimaryEbook": "Llibre electrònic principal",
|
||||
"LabelProgress": "Progrés",
|
||||
"LabelProvider": "Proveïdor",
|
||||
"LabelProviderAuthorizationValue": "Valor de l'encapçalament d'autorització",
|
||||
"LabelPubDate": "Data de Publicació",
|
||||
"LabelPublishYear": "Any de Publicació",
|
||||
"LabelPubDate": "Data de publicació",
|
||||
"LabelPublishYear": "Any de publicació",
|
||||
"LabelPublishedDate": "Publicat {0}",
|
||||
"LabelPublishedDecade": "Dècada de Publicació",
|
||||
"LabelPublishedDecade": "Dècada de publicació",
|
||||
"LabelPublishedDecades": "Dècades Publicades",
|
||||
"LabelPublisher": "Editor",
|
||||
"LabelPublishers": "Editors",
|
||||
"LabelRSSFeedCustomOwnerEmail": "Correu Electrònic Personalitzat del Propietari",
|
||||
"LabelRSSFeedCustomOwnerName": "Nom Personalitzat del Propietari",
|
||||
"LabelRSSFeedOpen": "Font RSS Oberta",
|
||||
"LabelRSSFeedPreventIndexing": "Evitar l'indexació",
|
||||
"LabelRSSFeedSlug": "Font RSS Slug",
|
||||
"LabelRSSFeedURL": "URL de la Font RSS",
|
||||
"LabelRSSFeedPreventIndexing": "Evita la indexació",
|
||||
"LabelRSSFeedSlug": "URL semàntic del canal RSS",
|
||||
"LabelRSSFeedURL": "URL del canal RSS",
|
||||
"LabelRandomly": "A l'atzar",
|
||||
"LabelReAddSeriesToContinueListening": "Reafegir la sèrie per continuar escoltant-la",
|
||||
"LabelRead": "Llegit",
|
||||
@@ -523,52 +525,61 @@
|
||||
"LabelRecentSeries": "Sèries recents",
|
||||
"LabelRecentlyAdded": "Addicions recents",
|
||||
"LabelRecommended": "Recomanats",
|
||||
"LabelRedo": "Refer",
|
||||
"LabelRedo": "Refés",
|
||||
"LabelRegion": "Regió",
|
||||
"LabelReleaseDate": "Data d'Estrena",
|
||||
"LabelRemoveAllMetadataAbs": "Eliminar tots els fitxers metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Eliminar tots els fitxers metadata.json",
|
||||
"LabelRemoveCover": "Eliminar Coberta",
|
||||
"LabelReleaseDate": "Data d'estrena",
|
||||
"LabelRemoveAllMetadataAbs": "Elimina tots els fitxers metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Elimina tots els fitxers metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Elimina la introducció i el tancament de l'Audible dels capítols",
|
||||
"LabelRemoveCover": "Elimina la coberta",
|
||||
"LabelRemoveMetadataFile": "Eliminar fitxers de metadades en carpetes d'elements de biblioteca",
|
||||
"LabelRemoveMetadataFileHelp": "Elimina tots els fitxers metadata.json i metadata.abs de les teves carpetes {0}.",
|
||||
"LabelRowsPerPage": "Files per Pàgina",
|
||||
"LabelSearchTerm": "Cercar Terme",
|
||||
"LabelSearchTitle": "Cercar Títol",
|
||||
"LabelSearchTitleOrASIN": "Cercar Títol o ASIN",
|
||||
"LabelRemoveMetadataFileHelp": "Elimina tots els fitxers metadata.json i metadata.abs de les vostres carpetes {0}.",
|
||||
"LabelRowsPerPage": "Files per pàgina",
|
||||
"LabelSearchTerm": "Cerca terme",
|
||||
"LabelSearchTitle": "Cerca títol",
|
||||
"LabelSearchTitleOrASIN": "Cerca títol o ASIN",
|
||||
"LabelSeason": "Temporada",
|
||||
"LabelSeasonNumber": "Temporada #{0}",
|
||||
"LabelSelectAll": "Seleccionar tot",
|
||||
"LabelSelectAllEpisodes": "Seleccionar tots els episodis",
|
||||
"LabelSeasonNumber": "{0}a temporada",
|
||||
"LabelSelectAll": "Selecciona-ho tot",
|
||||
"LabelSelectAllEpisodes": "Selecciona tots els episodis",
|
||||
"LabelSelectEpisodesShowing": "Seleccionar els {0} episodis visibles",
|
||||
"LabelSelectUsers": "Seleccionar usuaris",
|
||||
"LabelSendEbookToDevice": "Enviar Ebook a...",
|
||||
"LabelSequence": "Seqüència",
|
||||
"LabelSerial": "En sèrie",
|
||||
"LabelSeries": "Sèries",
|
||||
"LabelSeriesName": "Nom de la Sèrie",
|
||||
"LabelSeriesProgress": "Progrés de la Sèrie",
|
||||
"LabelSeries": "Sèrie",
|
||||
"LabelSeriesName": "Nom de la sèrie",
|
||||
"LabelSeriesProgress": "Progrés de la sèrie",
|
||||
"LabelServerLogLevel": "Nivell de registre del servidor",
|
||||
"LabelServerYearReview": "Resum de l'any del servidor ({0})",
|
||||
"LabelSetEbookAsPrimary": "Establir com a principal",
|
||||
"LabelSetEbookAsSupplementary": "Establir com a suplementari",
|
||||
"LabelSettingsAudiobooksOnly": "Només Audiollibres",
|
||||
"LabelSettingsAudiobooksOnlyHelp": "Activant aquesta opció s'ignoraran els fitxers d'ebook, excepte si estan dins d'una carpeta d'audiollibre, en aquest cas es marcaran com ebooks suplementaris",
|
||||
"LabelSettingsAudiobooksOnly": "Només audiollibres",
|
||||
"LabelSettingsAudiobooksOnlyHelp": "En activar aquesta opció s'ignoraran els fitxers de llibre electrònic, excepte si estan dins d'una carpeta d'audiollibre; en aquest cas es marcaran com a llibres suplementaris",
|
||||
"LabelSettingsBookshelfViewHelp": "Disseny esqueomorf amb prestatgeries de fusta",
|
||||
"LabelSettingsChromecastSupport": "Compatibilitat amb Chromecast",
|
||||
"LabelSettingsDateFormat": "Format de Data",
|
||||
"LabelSettingsDateFormat": "Format de data",
|
||||
"LabelSettingsEnableWatcherHelp": "Permet afegir/actualitzar elements automàticament quan es detectin canvis en els fitxers. *Requereix reiniciar el servidor",
|
||||
"LabelSettingsEpubsAllowScriptedContent": "Permetre scripts en epubs",
|
||||
"LabelSettingsEpubsAllowScriptedContentHelp": "Permetre que els fitxers epub executin scripts. Es recomana mantenir aquesta opció desactivada tret que confiïs en l'origen dels fitxers epub.",
|
||||
"LabelSettingsExperimentalFeatures": "Funcions Experimentals",
|
||||
"LabelSettingsExperimentalFeaturesHelp": "Funcions en desenvolupament que es beneficiarien dels teus comentaris i experiències de prova. Feu clic aquí per obrir una conversa a Github.",
|
||||
"LabelSettingsFindCovers": "Troba cobertes",
|
||||
"LabelSettingsHideSingleBookSeries": "Amaga les sèries amb un sol llibre",
|
||||
"LabelSettingsParseSubtitles": "Analitza els subtítols",
|
||||
"LabelSettingsSortingIgnorePrefixes": "Ignora els prefixos en ordenar",
|
||||
"LabelSettingsTimeFormat": "Format d'hora",
|
||||
"LabelShare": "Comparteix",
|
||||
"LabelShareDownloadableHelp": "Permet els usuaris amb l'enllaç de compartició de baixar un fitxer ZIP amb l'element de la biblioteca.",
|
||||
"LabelShareURL": "URL de compartició",
|
||||
"LabelShowAll": "Mostra-ho tot",
|
||||
"LabelShowSeconds": "Mostra segons",
|
||||
"LabelShowSubtitles": "Mostra subtítols",
|
||||
"LabelSize": "Mida",
|
||||
"LabelSleepTimer": "Temporitzador de repòs",
|
||||
"LabelSlug": "Slug",
|
||||
"LabelSortAscending": "Ascendent",
|
||||
"LabelSortDescending": "Descendent",
|
||||
"LabelStart": "Inicia",
|
||||
"LabelStartTime": "Hora d'inici",
|
||||
"LabelStarted": "Iniciat",
|
||||
@@ -656,89 +667,98 @@
|
||||
"LabelViewPlayerSettings": "Mostra els ajustaments del reproductor",
|
||||
"LabelViewQueue": "Mostra cua del reproductor",
|
||||
"LabelVolume": "Volum",
|
||||
"LabelWebRedirectURLsDescription": "Autoritza aquestes URL al teu proveïdor OAuth per permetre redirecció a l'aplicació web després d'iniciar sessió:",
|
||||
"LabelWebRedirectURLsDescription": "Autoritzeu aquests URL al vostre proveïdor OAuth per a permetre redirigir a l’aplicació web després d'iniciar sessió:",
|
||||
"LabelWebRedirectURLsSubfolder": "Subcarpeta per a URL de redirecció",
|
||||
"LabelWeekdaysToRun": "Executar en dies de la setmana",
|
||||
"LabelXBooks": "{0} llibres",
|
||||
"LabelXItems": "{0} elements",
|
||||
"LabelYearReviewHide": "Oculta resum de l'any",
|
||||
"LabelYearReviewShow": "Mostra resum de l'any",
|
||||
"LabelYourAudiobookDuration": "Duració del teu audiollibre",
|
||||
"LabelYourAudiobookDuration": "Duració del vostre audiollibre",
|
||||
"LabelYourBookmarks": "Els vostres marcadors",
|
||||
"LabelYourPlaylists": "Les teves llistes",
|
||||
"LabelYourPlaylists": "Les vostres llistes",
|
||||
"LabelYourProgress": "El vostre progrés",
|
||||
"MessageAddToPlayerQueue": "Afegeix a la cua del reproductor",
|
||||
"MessageAppriseDescription": "Per utilitzar aquesta funció, hauràs de tenir l'<a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">API d'Apprise</a> en funcionament o una API que gestioni resultats similars. <br/>La URL de l'API d'Apprise ha de tenir la mateixa ruta d'arxius que on s'envien les notificacions. Per exemple: si la teva API és a <code>http://192.168.1.1:8337</code>, llavors posaries <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Reengegueu el servidor després de desar perquè s'hi apliquin els canvis d'OIDC.",
|
||||
"MessageBackupsDescription": "Les còpies de seguretat inclouen: usuaris, progrés dels usuaris, detalls dels elements de la biblioteca, configuració del servidor i imatges a <code>/metadata/items</code> i <code>/metadata/authors</code>. Les còpies de seguretat <strong>NO</strong> inclouen cap fitxer guardat a la carpeta de la teva biblioteca.",
|
||||
"MessageBackupsLocationEditNote": "Nota: Actualitzar la ubicació de la còpia de seguretat no mourà ni modificarà les còpies existents",
|
||||
"MessageBackupsLocationNoEditNote": "Nota: La ubicació de la còpia de seguretat es defineix mitjançant una variable d'entorn i no es pot modificar aquí.",
|
||||
"MessageBackupsLocationPathEmpty": "La ruta de la còpia de seguretat no pot estar buida",
|
||||
"MessageBatchQuickMatchDescription": "La funció \"Troba Ràpid\" intentarà afegir portades i metadades que falten als elements seleccionats. Activa l'opció següent perquè \"Troba Ràpid\" pugui sobreescriure portades i/o metadades existents.",
|
||||
"MessageBookshelfNoCollections": "No tens cap col·lecció",
|
||||
"MessageBookshelfNoCollections": "Encara no heu fet cap col·lecció",
|
||||
"MessageBookshelfNoCollectionsHelp": "Les col·leccions són públiques. Tots els usuaris amb accés a la biblioteca les podran veure.",
|
||||
"MessageBookshelfNoRSSFeeds": "Cap font RSS està oberta",
|
||||
"MessageBookshelfNoResultsForFilter": "Cap resultat per al filtre \"{0}: {1}\"",
|
||||
"MessageBookshelfNoResultsForFilter": "Cap resultat per al filtre «{0}: {1}»",
|
||||
"MessageBookshelfNoResultsForQuery": "Cap resultat per a la consulta",
|
||||
"MessageBookshelfNoSeries": "No tens cap sèrie",
|
||||
"MessageBookshelfNoSeries": "No teniu cap sèrie",
|
||||
"MessageChapterEndIsAfter": "El final del capítol és després del final del teu audiollibre",
|
||||
"MessageChapterErrorFirstNotZero": "El primer capítol ha de començar a 0",
|
||||
"MessageChapterErrorStartGteDuration": "El temps d'inici no és vàlid: ha de ser inferior a la durada de l'audiollibre",
|
||||
"MessageChapterErrorStartLtPrev": "El temps d'inici no és vàlid: ha de ser igual o més gran que el temps d'inici del capítol anterior",
|
||||
"MessageChapterStartIsAfter": "L'inici del capítol és després del final del teu audiollibre",
|
||||
"MessageChaptersNotFound": "No s'han trobat els capítols",
|
||||
"MessageCheckingCron": "Comprovant cron...",
|
||||
"MessageConfirmCloseFeed": "Estàs segur que vols tancar aquesta font?",
|
||||
"MessageConfirmDeleteBackup": "Estàs segur que vols eliminar la còpia de seguretat {0}?",
|
||||
"MessageConfirmDeleteDevice": "Estàs segur que vols eliminar el lector electrònic \"{0}\"?",
|
||||
"MessageConfirmDeleteFile": "Això eliminarà el fitxer del teu sistema. Estàs segur?",
|
||||
"MessageConfirmDeleteLibrary": "Estàs segur que vols eliminar permanentment la biblioteca \"{0}\"?",
|
||||
"MessageConfirmDeleteLibraryItem": "Això eliminarà l'element de la base de dades i del sistema. Estàs segur?",
|
||||
"MessageConfirmDeleteLibraryItems": "Això eliminarà {0} element(s) de la base de dades i del sistema. Estàs segur?",
|
||||
"MessageConfirmDeleteMetadataProvider": "Estàs segur que vols eliminar el proveïdor de metadades personalitzat \"{0}\"?",
|
||||
"MessageConfirmDeleteNotification": "Estàs segur que vols eliminar aquesta notificació?",
|
||||
"MessageConfirmDeleteSession": "Estàs segur que vols eliminar aquesta sessió?",
|
||||
"MessageConfirmEmbedMetadataInAudioFiles": "Estàs segur que vols incrustar metadades a {0} fitxer(s) d'àudio?",
|
||||
"MessageConfirmForceReScan": "Estàs segur que vols forçar un reescaneig?",
|
||||
"MessageConfirmMarkAllEpisodesFinished": "Estàs segur que vols marcar tots els episodis com a acabats?",
|
||||
"MessageConfirmMarkAllEpisodesNotFinished": "Estàs segur que vols marcar tots els episodis com a no acabats?",
|
||||
"MessageConfirmMarkItemFinished": "Estàs segur que vols marcar \"{0}\" com a acabat?",
|
||||
"MessageConfirmMarkItemNotFinished": "Estàs segur que vols marcar \"{0}\" com a no acabat?",
|
||||
"MessageConfirmMarkSeriesFinished": "Estàs segur que vols marcar tots els llibres d'aquesta sèrie com a acabats?",
|
||||
"MessageConfirmMarkSeriesNotFinished": "Estàs segur que vols marcar tots els llibres d'aquesta sèrie com a no acabats?",
|
||||
"MessageConfirmNotificationTestTrigger": "Vols activar aquesta notificació amb dades de prova?",
|
||||
"MessageConfirmPurgeCache": "Esborrar la memòria cau eliminarà tot el directori localitzat a <code>/metadata/cache</code>. <br /><br />Estàs segur que vols eliminar-lo?",
|
||||
"MessageConfirmCloseFeed": "Segur que voleu tancar aquest canal?",
|
||||
"MessageConfirmDeleteBackup": "Segur que voleu suprimir la còpia de seguretat de {0}?",
|
||||
"MessageConfirmDeleteDevice": "Segur que voleu suprimir el lector electrònic «{0}»?",
|
||||
"MessageConfirmDeleteFile": "Això suprimirà el fitxer del vostre sistema de fitxers. N'esteu segur?",
|
||||
"MessageConfirmDeleteLibrary": "Segur que voleu suprimir permanentment la biblioteca «{0}»?",
|
||||
"MessageConfirmDeleteLibraryItem": "Això suprimirà l’element de la base de dades i del sistema de fitxers. N’esteu segur?",
|
||||
"MessageConfirmDeleteLibraryItems": "Això suprimirà {0} element(s) de la base de dades i del sistema de fitxers. N'esteu segur?",
|
||||
"MessageConfirmDeleteMetadataProvider": "Segur que voleu suprimir el proveïdor de metadades personalitzat «{0}»?",
|
||||
"MessageConfirmDeleteNotification": "Segur que voleu suprimir aquesta notificació?",
|
||||
"MessageConfirmDeleteSession": "Segur que voleu suprimir aquesta sessió?",
|
||||
"MessageConfirmEmbedMetadataInAudioFiles": "Segur que voleu incrustar metadades a {0} fitxer(s) d'àudio?",
|
||||
"MessageConfirmForceReScan": "Segur que voleu forçar un reescaneig?",
|
||||
"MessageConfirmMarkAllEpisodesFinished": "Segur que voleu marcar tots els episodis com a acabats?",
|
||||
"MessageConfirmMarkAllEpisodesNotFinished": "Segur que voleu marcar tots els episodis com a no acabats?",
|
||||
"MessageConfirmMarkItemFinished": "Segur que voleu marcar «{0}» com a acabat?",
|
||||
"MessageConfirmMarkItemNotFinished": "Segur que voleu marcar «{0}» com a no acabat?",
|
||||
"MessageConfirmMarkSeriesFinished": "Segur que voleu marcar tots els llibres d'aquesta sèrie com a acabats?",
|
||||
"MessageConfirmMarkSeriesNotFinished": "Segur que voleu marcar tots els llibres d'aquesta sèrie com a no acabats?",
|
||||
"MessageConfirmNotificationTestTrigger": "Voleu activar aquesta notificació amb dades de prova?",
|
||||
"MessageConfirmPurgeCache": "Purgar la memòria cau suprimirà tot el directori localitzat a <code>/metadata/cache</code>. <br /><br />Segur que voleu eliminar-lo?",
|
||||
"MessageConfirmPurgeItemsCache": "Esborrar la memòria cau dels elements eliminarà el directori <code>/metadata/cache/items</code>.<br />Estàs segur?",
|
||||
"MessageConfirmQuickEmbed": "Advertència! La integració ràpida no fa còpies de seguretat dels teus fitxers d'àudio. Assegura't d'haver-ne fet una còpia abans. <br><br>Vols continuar?",
|
||||
"MessageConfirmQuickEmbed": "Avís: la incrustació ràpida no fa còpies de seguretat dels vostres fitxers d'àudio. Assegureu-vos d'haver-ne fet una còpia abans. <br><br>Voleu continuar?",
|
||||
"MessageConfirmQuickMatchEpisodes": "El reconeixement ràpid sobreescriurà els detalls si es troba una coincidència. Estàs segur?",
|
||||
"MessageConfirmReScanLibraryItems": "Estàs segur que vols reescanejar {0} element(s)?",
|
||||
"MessageConfirmRemoveAllChapters": "Estàs segur que vols eliminar tots els capítols?",
|
||||
"MessageConfirmRemoveAuthor": "Estàs segur que vols eliminar l'autor \"{0}\"?",
|
||||
"MessageConfirmRemoveCollection": "Estàs segur que vols eliminar la col·lecció \"{0}\"?",
|
||||
"MessageConfirmRemoveEpisode": "Estàs segur que vols eliminar l'episodi \"{0}\"?",
|
||||
"MessageConfirmRemoveEpisodes": "Estàs segur que vols eliminar {0} episodis?",
|
||||
"MessageConfirmRemoveListeningSessions": "Estàs segur que vols eliminar {0} sessions d'escolta?",
|
||||
"MessageConfirmRemoveMetadataFiles": "Estàs segur que vols eliminar tots els fitxers de metadades.{0} de les carpetes dels elements de la teva biblioteca?",
|
||||
"MessageConfirmRemoveNarrator": "Estàs segur que vols eliminar el narrador \"{0}\"?",
|
||||
"MessageConfirmRemovePlaylist": "Estàs segur que vols eliminar la llista de reproducció \"{0}\"?",
|
||||
"MessageConfirmRenameGenre": "Estàs segur que vols canviar el gènere \"{0}\" a \"{1}\" per a tots els elements?",
|
||||
"MessageConfirmReScanLibraryItems": "Segur que voleu reescanejar {0} element(s)?",
|
||||
"MessageConfirmRemoveAllChapters": "Segur que voleu eliminar tots els capítols?",
|
||||
"MessageConfirmRemoveAuthor": "Segur que voleu eliminar l'autor «{0}»?",
|
||||
"MessageConfirmRemoveCollection": "Segur que voleu eliminar la col·lecció «{0}»?",
|
||||
"MessageConfirmRemoveEpisode": "Segur que voleu eliminar l'episodi «{0}»?",
|
||||
"MessageConfirmRemoveEpisodes": "Segur que voleu eliminar {0} episodis?",
|
||||
"MessageConfirmRemoveListeningSessions": "Segur que voleu eliminar {0} sessions d'escolta?",
|
||||
"MessageConfirmRemoveMetadataFiles": "Segur que voleu eliminar tots els fitxers metadata.{0} de les carpetes dels elements de la vostra biblioteca?",
|
||||
"MessageConfirmRemoveNarrator": "Segur que voleu eliminar el narrador «{0}»?",
|
||||
"MessageConfirmRemovePlaylist": "Segur que voleu eliminar la llista de reproducció «{0}»?",
|
||||
"MessageConfirmRenameGenre": "Segur que voleu canviar el nom del gènere «{0}» a «{1}» per a tots els elements?",
|
||||
"MessageConfirmRenameGenreMergeNote": "Nota: Aquest gènere ja existeix, i es fusionarà.",
|
||||
"MessageConfirmRenameGenreWarning": "Advertència! Ja existeix un gènere similar \"{0}\".",
|
||||
"MessageConfirmRenameTag": "Estàs segur que vols canviar l'etiqueta \"{0}\" a \"{1}\" per a tots els elements?",
|
||||
"MessageConfirmRenameTag": "Segur que voleu canviar el nom de l'etiqueta «{0}» a «{1}» per a tots els elements?",
|
||||
"MessageConfirmRenameTagMergeNote": "Nota: Aquesta etiqueta ja existeix, i es fusionarà.",
|
||||
"MessageConfirmRenameTagWarning": "Advertència! Ja existeix una etiqueta similar \"{0}\".",
|
||||
"MessageConfirmResetProgress": "Estàs segur que vols reiniciar el teu progrés?",
|
||||
"MessageConfirmSendEbookToDevice": "Estàs segur que vols enviar {0} ebook(s) \"{1}\" al dispositiu \"{2}\"?",
|
||||
"MessageConfirmUnlinkOpenId": "Estàs segur que vols desvincular aquest usuari d'OpenID?",
|
||||
"MessageConfirmResetProgress": "Segur que voleu reinicialitzar el vostre progrés?",
|
||||
"MessageConfirmSendEbookToDevice": "Segur que voleu enviar {0} llibre(s) «{1}» al dispositiu «{2}»?",
|
||||
"MessageConfirmUnlinkOpenId": "Segur que voleu desenllaçar aquest usuari d'OpenID?",
|
||||
"MessageDaysListenedInTheLastYear": "{0} dies escoltats l'any passat",
|
||||
"MessageDownloadingEpisode": "S'està baixant l'episodi",
|
||||
"MessageDragFilesIntoTrackOrder": "Arrossega els fitxers en l'ordre correcte de les pistes",
|
||||
"MessageEmbedFailed": "Error en incrustar!",
|
||||
"MessageEmbedFinished": "Incrustació acabada!",
|
||||
"MessageEmbedQueue": "En cua per incrustar metadades ({0} en cua)",
|
||||
"MessageFeedURLWillBe": "L'URL del canal serà {0}",
|
||||
"MessageFetching": "S'està recuperant...",
|
||||
"MessageImportantNotice": "Avís important",
|
||||
"MessageInsertChapterBelow": "Insereix un capítol a sota",
|
||||
"MessageInvalidAsin": "L'ASIN no és vàlid",
|
||||
"MessageItemsSelected": "{0} elements seleccionats",
|
||||
"MessageItemsUpdated": "{0} elements actualitzats",
|
||||
"MessageJoinUsOn": "Uniu-vos a nosaltres a",
|
||||
"MessageLoading": "S'està carregant...",
|
||||
"MessageLoadingFolders": "S'estan carregant les carpetes...",
|
||||
"MessageMarkAllEpisodesFinished": "Marca tots els episodis com a acabats",
|
||||
"MessageMarkAllEpisodesNotFinished": "Marca tots els episodis com a inacabats",
|
||||
"MessageMarkAsFinished": "Marcar com acabat",
|
||||
"MessageMarkAsNotFinished": "Marcar com no acabat",
|
||||
"MessageMatchBooksDescription": "S'intentarà fer coincidir els llibres de la biblioteca amb un llibre del proveïdor de cerca seleccionat, i s'ompliran els detalls buits i la portada. No sobreescriu els detalls.",
|
||||
@@ -779,38 +799,40 @@
|
||||
"MessagePauseChapter": "Pausar la reproducció del capítol",
|
||||
"MessagePlayChapter": "Escoltar l'inici del capítol",
|
||||
"MessagePlaylistCreateFromCollection": "Crear una llista de reproducció a partir d'una col·lecció",
|
||||
"MessagePleaseWait": "Espera si us plau...",
|
||||
"MessagePodcastHasNoRSSFeedForMatching": "El podcast no té una URL de font RSS que es pugui utilitzar",
|
||||
"MessagePodcastSearchField": "Introdueix el terme de cerca o la URL de la font RSS",
|
||||
"MessagePleaseWait": "Espereu...",
|
||||
"MessagePodcastHasNoRSSFeedForMatching": "El pòdcast no té un URL de canal RSS que es pugui utilitzar",
|
||||
"MessagePodcastSearchField": "Introduïu el terme de cerca o l'URL del canal RSS",
|
||||
"MessageQuickEmbedInProgress": "Integració ràpida en procés",
|
||||
"MessageQuickEmbedQueue": "En cua per a inserció ràpida ({0} en cua)",
|
||||
"MessageQuickMatchAllEpisodes": "Combina ràpidament tots els episodis",
|
||||
"MessageQuickMatchDescription": "Omple detalls d'elements buits i portades amb els primers resultats de '{0}'. No sobreescriu els detalls tret que l'opció \"Preferir metadades trobades\" del servidor estigui habilitada.",
|
||||
"MessageRemoveChapter": "Eliminar capítols",
|
||||
"MessageRemoveEpisodes": "Eliminar {0} episodi(s)",
|
||||
"MessageRemoveFromPlayerQueue": "Eliminar de la cua del reproductor",
|
||||
"MessageRemoveUserWarning": "Estàs segur que vols eliminar l'usuari \"{0}\"?",
|
||||
"MessageQuickMatchDescription": "Emplena els detalls i la coberta dels elements buits amb el resultat de la primera coincidència de «{0}». No sobreescriu els detalls tret que s'activi el paràmetre del servidor «Prefereix metadades coincidents».",
|
||||
"MessageRemoveChapter": "Elimina el capítol",
|
||||
"MessageRemoveEpisodes": "Elimina {0} episodi(s)",
|
||||
"MessageRemoveFromPlayerQueue": "Elimina de la cua del reproductor",
|
||||
"MessageRemoveUserWarning": "Segur que voleu suprimir permanentment l'usuari «{0}»?",
|
||||
"MessageReportBugsAndContribute": "Informa d'errors, sol·licita funcions i contribueix a",
|
||||
"MessageResetChaptersConfirm": "Estàs segur que vols desfer els canvis i revertir els capítols al seu estat original?",
|
||||
"MessageRestoreBackupConfirm": "Estàs segur que vols restaurar la còpia de seguretat creada a",
|
||||
"MessageResetChaptersConfirm": "Segur que voleu desfer els canvis i revertir els capítols al seu estat original?",
|
||||
"MessageRestoreBackupConfirm": "Segur que voleu restaurar la còpia de seguretat creada a",
|
||||
"MessageRestoreBackupWarning": "Restaurar sobreescriurà tota la base de dades situada a /config i les imatges de portades a /metadata/items i /metadata/authors.<br /><br />La còpia de seguretat no modifica cap fitxer a les carpetes de la teva biblioteca. Si has activat l'opció del servidor per guardar portades i metadades a les carpetes de la biblioteca, aquests fitxers no es guarden ni sobreescriuen.<br /><br />Tots els clients que utilitzin el teu servidor s'actualitzaran automàticament.",
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Executa cada {0} a les {1}",
|
||||
"MessageSearchResultsFor": "Resultats de la cerca de",
|
||||
"MessageSelected": "{0} seleccionat(s)",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "La seqüència de la sèrie no pot contenir espais",
|
||||
"MessageServerCouldNotBeReached": "No es va poder establir la connexió amb el servidor",
|
||||
"MessageSetChaptersFromTracksDescription": "Establir capítols utilitzant cada fitxer d'àudio com un capítol i el títol del capítol com el nom del fitxer d'àudio",
|
||||
"MessageShareExpirationWillBe": "La caducitat serà <strong>{0}</strong>",
|
||||
"MessageShareExpiresIn": "Caduca en {0}",
|
||||
"MessageShareURLWillBe": "La URL per compartir serà <strong>{0}</strong>",
|
||||
"MessageStartPlaybackAtTime": "Començar la reproducció per a \"{0}\" a {1}?",
|
||||
"MessageTaskAudioFileNotWritable": "El fitxer d'àudio \"{0}\" no es pot escriure",
|
||||
"MessageStartPlaybackAtTime": "Voleu començar la reproducció per a «{0}» a {1}?",
|
||||
"MessageTaskAudioFileNotWritable": "El fitxer d'àudio «{0}» no es pot escriure",
|
||||
"MessageTaskCanceledByUser": "Tasca cancel·lada per l'usuari",
|
||||
"MessageTaskDownloadingEpisodeDescription": "Descarregant l'episodi \"{0}\"",
|
||||
"MessageTaskDownloadingEpisodeDescription": "S'està baixant l'episodi «{0}»",
|
||||
"MessageTaskEmbeddingMetadata": "Inserint metadades",
|
||||
"MessageTaskEmbeddingMetadataDescription": "Inserint metadades en l'audiollibre \"{0}\"",
|
||||
"MessageTaskEncodingM4b": "Codificant M4B",
|
||||
"MessageTaskEncodingM4bDescription": "Codificant l'audiollibre \"{0}\" en un únic fitxer M4B",
|
||||
"MessageTaskEncodingM4bDescription": "S'està codificant l'audiollibre «{0}» en un únic fitxer M4B",
|
||||
"MessageTaskFailed": "Fallada",
|
||||
"MessageTaskFailedToBackupAudioFile": "Error en fer una còpia de seguretat del fitxer d'àudio \"{0}\"",
|
||||
"MessageTaskFailedToBackupAudioFile": "No s'ha pogut fer una còpia de seguretat del fitxer d'àudio «{0}»",
|
||||
"MessageTaskFailedToCreateCacheDirectory": "Error en crear el directori de la memòria cau",
|
||||
"MessageTaskFailedToEmbedMetadataInFile": "Error en incrustar metadades en el fitxer \"{0}\"",
|
||||
"MessageTaskFailedToMergeAudioFiles": "Error en fusionar fitxers d'àudio",
|
||||
@@ -819,14 +841,14 @@
|
||||
"MessageTaskMatchingBooksInLibrary": "Coincidint llibres a la biblioteca \"{0}\"",
|
||||
"MessageTaskNoFilesToScan": "Sense fitxers per escanejar",
|
||||
"MessageTaskOpmlImport": "Importar OPML",
|
||||
"MessageTaskOpmlImportDescription": "Creant podcasts a partir de {0} fonts RSS",
|
||||
"MessageTaskOpmlImportFeed": "Importació de feed OPML",
|
||||
"MessageTaskOpmlImportFeedDescription": "Importació del feed RSS \"{0}\"",
|
||||
"MessageTaskOpmlImportFeedFailed": "No es pot obtenir el podcast",
|
||||
"MessageTaskOpmlImportFeedPodcastDescription": "Creant el podcast \"{0}\"",
|
||||
"MessageTaskOpmlImportFeedPodcastExists": "El podcast ja existeix a la ruta",
|
||||
"MessageTaskOpmlImportFeedPodcastFailed": "Error en crear el podcast",
|
||||
"MessageTaskOpmlImportFinished": "Afegit {0} podcasts",
|
||||
"MessageTaskOpmlImportDescription": "S'estan creant pòdcasts a partir de {0} canals RSS",
|
||||
"MessageTaskOpmlImportFeed": "Importació d'un canal OPML",
|
||||
"MessageTaskOpmlImportFeedDescription": "S'està important el canal RSS «{0}»",
|
||||
"MessageTaskOpmlImportFeedFailed": "No s'ha pogut obtenir el canal del pòdcast",
|
||||
"MessageTaskOpmlImportFeedPodcastDescription": "S'està creant el pòdcast «{0}»",
|
||||
"MessageTaskOpmlImportFeedPodcastExists": "El pòdcast ja existeix al camí",
|
||||
"MessageTaskOpmlImportFeedPodcastFailed": "No s'ha pogut crear el pòdcast",
|
||||
"MessageTaskOpmlImportFinished": "S'han afegit {0} pòdcasts",
|
||||
"MessageTaskOpmlParseFailed": "No s'ha pogut analitzar el fitxer OPML",
|
||||
"MessageTaskOpmlParseFastFail": "No s'ha trobat l'etiqueta <opml> o <outline> al fitxer OPML",
|
||||
"MessageTaskOpmlParseNoneFound": "No s'han trobat fonts al fitxer OPML",
|
||||
@@ -844,13 +866,13 @@
|
||||
"MessageValidCronExpression": "Expressió de cron vàlida",
|
||||
"MessageWatcherIsDisabledGlobally": "El watcher està desactivat globalment a la configuració del servidor",
|
||||
"MessageXLibraryIsEmpty": "La biblioteca {0} està buida!",
|
||||
"MessageYourAudiobookDurationIsLonger": "La durada del teu audiollibre és més llarga que la durada trobada",
|
||||
"MessageYourAudiobookDurationIsShorter": "La durada del teu audiollibre és més curta que la durada trobada",
|
||||
"MessageYourAudiobookDurationIsLonger": "La durada del vostre audiollibre és major que la durada trobada",
|
||||
"MessageYourAudiobookDurationIsShorter": "La durada del vostre audiollibre és menor que la durada trobada",
|
||||
"NoteChangeRootPassword": "L'usuari Root és l'únic usuari que pot no tenir una contrasenya",
|
||||
"NoteChapterEditorTimes": "Nota: El temps d'inici del primer capítol ha de romandre a 0:00, i el temps d'inici de l'últim capítol no pot superar la durada de l'audiollibre.",
|
||||
"NoteFolderPicker": "Nota: Les carpetes ja assignades no es mostraran",
|
||||
"NoteRSSFeedPodcastAppsHttps": "Advertència: La majoria d'aplicacions de podcast requereixen que la URL de la font RSS utilitzi HTTPS",
|
||||
"NoteRSSFeedPodcastAppsPubDate": "Advertència: Un o més dels teus episodis no tenen data de publicació. Algunes aplicacions de podcast ho requereixen.",
|
||||
"NoteChapterEditorTimes": "Nota: el temps d'inici del primer capítol ha de romandre a 0:00, i el temps d'inici de l'últim capítol no pot superar la durada de l'audiollibre.",
|
||||
"NoteFolderPicker": "Nota: les carpetes ja assignades no es mostraran",
|
||||
"NoteRSSFeedPodcastAppsHttps": "Avís: la majoria d'aplicacions de pòdcast requereixen que l'URL del canal RSS utilitzi HTTPS",
|
||||
"NoteRSSFeedPodcastAppsPubDate": "Avís: un o més dels vostres episodis no tenen data de publicació. Algunes aplicacions de pòdcast ho requereixen.",
|
||||
"NoteUploaderFoldersWithMediaFiles": "Les carpetes amb fitxers multimèdia es gestionaran com a elements separats a la biblioteca.",
|
||||
"NoteUploaderOnlyAudioFiles": "Si només puges fitxers d'àudio, cada fitxer es gestionarà com un audiollibre separat.",
|
||||
"NoteUploaderUnsupportedFiles": "Els fitxers no compatibles seran ignorats. Si selecciona o arrossega una carpeta, els fitxers que no estiguin dins d'una subcarpeta seran ignorats.",
|
||||
@@ -859,7 +881,7 @@
|
||||
"NotificationOnEpisodeDownloadedDescription": "S'activa quan es descarrega automàticament un episodi d'un podcast",
|
||||
"NotificationOnTestDescription": "Esdeveniment per provar el sistema de notificacions",
|
||||
"PlaceholderNewCollection": "Nou nom de la col·lecció",
|
||||
"PlaceholderNewFolderPath": "Nova ruta de carpeta",
|
||||
"PlaceholderNewFolderPath": "Camí de carpeta nou",
|
||||
"PlaceholderNewPlaylist": "Nou nom de la llista de reproducció",
|
||||
"PlaceholderSearch": "Cerca...",
|
||||
"PlaceholderSearchEpisode": "Cerca d'episodis...",
|
||||
@@ -885,7 +907,7 @@
|
||||
"ToastAppriseUrlRequired": "Cal introduir una URL de Apprise",
|
||||
"ToastAsinRequired": "ASIN requerit",
|
||||
"ToastAuthorImageRemoveSuccess": "S'ha eliminat la imatge de l'autor",
|
||||
"ToastAuthorNotFound": "No s'ha trobat l'autor \"{0}\"",
|
||||
"ToastAuthorNotFound": "No s'ha trobat l'autor «{0}»",
|
||||
"ToastAuthorRemoveSuccess": "Autor eliminat",
|
||||
"ToastAuthorSearchNotFound": "No s'ha trobat l'autor",
|
||||
"ToastAuthorUpdateMerged": "Autor combinat",
|
||||
@@ -901,6 +923,7 @@
|
||||
"ToastBackupRestoreFailed": "Error en restaurar la còpia de seguretat",
|
||||
"ToastBackupUploadFailed": "Error en carregar la còpia de seguretat",
|
||||
"ToastBackupUploadSuccess": "Còpia de seguretat carregada",
|
||||
"ToastBatchApplyDetailsToItemsSuccess": "S'han aplicat els detalls als elements",
|
||||
"ToastBatchDeleteFailed": "Error en l'eliminació per lots",
|
||||
"ToastBatchDeleteSuccess": "Eliminació per lots correcte",
|
||||
"ToastBatchQuickMatchFailed": "Error en la sincronització ràpida per lots!",
|
||||
@@ -913,6 +936,8 @@
|
||||
"ToastCachePurgeFailed": "Error en purgar la memòria cau",
|
||||
"ToastCachePurgeSuccess": "Memòria cau purgada amb èxit",
|
||||
"ToastChaptersHaveErrors": "Els capítols tenen errors",
|
||||
"ToastChaptersInvalidShiftAmountLast": "La quantitat de desplaçament no és vàlida. L'hora d'inici de l'últim capítol s'estendria més enllà de la durada d'aquest audiollibre.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "La quantitat de desplaçament no és vàlida. El primer capítol tindria una durada zero o negativa i el sobreescriuria el segon capítol. Augmenteu la durada inicial del segon capítol.",
|
||||
"ToastChaptersMustHaveTitles": "Els capítols han de tenir un títol",
|
||||
"ToastChaptersRemoved": "Capítols eliminats",
|
||||
"ToastChaptersUpdated": "Capítols actualitzats",
|
||||
@@ -920,6 +945,7 @@
|
||||
"ToastCollectionRemoveSuccess": "Col·lecció eliminada",
|
||||
"ToastCollectionUpdateSuccess": "Col·lecció actualitzada",
|
||||
"ToastCoverUpdateFailed": "Error en actualitzar la portada",
|
||||
"ToastDateTimeInvalidOrIncomplete": "La data i hora no és vàlida o està incompleta",
|
||||
"ToastDeleteFileFailed": "No s'ha pogut suprimir el fitxer",
|
||||
"ToastDeleteFileSuccess": "Fitxer suprimit",
|
||||
"ToastDeviceAddFailed": "Error en afegir el dispositiu",
|
||||
@@ -950,34 +976,35 @@
|
||||
"ToastItemMarkedAsNotFinishedSuccess": "Element marcat com a no acabat",
|
||||
"ToastItemUpdateSuccess": "Element actualitzat",
|
||||
"ToastLibraryCreateFailed": "Error en crear la biblioteca",
|
||||
"ToastLibraryCreateSuccess": "Biblioteca \"{0}\" creada",
|
||||
"ToastLibraryCreateSuccess": "S'ha creat la biblioteca «{0}»",
|
||||
"ToastLibraryDeleteFailed": "Error en eliminar la biblioteca",
|
||||
"ToastLibraryDeleteSuccess": "Biblioteca eliminada",
|
||||
"ToastLibraryScanFailedToStart": "Error en iniciar l'escaneig",
|
||||
"ToastLibraryScanStarted": "S'ha iniciat l'escaneig de la biblioteca",
|
||||
"ToastLibraryUpdateSuccess": "Biblioteca \"{0}\" actualitzada",
|
||||
"ToastLibraryUpdateSuccess": "S'ha actualitzat la biblioteca «{0}»",
|
||||
"ToastMatchAllAuthorsFailed": "No coincideix amb tots els autors",
|
||||
"ToastMetadataFilesRemovedError": "Error en eliminar metadades de {0} arxius",
|
||||
"ToastMetadataFilesRemovedNoneFound": "No s'han trobat metadades en {0} arxius",
|
||||
"ToastMetadataFilesRemovedNoneRemoved": "Cap metadada eliminada en {0} arxius",
|
||||
"ToastMetadataFilesRemovedError": "S’ha produït un error en eliminar els fitxers metadata.{0}",
|
||||
"ToastMetadataFilesRemovedNoneFound": "No hi ha cap fitxer metadata.{0} a la biblioteca",
|
||||
"ToastMetadataFilesRemovedNoneRemoved": "No s'ha eliminat cap fitxer metadata.{0}",
|
||||
"ToastMetadataFilesRemovedSuccess": "{0} metadades eliminades en {1} arxius",
|
||||
"ToastMustHaveAtLeastOnePath": "Ha de tenir almenys una ruta",
|
||||
"ToastNameEmailRequired": "El nom i el correu electrònic són obligatoris",
|
||||
"ToastNameRequired": "Nom obligatori",
|
||||
"ToastNewEpisodesFound": "{0} episodi(s) nou(s) trobat(s)",
|
||||
"ToastNewUserCreatedFailed": "Error en crear el compte: \"{0}\"",
|
||||
"ToastNewUserCreatedFailed": "No s'ha pogut crear el compte: «{0}»",
|
||||
"ToastNewUserCreatedSuccess": "Nou compte creat",
|
||||
"ToastNewUserLibraryError": "Ha de seleccionar almenys una biblioteca",
|
||||
"ToastNewUserPasswordError": "Necessites una contrasenya, només el root pot estar sense contrasenya",
|
||||
"ToastNewUserTagError": "Selecciona almenys una etiqueta",
|
||||
"ToastNewUserUsernameError": "Introdueix un nom d'usuari",
|
||||
"ToastNewUserLibraryError": "S'ha de seleccionar almenys una biblioteca",
|
||||
"ToastNewUserPasswordError": "Cal una contrasenya; només l'usuari primari pot estar sense contrasenya",
|
||||
"ToastNewUserTagError": "S'ha de seleccionar almenys una etiqueta",
|
||||
"ToastNewUserUsernameError": "Introduïu un nom d'usuari",
|
||||
"ToastNoNewEpisodesFound": "No s'han trobat nous episodis",
|
||||
"ToastNoRSSFeed": "El pòdcast no té canal RSS",
|
||||
"ToastNoUpdatesNecessary": "No cal actualitzar",
|
||||
"ToastNotificationCreateFailed": "Error en crear la notificació",
|
||||
"ToastNotificationDeleteFailed": "Error en eliminar la notificació",
|
||||
"ToastNotificationCreateFailed": "No s'ha pogut crear la notificació",
|
||||
"ToastNotificationDeleteFailed": "No s'ha pogut suprimir la notificació",
|
||||
"ToastNotificationFailedMaximum": "El nombre màxim d'intents fallits ha de ser ≥ 0",
|
||||
"ToastNotificationQueueMaximum": "La cua de notificació màxima ha de ser ≥ 0",
|
||||
"ToastNotificationSettingsUpdateSuccess": "Configuració de notificació actualitzada",
|
||||
"ToastNotificationSettingsUpdateSuccess": "S'han actualitzat els paràmetres de notificacions",
|
||||
"ToastNotificationTestTriggerFailed": "No s'ha pogut activar la notificació de prova",
|
||||
"ToastNotificationTestTriggerSuccess": "Notificació de prova activada",
|
||||
"ToastNotificationUpdateSuccess": "Notificació actualitzada",
|
||||
@@ -987,16 +1014,16 @@
|
||||
"ToastPlaylistUpdateSuccess": "Llista de reproducció actualitzada",
|
||||
"ToastPodcastCreateFailed": "No s'ha pogut crear el pòdcast",
|
||||
"ToastPodcastCreateSuccess": "S'ha creat el pòdcast correctament",
|
||||
"ToastPodcastGetFeedFailed": "No s'ha pogut obtenir el podcast",
|
||||
"ToastPodcastNoEpisodesInFeed": "No s'han trobat episodis en el feed RSS",
|
||||
"ToastPodcastNoRssFeed": "El podcast no té un feed RSS",
|
||||
"ToastPodcastGetFeedFailed": "No s'ha pogut obtenir el canal del pòdcast",
|
||||
"ToastPodcastNoEpisodesInFeed": "No s'ha trobat cap episodi al canal RSS",
|
||||
"ToastPodcastNoRssFeed": "El pòdcast no té un canal RSS",
|
||||
"ToastProgressIsNotBeingSynced": "El progrés no s'està sincronitzant, reinicia la reproducció",
|
||||
"ToastProviderCreatedFailed": "Error en afegir el proveïdor",
|
||||
"ToastProviderCreatedSuccess": "Nou proveïdor afegit",
|
||||
"ToastProviderNameAndUrlRequired": "Nom i URL obligatoris",
|
||||
"ToastProviderRemoveSuccess": "Proveïdor eliminat",
|
||||
"ToastRSSFeedCloseFailed": "Error en tancar el feed RSS",
|
||||
"ToastRSSFeedCloseSuccess": "Feed RSS tancat",
|
||||
"ToastRSSFeedCloseFailed": "No s'ha pogut tancar el canal RSS",
|
||||
"ToastRSSFeedCloseSuccess": "Canal RSS tancat",
|
||||
"ToastRemoveFailed": "Error en eliminar",
|
||||
"ToastRemoveItemFromCollectionFailed": "Error en eliminar l'element de la col·lecció",
|
||||
"ToastRemoveItemFromCollectionSuccess": "Element eliminat de la col·lecció",
|
||||
@@ -1010,7 +1037,8 @@
|
||||
"ToastScanFailed": "No s'ha pogut escanejar l'element de la biblioteca",
|
||||
"ToastSelectAtLeastOneUser": "Selecciona almenys un usuari",
|
||||
"ToastSendEbookToDeviceFailed": "Error en enviar l'ebook al dispositiu",
|
||||
"ToastSendEbookToDeviceSuccess": "Ebook enviat al dispositiu \"{0}\"",
|
||||
"ToastSendEbookToDeviceSuccess": "El llibre electrònic s'ha enviat al dispositiu «{0}»",
|
||||
"ToastSeriesSubmitFailedSameName": "No és possible afegir dues sèries amb el mateix nom",
|
||||
"ToastSeriesUpdateFailed": "Error en actualitzar la sèrie",
|
||||
"ToastSeriesUpdateSuccess": "Sèrie actualitzada",
|
||||
"ToastServerSettingsUpdateSuccess": "Configuració del servidor actualitzada",
|
||||
|
||||
@@ -88,7 +88,7 @@
|
||||
"ButtonSave": "Speichern",
|
||||
"ButtonSaveAndClose": "Speichern & Schließen",
|
||||
"ButtonSaveTracklist": "Speichere die Titelliste",
|
||||
"ButtonScan": "Partial-Scan (nur geänderte/neue Medien)",
|
||||
"ButtonScan": "Scannen",
|
||||
"ButtonScanLibrary": "Bibliothek scannen",
|
||||
"ButtonScrollLeft": "Nach Links scrollen",
|
||||
"ButtonScrollRight": "Nach Rechts scrollen",
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Wiedergabeliste",
|
||||
"HeaderPlaylistItems": "Einträge in der Wiedergabeliste",
|
||||
"HeaderPodcastsToAdd": "Podcasts zum Hinzufügen",
|
||||
"HeaderPresets": "Voreinstellungen",
|
||||
"HeaderPreviewCover": "Vorschau Titelbild",
|
||||
"HeaderRSSFeedGeneral": "RSS Details",
|
||||
"HeaderRSSFeedIsOpen": "RSS-Feed ist geöffnet",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Veröffentlichungsdatum",
|
||||
"LabelRemoveAllMetadataAbs": "Alle metadata.abs Dateien löschen",
|
||||
"LabelRemoveAllMetadataJson": "Alle metadata.json Dateien löschen",
|
||||
"LabelRemoveAudibleBranding": "Audible Intro sowie Outro aus Kapiteln entfernen",
|
||||
"LabelRemoveCover": "Entferne Titelbild",
|
||||
"LabelRemoveMetadataFile": "Metadaten-Dateien in Bibliotheksordnern löschen",
|
||||
"LabelRemoveMetadataFileHelp": "Alle metadata.json und metadata.abs Dateien aus den Ordnern {0} löschen.",
|
||||
@@ -705,6 +707,8 @@
|
||||
"LabelYourProgress": "Fortschritt",
|
||||
"MessageAddToPlayerQueue": "Zur Abspielwarteliste hinzufügen",
|
||||
"MessageAppriseDescription": "Um diese Funktion nutzen zu können, musst du eine Instanz von <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> laufen haben oder eine API verwenden welche dieselbe Anfragen bearbeiten kann. <br />Die Apprise API Url muss der vollständige URL-Pfad sein, an den die Benachrichtigung gesendet werden soll, z.B. wenn Ihre API-Instanz unter <code>http://192.168.1.1:8337</code> läuft, würdest du <code>http://192.168.1.1:8337/notify</code> eingeben.",
|
||||
"MessageAsinCheck": "Stellen Sie sicher, dass Sie die ASIN aus der richtigen Audible Region verwenden, nicht Amazon.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Nach dem Speichern muß der Server neugestartet werden um die OIDC Änderungen zu übernehmen.",
|
||||
"MessageBackupsDescription": "In einer Sicherung werden Benutzer, Benutzerfortschritte, Details zu den Bibliotheksobjekten, Servereinstellungen und Bilder welche in <code>/metadata/items</code> & <code>/metadata/authors</code> gespeichert sind gespeichert. Sicherungen enthalten keine Dateien welche in den einzelnen Bibliotheksordnern (Medien-Ordnern) gespeichert sind.",
|
||||
"MessageBackupsLocationEditNote": "Hinweis: Durch das Aktualisieren des Backup-Speicherorts werden vorhandene Sicherungen nicht verschoben oder geändert",
|
||||
"MessageBackupsLocationNoEditNote": "Hinweis: Der Sicherungsspeicherort wird über eine Umgebungsvariable festgelegt und kann hier nicht geändert werden.",
|
||||
@@ -723,6 +727,7 @@
|
||||
"MessageChapterErrorStartGteDuration": "Ungültige Kapitelstartzeit: Kapitelanfang > Mediumlänge (Kapitelanfang liegt zeitlich nach dem Ende des Mediums -> Lösung: Kapitelanfang < Mediumlänge)",
|
||||
"MessageChapterErrorStartLtPrev": "Ungültige Kapitelstartzeit: Kapitelanfang < Kapitelanfang vorheriges Kapitel (Kapitelanfang liegt zeitlich vor dem Beginn des vorherigen Kapitels -> Lösung: Kapitelanfang >= Startzeit des vorherigen Kapitels)",
|
||||
"MessageChapterStartIsAfter": "Ungültige Kapitelstartzeit: Kapitelanfang > Mediumende (Kapitelanfang liegt nach dem Ende des Mediums)",
|
||||
"MessageChaptersNotFound": "Kapitel gefunden nicht",
|
||||
"MessageCheckingCron": "Überprüfe Cron...",
|
||||
"MessageConfirmCloseFeed": "Feed wird geschlossen! Bist du dir sicher?",
|
||||
"MessageConfirmDeleteBackup": "Sicherung für {0} wird gelöscht! Bist du dir sicher?",
|
||||
@@ -779,6 +784,7 @@
|
||||
"MessageForceReScanDescription": "Durchsucht alle Dateien erneut, wie bei einem frischen Scan. ID3-Tags von Audiodateien, OPF-Dateien und Textdateien werden neu durchsucht.",
|
||||
"MessageImportantNotice": "Wichtiger Hinweis!",
|
||||
"MessageInsertChapterBelow": "Kapitel unten einfügen",
|
||||
"MessageInvalidAsin": "Ungültige ASIN",
|
||||
"MessageItemsSelected": "{0} ausgewählte Medien",
|
||||
"MessageItemsUpdated": "{0} Medien aktualisiert",
|
||||
"MessageJoinUsOn": "Besuche uns auf",
|
||||
@@ -850,6 +856,7 @@
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Immer {0} um {1} ausführen",
|
||||
"MessageSearchResultsFor": "Suchergebnisse für",
|
||||
"MessageSelected": "{0} ausgewählt",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "Serie Abfolge kann keine Leerzeichen enthalten",
|
||||
"MessageServerCouldNotBeReached": "Server kann nicht erreicht werden",
|
||||
"MessageSetChaptersFromTracksDescription": "Kaitelerstellung basiert auf den existierenden einzelnen Audiodateien. Pro existierende Audiodatei wird 1 Kapitel erstellt, wobei deren Kapitelname aus dem Audiodateinamen extrahiert wird",
|
||||
"MessageShareExpirationWillBe": "Läuft am <strong>{0}</strong> ab",
|
||||
@@ -968,6 +975,8 @@
|
||||
"ToastCachePurgeFailed": "Cache leeren fehlgeschlagen",
|
||||
"ToastCachePurgeSuccess": "Cache geleert",
|
||||
"ToastChaptersHaveErrors": "Kapitel sind fehlerhaft",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Die Verschiebung ist nicht möglich, da die Startzeit des letzten Kapitels über die Gesamtdauer dieses Hörbuchs hinausgehen würde.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Ungültige Höhe der Verschiebung. Das erste Kapitel hätte eine Länge von Null oder eine negative Länge und würde vom zweiten Kapitel überschrieben werden. Erhöhen Sie die Startdauer des zweiten Kapitels.",
|
||||
"ToastChaptersMustHaveTitles": "Kapitel benötigen eindeutige Namen",
|
||||
"ToastChaptersRemoved": "Kapitel entfernt",
|
||||
"ToastChaptersUpdated": "Kapitel aktualisiert",
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Playlist",
|
||||
"HeaderPlaylistItems": "Playlist Items",
|
||||
"HeaderPodcastsToAdd": "Podcasts to Add",
|
||||
"HeaderPresets": "Presets",
|
||||
"HeaderPreviewCover": "Preview Cover",
|
||||
"HeaderRSSFeedGeneral": "RSS Details",
|
||||
"HeaderRSSFeedIsOpen": "RSS Feed is Open",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Release Date",
|
||||
"LabelRemoveAllMetadataAbs": "Remove all metadata.abs files",
|
||||
"LabelRemoveAllMetadataJson": "Remove all metadata.json files",
|
||||
"LabelRemoveAudibleBranding": "Remove Audible intro and outro from chapters",
|
||||
"LabelRemoveCover": "Remove cover",
|
||||
"LabelRemoveMetadataFile": "Remove metadata files in library item folders",
|
||||
"LabelRemoveMetadataFileHelp": "Remove all metadata.json and metadata.abs files in your {0} folders.",
|
||||
@@ -706,6 +708,7 @@
|
||||
"MessageAddToPlayerQueue": "Add to player queue",
|
||||
"MessageAppriseDescription": "To use this feature you will need to have an instance of <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> running or an api that will handle those same requests. <br />The Apprise API Url should be the full URL path to send the notification, e.g., if your API instance is served at <code>http://192.168.1.1:8337</code> then you would put <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Ensure you are using the ASIN from the correct Audible region, not Amazon.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Restart your server after saving to apply OIDC changes.",
|
||||
"MessageBackupsDescription": "Backups include users, user progress, library item details, server settings, and images stored in <code>/metadata/items</code> & <code>/metadata/authors</code>. Backups <strong>do not</strong> include any files stored in your library folders.",
|
||||
"MessageBackupsLocationEditNote": "Note: Updating the backup location will not move or modify existing backups",
|
||||
"MessageBackupsLocationNoEditNote": "Note: The backup location is set through an environment variable and cannot be changed here.",
|
||||
@@ -853,6 +856,7 @@
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Run every {0} at {1}",
|
||||
"MessageSearchResultsFor": "Search results for",
|
||||
"MessageSelected": "{0} selected",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "Series sequence cannot contain spaces",
|
||||
"MessageServerCouldNotBeReached": "Server could not be reached",
|
||||
"MessageSetChaptersFromTracksDescription": "Set chapters using each audio file as a chapter and chapter title as the audio file name",
|
||||
"MessageShareExpirationWillBe": "Expiration will be <strong>{0}</strong>",
|
||||
@@ -971,6 +975,8 @@
|
||||
"ToastCachePurgeFailed": "Failed to purge cache",
|
||||
"ToastCachePurgeSuccess": "Cache purged successfully",
|
||||
"ToastChaptersHaveErrors": "Chapters have errors",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Invalid shift amount. The last chapter start time would extend beyond the duration of this audiobook.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Invalid shift amount. The first chapter would have zero or negative length and would be overwritten by the second chapter. Increase the start duration of second chapter.",
|
||||
"ToastChaptersMustHaveTitles": "Chapters must have titles",
|
||||
"ToastChaptersRemoved": "Chapters removed",
|
||||
"ToastChaptersUpdated": "Chapters updated",
|
||||
|
||||
@@ -10,13 +10,15 @@
|
||||
"ButtonApplyChapters": "Aplicar capítulos",
|
||||
"ButtonAuthors": "Autores",
|
||||
"ButtonBack": "Atrás",
|
||||
"ButtonBatchEditPopulateFromExisting": "Rellenar desde existentes",
|
||||
"ButtonBatchEditPopulateMapDetails": "Rellenar detalles de mapa",
|
||||
"ButtonBrowseForFolder": "Buscar carpeta",
|
||||
"ButtonCancel": "Cancelar",
|
||||
"ButtonCancelEncode": "Cancelar Codificador",
|
||||
"ButtonCancelEncode": "Cancelar codificación",
|
||||
"ButtonChangeRootPassword": "Cambiar contraseña administrativa",
|
||||
"ButtonCheckAndDownloadNewEpisodes": "Comprobar y descargar episodios nuevos",
|
||||
"ButtonChooseAFolder": "Escoger una Carpeta",
|
||||
"ButtonChooseFiles": "Escoger un Archivo",
|
||||
"ButtonChooseAFolder": "Elegir una carpeta",
|
||||
"ButtonChooseFiles": "Elegir archivos",
|
||||
"ButtonClearFilter": "Quitar filtros",
|
||||
"ButtonCloseFeed": "Cerrar suministro",
|
||||
"ButtonCloseSession": "Cerrar la sesión abierta",
|
||||
@@ -87,7 +89,7 @@
|
||||
"ButtonSaveAndClose": "Guardar y cerrar",
|
||||
"ButtonSaveTracklist": "Guardar lista de pistas",
|
||||
"ButtonScan": "Escanear",
|
||||
"ButtonScanLibrary": "Escanear Biblioteca",
|
||||
"ButtonScanLibrary": "Escanear biblioteca",
|
||||
"ButtonScrollLeft": "Desplazarse hacia la izquierda",
|
||||
"ButtonScrollRight": "Desplazarse hacia la derecha",
|
||||
"ButtonSearch": "Buscar",
|
||||
@@ -147,7 +149,7 @@
|
||||
"HeaderLastListeningSession": "Última sesión de escucha",
|
||||
"HeaderLatestEpisodes": "Episodios más recientes",
|
||||
"HeaderLibraries": "Bibliotecas",
|
||||
"HeaderLibraryFiles": "Archivos de Biblioteca",
|
||||
"HeaderLibraryFiles": "Archivos de biblioteca",
|
||||
"HeaderLibraryStats": "Estadísticas de biblioteca",
|
||||
"HeaderListeningSessions": "Sesión",
|
||||
"HeaderListeningStats": "Estadísticas de Tiempo Escuchado",
|
||||
@@ -175,6 +177,7 @@
|
||||
"HeaderPlaylist": "Lista de reproducción",
|
||||
"HeaderPlaylistItems": "Elementos de lista de reproducción",
|
||||
"HeaderPodcastsToAdd": "Pódcast para añadir",
|
||||
"HeaderPresets": "Preconfiguraciones",
|
||||
"HeaderPreviewCover": "Previsualizar cubierta",
|
||||
"HeaderRSSFeedGeneral": "Detalles de RSS",
|
||||
"HeaderRSSFeedIsOpen": "El suministro RSS está abierto",
|
||||
@@ -190,7 +193,7 @@
|
||||
"HeaderSettings": "Configuración",
|
||||
"HeaderSettingsDisplay": "Interfaz",
|
||||
"HeaderSettingsExperimental": "Funcionalidades experimentales",
|
||||
"HeaderSettingsGeneral": "General",
|
||||
"HeaderSettingsGeneral": "Generales",
|
||||
"HeaderSettingsScanner": "Escáner",
|
||||
"HeaderSettingsWebClient": "Cliente web",
|
||||
"HeaderSleepTimer": "Temporizador de apagado",
|
||||
@@ -262,14 +265,14 @@
|
||||
"LabelBooks": "Libros",
|
||||
"LabelButtonText": "Texto del botón",
|
||||
"LabelByAuthor": "por {0}",
|
||||
"LabelChangePassword": "Cambiar Contraseña",
|
||||
"LabelChangePassword": "Cambiar contraseña",
|
||||
"LabelChannels": "Canales",
|
||||
"LabelChapterCount": "{0} capítulos",
|
||||
"LabelChapterTitle": "Titulo del Capítulo",
|
||||
"LabelChapterTitle": "Título del capítulo",
|
||||
"LabelChapters": "Capítulos",
|
||||
"LabelChaptersFound": "Capítulo Encontrado",
|
||||
"LabelClickForMoreInfo": "Click para más información",
|
||||
"LabelClickToUseCurrentValue": "Haz clic para utilizar el valor actual",
|
||||
"LabelChaptersFound": "capítulos encontrados",
|
||||
"LabelClickForMoreInfo": "Pulse para más información",
|
||||
"LabelClickToUseCurrentValue": "Pulse para utilizar el valor actual",
|
||||
"LabelClosePlayer": "Cerrar reproductor",
|
||||
"LabelCodec": "Codec",
|
||||
"LabelCollapseSeries": "Colapsar serie",
|
||||
@@ -277,7 +280,7 @@
|
||||
"LabelCollection": "Colección",
|
||||
"LabelCollections": "Colecciones",
|
||||
"LabelComplete": "Completo",
|
||||
"LabelConfirmPassword": "Confirmar Contraseña",
|
||||
"LabelConfirmPassword": "Confirmar contraseña",
|
||||
"LabelContinueListening": "Seguir escuchando",
|
||||
"LabelContinueReading": "Continuar leyendo",
|
||||
"LabelContinueSeries": "Continuar series",
|
||||
@@ -288,8 +291,8 @@
|
||||
"LabelCronExpression": "Expresión de Cron",
|
||||
"LabelCurrent": "Actual",
|
||||
"LabelCurrently": "En este momento:",
|
||||
"LabelCustomCronExpression": "Expresión de Cron Personalizada:",
|
||||
"LabelDatetime": "Hora y Fecha",
|
||||
"LabelCustomCronExpression": "Expresión de Cron personalizada:",
|
||||
"LabelDatetime": "Hora y fecha",
|
||||
"LabelDays": "Días",
|
||||
"LabelDeleteFromFileSystemCheckbox": "Eliminar del sistema de archivos (desmarque para quitar de la base de datos solamente)",
|
||||
"LabelDescription": "Descripción",
|
||||
@@ -368,7 +371,7 @@
|
||||
"LabelFontStrikethrough": "Tachado",
|
||||
"LabelFormat": "Formato",
|
||||
"LabelFull": "Completo",
|
||||
"LabelGenre": "Genero",
|
||||
"LabelGenre": "Género",
|
||||
"LabelGenres": "Géneros",
|
||||
"LabelHardDeleteFile": "Eliminar Definitivamente",
|
||||
"LabelHasEbook": "Tiene un libro",
|
||||
@@ -459,17 +462,17 @@
|
||||
"LabelNotes": "Notas",
|
||||
"LabelNotificationAppriseURL": "URL(s) de Apprise",
|
||||
"LabelNotificationAvailableVariables": "Variables disponibles",
|
||||
"LabelNotificationBodyTemplate": "Plantilla de Cuerpo",
|
||||
"LabelNotificationEvent": "Evento de Notificación",
|
||||
"LabelNotificationTitleTemplate": "Plantilla de Titulo",
|
||||
"LabelNotificationsMaxFailedAttempts": "Máximo de Intentos Fallidos",
|
||||
"LabelNotificationBodyTemplate": "Plantilla de cuerpo",
|
||||
"LabelNotificationEvent": "Evento de notificación",
|
||||
"LabelNotificationTitleTemplate": "Plantilla de título",
|
||||
"LabelNotificationsMaxFailedAttempts": "Máximo de intentos fallidos",
|
||||
"LabelNotificationsMaxFailedAttemptsHelp": "Las notificaciones se desactivan después de fallar este número de veces",
|
||||
"LabelNotificationsMaxQueueSize": "Tamaño máximo de la cola de notificaciones",
|
||||
"LabelNotificationsMaxQueueSizeHelp": "Las notificaciones están limitadas a 1 por segundo. Las notificaciones serán ignoradas si llegan al numero máximo de cola para prevenir spam de eventos.",
|
||||
"LabelNumberOfBooks": "Número de libros",
|
||||
"LabelNumberOfEpisodes": "N.º de episodios",
|
||||
"LabelOpenIDAdvancedPermsClaimDescription": "Nombre de la notificación de OpenID que contiene permisos avanzados para acciones de usuario dentro de la aplicación que se aplicarán a roles que no sean de administrador (<b>si están configurados</b>). Si el reclamo no aparece en la respuesta, se denegará el acceso a ABS. Si falta una sola opción, se tratará como <code>falsa</code>. Asegúrese de que la notificación del proveedor de identidades coincida con la estructura esperada:",
|
||||
"LabelOpenIDClaims": "Deje las siguientes opciones vacías para deshabilitar la asignación avanzada de grupos y permisos, lo que asignaría de manera automática al grupo 'Usuario'.",
|
||||
"LabelOpenIDClaims": "Deje las siguientes opciones vacías para desactivar la asignación avanzada de grupos y permisos, lo que asignaría de manera automática al grupo «Usuario».",
|
||||
"LabelOpenIDGroupClaimDescription": "Nombre de la declaración OpenID que contiene una lista de grupos del usuario. Comúnmente conocidos como <code>grupos</code>. <b>Si se configura</b>, la aplicación asignará automáticamente roles en función de la pertenencia a grupos del usuario, siempre que estos grupos se denominen \"admin\", \"user\" o \"guest\" en la notificación. La solicitud debe contener una lista, y si un usuario pertenece a varios grupos, la aplicación asignará el rol correspondiente al mayor nivel de acceso. Si ningún grupo coincide, se denegará el acceso.",
|
||||
"LabelOpenRSSFeed": "Abrir suministro RSS",
|
||||
"LabelOverwrite": "Sobrescribir",
|
||||
@@ -496,7 +499,7 @@
|
||||
"LabelPodcastType": "Tipo de pódcast",
|
||||
"LabelPodcasts": "Pódcast",
|
||||
"LabelPort": "Puerto",
|
||||
"LabelPrefixesToIgnore": "Prefijos para Ignorar (no distingue entre mayúsculas y minúsculas.)",
|
||||
"LabelPrefixesToIgnore": "Prefijos para ignorar (no distingue entre mayúsculas y minúsculas)",
|
||||
"LabelPreventIndexing": "Evite que los directorios de pódcast de iTunes y Google indicen su suministro",
|
||||
"LabelPrimaryEbook": "Libro electrónico principal",
|
||||
"LabelProgress": "Progreso",
|
||||
@@ -512,7 +515,7 @@
|
||||
"LabelRSSFeedCustomOwnerEmail": "Correo electrónico de dueño personalizado",
|
||||
"LabelRSSFeedCustomOwnerName": "Nombre de dueño personalizado",
|
||||
"LabelRSSFeedOpen": "Suministro RSS abierto",
|
||||
"LabelRSSFeedPreventIndexing": "Prevenir indexado",
|
||||
"LabelRSSFeedPreventIndexing": "Evitar indización",
|
||||
"LabelRSSFeedSlug": "«Slug» de suministro RSS",
|
||||
"LabelRSSFeedURL": "URL de suministro RSS",
|
||||
"LabelRandomly": "Aleatorio",
|
||||
@@ -528,6 +531,7 @@
|
||||
"LabelReleaseDate": "Fecha de estreno",
|
||||
"LabelRemoveAllMetadataAbs": "Eliminar todos los archivos metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Eliminar todos los archivos metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Quitar introducción y cierre de Audible de los capítulos",
|
||||
"LabelRemoveCover": "Quitar cubierta",
|
||||
"LabelRemoveMetadataFile": "Eliminar archivos de metadatos en carpetas de elementos de biblioteca",
|
||||
"LabelRemoveMetadataFileHelp": "Elimine todos los archivos metadata.json y metadata.abs de sus carpetas {0}.",
|
||||
@@ -536,7 +540,7 @@
|
||||
"LabelSearchTitle": "Buscar título",
|
||||
"LabelSearchTitleOrASIN": "Buscar título o ASIN",
|
||||
"LabelSeason": "Temporada",
|
||||
"LabelSeasonNumber": "Sesión #{0}",
|
||||
"LabelSeasonNumber": "{0}.ª temporada",
|
||||
"LabelSelectAll": "Seleccionar todo",
|
||||
"LabelSelectAllEpisodes": "Seleccionar todos los episodios",
|
||||
"LabelSelectEpisodesShowing": "Seleccionar los {0} episodios visibles",
|
||||
@@ -576,7 +580,7 @@
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeries": "Saltar libros anteriores de la serie Continuada",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeriesHelp": "El estante de la página de inicio de Continuar Serie muestra el primer libro no iniciado de una serie que tenga por lo menos un libro finalizado y no tenga libros en progreso. Habilitar esta opción le permitirá continuar series desde el último libro que ha completado en vez del primer libro que no ha empezado.",
|
||||
"LabelSettingsParseSubtitles": "Extraer Subtítulos",
|
||||
"LabelSettingsParseSubtitlesHelp": "Extraer subtítulos de los nombres de las carpetas de los audiolibros.<br>Los subtítulos deben estar separados por \" - \"<br>Por ejemplo: \"Ejemplo de Título - Subtítulo Aquí\" tiene el subtítulo \"Subtítulo Aquí\"",
|
||||
"LabelSettingsParseSubtitlesHelp": "Extraer subtítulos de los nombres de las carpetas de los audiolibros.<br>Los subtítulos deben estar separados por « - »<br>Así, «Título de libro - Un subtítulo aquí» tiene el subtítulo «Un subtítulo aquí»",
|
||||
"LabelSettingsPreferMatchedMetadata": "Preferir metadatos encontrados",
|
||||
"LabelSettingsPreferMatchedMetadataHelp": "Los datos encontrados sobreescribirán los detalles del elemento cuando se use \"Encontrar Rápido\". Por defecto, \"Encontrar Rápido\" sólo completará los detalles faltantes.",
|
||||
"LabelSettingsSkipMatchingBooksWithASIN": "Omitir libros coincidentes que ya tengan un ASIN",
|
||||
@@ -594,7 +598,7 @@
|
||||
"LabelShareDownloadableHelp": "Permite a quienes posean el enlace de compartición descargar un archivo ZIP del elemento de la biblioteca.",
|
||||
"LabelShareOpen": "abrir un recurso compartido",
|
||||
"LabelShareURL": "Compartir la URL",
|
||||
"LabelShowAll": "Mostrar Todos",
|
||||
"LabelShowAll": "Mostrar todo",
|
||||
"LabelShowSeconds": "Mostrar segundos",
|
||||
"LabelShowSubtitles": "Mostrar subtítulos",
|
||||
"LabelSize": "Tamaño",
|
||||
@@ -602,33 +606,34 @@
|
||||
"LabelSlug": "Slug",
|
||||
"LabelSortAscending": "Ascendente",
|
||||
"LabelSortDescending": "Descendente",
|
||||
"LabelSortPubDate": "Ord. fecha pub.",
|
||||
"LabelStart": "Iniciar",
|
||||
"LabelStartTime": "Tiempo de Inicio",
|
||||
"LabelStarted": "Iniciado",
|
||||
"LabelStartedAt": "Iniciado En",
|
||||
"LabelStatsAudioTracks": "Pistas de Audio",
|
||||
"LabelStatsAuthors": "Autores",
|
||||
"LabelStatsBestDay": "Mejor Día",
|
||||
"LabelStatsDailyAverage": "Promedio Diario",
|
||||
"LabelStatsBestDay": "Mejor día",
|
||||
"LabelStatsDailyAverage": "Promedio diario",
|
||||
"LabelStatsDays": "Días",
|
||||
"LabelStatsDaysListened": "Días Escuchando",
|
||||
"LabelStatsDaysListened": "Días escuchando",
|
||||
"LabelStatsHours": "Horas",
|
||||
"LabelStatsInARow": "seguidos",
|
||||
"LabelStatsItemsFinished": "Elementos Terminados",
|
||||
"LabelStatsItemsFinished": "Elementos terminados",
|
||||
"LabelStatsItemsInLibrary": "Elementos en biblioteca",
|
||||
"LabelStatsMinutes": "minutos",
|
||||
"LabelStatsMinutesListening": "Minutos Escuchando",
|
||||
"LabelStatsOverallDays": "Total de Dias",
|
||||
"LabelStatsOverallHours": "Total de Horas",
|
||||
"LabelStatsWeekListening": "Tiempo escuchando en la Semana",
|
||||
"LabelStatsMinutesListening": "Minutos escuchando",
|
||||
"LabelStatsOverallDays": "Total de días",
|
||||
"LabelStatsOverallHours": "Total de horas",
|
||||
"LabelStatsWeekListening": "Tiempo escuchando en la semana",
|
||||
"LabelSubtitle": "Subtítulo",
|
||||
"LabelSupportedFileTypes": "Tipos de archivo admitidos",
|
||||
"LabelTag": "Etiqueta",
|
||||
"LabelTags": "Etiquetas",
|
||||
"LabelTagsAccessibleToUser": "Etiquetas Accessibles al Usuario",
|
||||
"LabelTagsNotAccessibleToUser": "Etiquetas no Accesibles al Usuario",
|
||||
"LabelTasks": "Tareas Corriendo",
|
||||
"LabelTextEditorBulletedList": "Lista con viñetas",
|
||||
"LabelTagsAccessibleToUser": "Etiquetas accessibles al usuario",
|
||||
"LabelTagsNotAccessibleToUser": "Etiquetas no accesibles al usuario",
|
||||
"LabelTasks": "Tareas en ejecución",
|
||||
"LabelTextEditorBulletedList": "Lista con bolos",
|
||||
"LabelTextEditorLink": "Enlazar",
|
||||
"LabelTextEditorNumberedList": "Lista numerada",
|
||||
"LabelTextEditorUnlink": "Desenlazar",
|
||||
@@ -681,7 +686,7 @@
|
||||
"LabelUseFullTrack": "Usar pista completa",
|
||||
"LabelUseZeroForUnlimited": "Utilice 0 para ilimitado",
|
||||
"LabelUser": "Usuario",
|
||||
"LabelUsername": "Nombre de Usuario",
|
||||
"LabelUsername": "Nombre de usuario",
|
||||
"LabelValue": "Valor",
|
||||
"LabelVersion": "Versión",
|
||||
"LabelViewBookmarks": "Ver Marcadores",
|
||||
@@ -694,23 +699,27 @@
|
||||
"LabelWeekdaysToRun": "Correr en Días de la Semana",
|
||||
"LabelXBooks": "{0} libros",
|
||||
"LabelXItems": "{0} elementos",
|
||||
"LabelYearReviewHide": "Ocultar Resumen del año",
|
||||
"LabelYearReviewShow": "Resumen del año",
|
||||
"LabelYearReviewHide": "Ocultar resumen del año",
|
||||
"LabelYearReviewShow": "Ver resumen del año",
|
||||
"LabelYourAudiobookDuration": "Duración de tu Audiolibro",
|
||||
"LabelYourBookmarks": "Tus Marcadores",
|
||||
"LabelYourBookmarks": "Sus marcadores",
|
||||
"LabelYourPlaylists": "Tus Listas",
|
||||
"LabelYourProgress": "Su progreso",
|
||||
"MessageAddToPlayerQueue": "Agregar a fila del Reproductor",
|
||||
"MessageAppriseDescription": "Para usar esta función deberás tener <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">la API de Apprise</a> corriendo o una API que maneje los mismos resultados. <br/>La URL de la API de Apprise debe tener la misma ruta de archivos que donde se envían las notificaciones. Por ejemplo: si su API esta en <code>http://192.168.1.1:8337</code> entonces pondría <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Cerciórese de usar el ASIN de la región correcta de Audible, no de Amazon.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Reinicie el servidor tras el guardado para aplicar los cambios de OIDC.",
|
||||
"MessageBackupsDescription": "Los respaldos incluyen: usuarios, el progreso del los usuarios, los detalles de los elementos de la biblioteca, la configuración del servidor y las imágenes en <code>/metadata/items</code> y <code>/metadata/authors</code>. Los Respaldos <strong>NO</strong> incluyen ningún archivo guardado en la carpeta de tu biblioteca.",
|
||||
"MessageBackupsLocationEditNote": "Nota: actualizar la ubicación de la copia de respaldo no moverá ni modificará los respaldos existentes",
|
||||
"MessageBackupsLocationNoEditNote": "Nota: la ubicación de la copia de respaldo se establece a través de una variable de entorno y no se puede cambiar aquí.",
|
||||
"MessageBackupsLocationPathEmpty": "La ruta de la copia de seguridad no puede estar vacía",
|
||||
"MessageBatchEditPopulateMapDetailsAllHelp": "Rellenar campos activados con datos de todos los elementos. Los campos con varios valores se combinarán",
|
||||
"MessageBatchEditPopulateMapDetailsItemHelp": "Rellenar campos de detalles de mapa con datos de este elemento",
|
||||
"MessageBatchQuickMatchDescription": "\"Encontrar Rápido\" tratará de agregar portadas y metadatos faltantes de los elementos seleccionados. Habilite la opción de abajo para que \"Encontrar Rápido\" pueda sobrescribir portadas y/o metadatos existentes.",
|
||||
"MessageBookshelfNoCollections": "Aún no ha hecho ninguna colección",
|
||||
"MessageBookshelfNoCollectionsHelp": "Las colecciones son públicas. Cualquiera que pueda acceder a la biblioteca las podrá ver.",
|
||||
"MessageBookshelfNoRSSFeeds": "Ningún suministro RSS está abierto",
|
||||
"MessageBookshelfNoResultsForFilter": "Ningún Resultado para el filtro \"{0}: {1}\"",
|
||||
"MessageBookshelfNoResultsForFilter": "El filtro «{0}: {1}» no produjo ningún resultado",
|
||||
"MessageBookshelfNoResultsForQuery": "No hay resultados para la consulta",
|
||||
"MessageBookshelfNoSeries": "No tiene ninguna serie",
|
||||
"MessageChapterEndIsAfter": "El final del capítulo es después del final de tu audiolibro",
|
||||
@@ -718,30 +727,31 @@
|
||||
"MessageChapterErrorStartGteDuration": "El tiempo de inicio no es válido: debe ser inferior a la duración del audiolibro",
|
||||
"MessageChapterErrorStartLtPrev": "El tiempo de inicio no es válido: debe ser mayor o igual que el tiempo de inicio del capítulo anterior",
|
||||
"MessageChapterStartIsAfter": "El comienzo del capítulo es después del final de su audiolibro",
|
||||
"MessageChaptersNotFound": "Capítulos no encontrados",
|
||||
"MessageCheckingCron": "Revisando cron...",
|
||||
"MessageConfirmCloseFeed": "¿Confirma que quiere cerrar este suministro?",
|
||||
"MessageConfirmDeleteBackup": "¿Está seguro de que desea eliminar el respaldo {0}?",
|
||||
"MessageConfirmDeleteBackup": "¿Confirma que quiere eliminar el respaldo de {0}?",
|
||||
"MessageConfirmDeleteDevice": "¿Confirma que quiere eliminar el lector electrónico «{0}»?",
|
||||
"MessageConfirmDeleteFile": "Esto eliminará el archivo de su sistema de archivos. ¿Está seguro?",
|
||||
"MessageConfirmDeleteFile": "Esto eliminará el archivo del sistema de archivos. ¿Quiere continuar?",
|
||||
"MessageConfirmDeleteLibrary": "¿Confirma que quiere eliminar permanentemente la biblioteca «{0}»?",
|
||||
"MessageConfirmDeleteLibraryItem": "Esto eliminará el elemento de la biblioteca de la base de datos y del sistema de archivos. ¿Confirma que quiere hacerlo?",
|
||||
"MessageConfirmDeleteLibraryItems": "Esto eliminará {0} elementos de la biblioteca de la base de datos y del sistema de archivos. ¿Confirma que quiere hacerlo?",
|
||||
"MessageConfirmDeleteMetadataProvider": "¿Confirma que quiere eliminar el proveedor de metadatos personalizado «{0}»?",
|
||||
"MessageConfirmDeleteNotification": "¿Confirma que quiere eliminar esta notificación?",
|
||||
"MessageConfirmDeleteSession": "¿Está seguro de que desea eliminar esta sesión?",
|
||||
"MessageConfirmEmbedMetadataInAudioFiles": "¿Está seguro de que desea incrustar metadatos en {0} archivos de audio?",
|
||||
"MessageConfirmForceReScan": "¿Está seguro de que desea forzar un re-escaneo?",
|
||||
"MessageConfirmMarkAllEpisodesFinished": "¿Está seguro de que desea marcar todos los episodios como terminados?",
|
||||
"MessageConfirmMarkAllEpisodesNotFinished": "¿Está seguro de que desea marcar todos los episodios como no terminados?",
|
||||
"MessageConfirmMarkItemFinished": "¿Estás seguro de que deseas marcar \"{0}\" como terminado?",
|
||||
"MessageConfirmMarkItemNotFinished": "¿Estás seguro de que deseas marcar \"{0}\" como no acabado?",
|
||||
"MessageConfirmMarkSeriesFinished": "¿Está seguro de que desea marcar todos los libros en esta serie como terminados?",
|
||||
"MessageConfirmMarkSeriesNotFinished": "¿Está seguro de que desea marcar todos los libros en esta serie como no terminados?",
|
||||
"MessageConfirmDeleteSession": "¿Confirma que quiere eliminar esta sesión?",
|
||||
"MessageConfirmEmbedMetadataInAudioFiles": "¿Confirma que quiere incrustar metadatos en {0} archivos de audio?",
|
||||
"MessageConfirmForceReScan": "¿Confirma que quiere forzar un reescaneo?",
|
||||
"MessageConfirmMarkAllEpisodesFinished": "¿Confirma que quiere marcar todos los episodios como terminados?",
|
||||
"MessageConfirmMarkAllEpisodesNotFinished": "¿Confirma que quiere marcar todos los episodios como no terminados?",
|
||||
"MessageConfirmMarkItemFinished": "¿Confirma que quiere marcar «{0}» como terminado?",
|
||||
"MessageConfirmMarkItemNotFinished": "¿Confirma que quiere marcar «{0}» como no terminado?",
|
||||
"MessageConfirmMarkSeriesFinished": "¿Confirma que quiere marcar todos los libros de esta serie como terminados?",
|
||||
"MessageConfirmMarkSeriesNotFinished": "¿Confirma que quiere marcar todos los libros de esta serie como no terminados?",
|
||||
"MessageConfirmNotificationTestTrigger": "¿Activar esta notificación con datos de prueba?",
|
||||
"MessageConfirmPurgeCache": "Purgar el caché eliminará el directorio completo ubicado en <code>/metadata/cache</code>. <br /><br />¿Está seguro que desea eliminar el directorio del caché?",
|
||||
"MessageConfirmPurgeItemsCache": "Purgar la caché de los elementos eliminará todo el directorio <code>/metadata/cache/items</code>.<br />¿Estás seguro?",
|
||||
"MessageConfirmQuickEmbed": "¡Advertencia! La integración rápida no realiza copias de seguridad a ninguno de tus archivos de audio. Asegúrate de haber realizado una copia de los mismos previamente. <br><br>¿Deseas continuar?",
|
||||
"MessageConfirmQuickMatchEpisodes": "El reconocimiento rápido de extensiones sobrescribirá los detalles si se encuentra una coincidencia. Se actualizarán las extensiones no reconocidas. ¿Está seguro?",
|
||||
"MessageConfirmPurgeCache": "Purgar la antememoria eliminará el directorio completo ubicado en <code>/metadata/cache</code>. <br /><br />¿Confirma que quiere eliminar el directorio de antememoria?",
|
||||
"MessageConfirmPurgeItemsCache": "Purgar la antememoria de elementos eliminará el directorio completo ubicado en <code>/metadata/cache/items</code>.<br />¿Lo confirma?",
|
||||
"MessageConfirmQuickEmbed": "Atención: la incrustación rápida no realiza copias de respaldo a ninguno de sus archivos de audio. Cerciórese de haber realizado una copia de los mismos previamente. <br><br>¿Quiere continuar?",
|
||||
"MessageConfirmQuickMatchEpisodes": "El reconocimiento rápido de extensiones sobrescribirá los detalles si se encuentra una coincidencia. Se actualizarán las extensiones no reconocidas. ¿Quiere continuar?",
|
||||
"MessageConfirmReScanLibraryItems": "¿Confirma que quiere volver a analizar {0} elementos?",
|
||||
"MessageConfirmRemoveAllChapters": "¿Confirma que quiere quitar todos los capítulos?",
|
||||
"MessageConfirmRemoveAuthor": "¿Confirma que quiere quitar el autor «{0}»?",
|
||||
@@ -749,34 +759,35 @@
|
||||
"MessageConfirmRemoveEpisode": "¿Confirma que quiere quitar el episodio «{0}»?",
|
||||
"MessageConfirmRemoveEpisodes": "¿Confirma que quiere quitar {0} episodios?",
|
||||
"MessageConfirmRemoveListeningSessions": "¿Confirma que quiere quitar {0} sesiones de escucha?",
|
||||
"MessageConfirmRemoveMetadataFiles": "¿Está seguro de que desea eliminar todos los archivos de metadatos.{0} en las carpetas de elementos de su biblioteca?",
|
||||
"MessageConfirmRemoveMetadataFiles": "¿Confirma que quiere quitar todos los archivos metadata.{0} en las carpetas de elementos de su biblioteca?",
|
||||
"MessageConfirmRemoveNarrator": "¿Confirma que quiere quitar el narrador «{0}»?",
|
||||
"MessageConfirmRemovePlaylist": "¿Confirma que quiere quitar la lista de reproducción «{0}»?",
|
||||
"MessageConfirmRenameGenre": "¿Está seguro de que desea renombrar el genero \"{0}\" a \"{1}\" de todos los elementos?",
|
||||
"MessageConfirmRenameGenre": "¿Confirma que quiere cambiar el nombre del género «{0}» a «{1}» en todos los elementos?",
|
||||
"MessageConfirmRenameGenreMergeNote": "Nota: Este género ya existe, por lo que se fusionarán.",
|
||||
"MessageConfirmRenameGenreWarning": "Advertencia! Un genero similar ya existe \"{0}\".",
|
||||
"MessageConfirmRenameTag": "¿Está seguro de que desea renombrar la etiqueta \"{0}\" a \"{1}\" de todos los elementos?",
|
||||
"MessageConfirmRenameTagMergeNote": "Nota: Esta etiqueta ya existe, por lo que se fusionarán.",
|
||||
"MessageConfirmRenameTagWarning": "Advertencia! Una etiqueta similar ya existe \"{0}\".",
|
||||
"MessageConfirmRenameGenreWarning": "¡Atención! Ya existe un género similar con distinta mayusculación, «{0}».",
|
||||
"MessageConfirmRenameTag": "¿Confirma que quiere cambiar el nombre de la etiqueta «{0}» a «{1}» en todos los elementos?",
|
||||
"MessageConfirmRenameTagMergeNote": "Nota: esta etiqueta ya existe, por lo que se fusionarán.",
|
||||
"MessageConfirmRenameTagWarning": "¡Atención! Ya existe una etiqueta similar con distinta mayusculación, «{0}».",
|
||||
"MessageConfirmResetProgress": "¿Confirma que quiere restablecer su progreso?",
|
||||
"MessageConfirmSendEbookToDevice": "¿Confirma que quiere enviar el libro electrónico {0} «{1}» al dispositivo «{2}»?",
|
||||
"MessageConfirmUnlinkOpenId": "¿Estás seguro de que deseas desvincular este usuario de OpenID?",
|
||||
"MessageConfirmUnlinkOpenId": "¿Confirma que quiere desenlazar este usuario de OpenID?",
|
||||
"MessageDaysListenedInTheLastYear": "{0} días escuchados el año pasado",
|
||||
"MessageDownloadingEpisode": "Descargando episodio",
|
||||
"MessageDragFilesIntoTrackOrder": "Arrastra los archivos al orden correcto de las pistas",
|
||||
"MessageEmbedFailed": "¡Error al insertar!",
|
||||
"MessageEmbedFinished": "Incrustación Terminada!",
|
||||
"MessageDragFilesIntoTrackOrder": "Arrastre los archivos al orden correcto de las pistas",
|
||||
"MessageEmbedFailed": "Falló la incrustación.",
|
||||
"MessageEmbedFinished": "Finalizó la incrustación.",
|
||||
"MessageEmbedQueue": "En cola para incrustar metadatos ({0} en cola)",
|
||||
"MessageEpisodesQueuedForDownload": "{0} Episodio(s) en cola para descargar",
|
||||
"MessageEpisodesQueuedForDownload": "{0} episodio(s) en cola para descargar",
|
||||
"MessageEreaderDevices": "Para garantizar la entrega de libros electrónicos, es posible que tenga que agregar la dirección de correo electrónico anterior como remitente válido para cada dispositivo enumerado a continuación.",
|
||||
"MessageFeedURLWillBe": "El URL del suministro será {0}",
|
||||
"MessageFetching": "Buscando...",
|
||||
"MessageFetching": "Recuperando...",
|
||||
"MessageForceReScanDescription": "Escaneará todos los archivos como un nuevo escaneo. Archivos de audio con etiquetas ID3, archivos OPF y archivos de texto serán escaneados como nuevos.",
|
||||
"MessageImportantNotice": "¡Notificación importante!",
|
||||
"MessageInsertChapterBelow": "Insertar Capítulo Abajo",
|
||||
"MessageInsertChapterBelow": "Insertar capítulo debajo",
|
||||
"MessageInvalidAsin": "ASIN no válido",
|
||||
"MessageItemsSelected": "{0} elementos seleccionados",
|
||||
"MessageItemsUpdated": "{0} elementos actualizados",
|
||||
"MessageJoinUsOn": "Únetenos en",
|
||||
"MessageJoinUsOn": "Únase a nosotros en",
|
||||
"MessageLoading": "Cargando...",
|
||||
"MessageLoadingFolders": "Cargando archivos...",
|
||||
"MessageLogsDescription": "Logs son almacenados en <code>/metadata/logs</code> en archivos bajo formato JSON. Logs de fallos son almacenados en <code>/metadata/logs/crash_logs.txt</code>.",
|
||||
@@ -800,19 +811,19 @@
|
||||
"MessageNoDownloadsInProgress": "No hay descargas actualmente en curso",
|
||||
"MessageNoDownloadsQueued": "Sin Lista de Descarga",
|
||||
"MessageNoEpisodeMatchesFound": "No se encontraron episodios que coinciden",
|
||||
"MessageNoEpisodes": "Sin Episodios",
|
||||
"MessageNoFoldersAvailable": "No Hay Carpetas Disponibles",
|
||||
"MessageNoGenres": "Sin Géneros",
|
||||
"MessageNoIssues": "Sin Problemas",
|
||||
"MessageNoItems": "Sin elementos",
|
||||
"MessageNoEpisodes": "Ningún episodio",
|
||||
"MessageNoFoldersAvailable": "Ninguna carpeta disponible",
|
||||
"MessageNoGenres": "Ningún género",
|
||||
"MessageNoIssues": "Ningún número",
|
||||
"MessageNoItems": "Ningún elemento",
|
||||
"MessageNoItemsFound": "Ningún elemento encontrado",
|
||||
"MessageNoListeningSessions": "Ninguna sesión escuchada",
|
||||
"MessageNoListeningSessions": "Ninguna sesión de escucha",
|
||||
"MessageNoLogs": "Ningún registro",
|
||||
"MessageNoMediaProgress": "Multimedia sin Progreso",
|
||||
"MessageNoNotifications": "Ninguna notificación",
|
||||
"MessageNoPodcastFeed": "Podcast no válido: Sin feed",
|
||||
"MessageNoPodcastFeed": "Pódcast no válido: no hay suministro",
|
||||
"MessageNoPodcastsFound": "No se encontró ningún pódcast",
|
||||
"MessageNoResults": "Sin Resultados",
|
||||
"MessageNoResults": "Ningún resultado",
|
||||
"MessageNoSearchResultsFor": "La búsqueda «{0}» no produjo ningún resultado",
|
||||
"MessageNoSeries": "Ninguna serie",
|
||||
"MessageNoTags": "Ninguna etiqueta",
|
||||
@@ -820,47 +831,49 @@
|
||||
"MessageNoUpdatesWereNecessary": "No fue necesario actualizar",
|
||||
"MessageNoUserPlaylists": "No tiene ninguna lista de reproducción",
|
||||
"MessageNoUserPlaylistsHelp": "Las listas de reproducción son privadas. Solo quien las cree podrá verlas.",
|
||||
"MessageNotYetImplemented": "Aun no implementado",
|
||||
"MessageNotYetImplemented": "Aún no implementado",
|
||||
"MessageOpmlPreviewNote": "Nota: Esta es una vista previa del archivo OPML analizado. El título real del podcast se obtendrá del canal RSS.",
|
||||
"MessageOr": "o",
|
||||
"MessagePauseChapter": "Pausar la reproducción del capítulo",
|
||||
"MessagePlayChapter": "Escuchar el comienzo del capítulo",
|
||||
"MessagePlaylistCreateFromCollection": "Crear una lista de reproducción a partir de una colección",
|
||||
"MessagePleaseWait": "Por favor, espera...",
|
||||
"MessagePleaseWait": "Espere…",
|
||||
"MessagePodcastHasNoRSSFeedForMatching": "El pódcast no tiene un URL de suministro RSS que pueda usarse para encontrar correspondencias",
|
||||
"MessagePodcastSearchField": "Introduzca el término de búsqueda o el URL del suministro RSS",
|
||||
"MessageQuickEmbedInProgress": "Integración rápida en proceso",
|
||||
"MessageQuickEmbedQueue": "En cola para inserción rápida ({0} en cola)",
|
||||
"MessageQuickMatchAllEpisodes": "Combina rápidamente todos los episodios",
|
||||
"MessageQuickMatchDescription": "Rellenar detalles de elementos vacíos y portada con los primeros resultados de '{0}'. No sobrescribe los detalles a menos que la opción \"Preferir Metadatos Encontrados\" del servidor esté habilitada.",
|
||||
"MessageQuickMatchDescription": "Rellena los detalles y la cubierta de los elementos vacíos con el primer resultado coincidente de «{0}». No sobrescribe los detalles a menos que se active la opción del servidor «Preferir metadatos coincidentes».",
|
||||
"MessageRemoveChapter": "Quitar capítulo",
|
||||
"MessageRemoveEpisodes": "Quitar {0} episodio(s)",
|
||||
"MessageRemoveFromPlayerQueue": "Quitar de la cola de reproducción",
|
||||
"MessageRemoveUserWarning": "¿Confirma que quiere eliminar permanentemente el usuario «{0}»?",
|
||||
"MessageReportBugsAndContribute": "Reporte erres, solicite funciones y contribuya en",
|
||||
"MessageResetChaptersConfirm": "¿Está seguro de que desea deshacer los cambios y revertir los capítulos a su estado original?",
|
||||
"MessageRestoreBackupConfirm": "¿Está seguro de que desea para restaurar del respaldo creado en",
|
||||
"MessageReportBugsAndContribute": "Informe de defectos, solicite funciones y contribuya en",
|
||||
"MessageResetChaptersConfirm": "¿Confirma que quiere deshacer los cambios y restablecer los capítulos a su estado original?",
|
||||
"MessageRestoreBackupConfirm": "¿Confirma que quiere restaurar el respaldo creado el",
|
||||
"MessageRestoreBackupWarning": "Restaurar sobrescribirá toda la base de datos localizada en /config y las imágenes de portadas en /metadata/items y /metadata/authors.<br /><br />El respaldo no modifica ningún archivo en las carpetas de su biblioteca. Si ha habilitado la opción del servidor para almacenar portadas y metadata en las carpetas de su biblioteca, esos archivos no se respaldan o sobrescriben.<br /><br />Todos los clientes que usen su servidor se actualizarán automáticamente.",
|
||||
"MessageScheduleLibraryScanNote": "Para la mayoría de los usuarios, se recomienda dejar esta función desactivada y mantener activada la configuración del observador de carpetas. El observador de carpetas detectará automáticamente los cambios en las carpetas de la biblioteca. El observador de carpetas no funciona para todos los sistemas de archivos (como NFS), por lo que se pueden utilizar exploraciones programadas de la biblioteca en su lugar.",
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Ejecutar cada {0} a las {1}",
|
||||
"MessageSearchResultsFor": "Resultados de la búsqueda de",
|
||||
"MessageSelected": "{0} seleccionado(s)",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "La secuencia de la serie no puede contener espacios",
|
||||
"MessageServerCouldNotBeReached": "No se pudo establecer la conexión con el servidor",
|
||||
"MessageSetChaptersFromTracksDescription": "Establecer capítulos usando cada archivo de audio como un capítulo y el título del capítulo como el nombre del archivo de audio",
|
||||
"MessageShareExpirationWillBe": "La caducidad será <strong>{0}</strong>",
|
||||
"MessageShareExpiresIn": "Caduduca en {0}",
|
||||
"MessageShareURLWillBe": "La URL para compartir será <strong> {0} </strong>",
|
||||
"MessageStartPlaybackAtTime": "Iniciar reproducción para \"{0}\" en {1}?",
|
||||
"MessageTaskAudioFileNotWritable": "El archivo de audio \"{0}\" no se puede grabar",
|
||||
"MessageStartPlaybackAtTime": "¿Iniciar reproducción para «{0}» en {1}?",
|
||||
"MessageTaskAudioFileNotWritable": "El archivo de audio «{0}» no se puede grabar",
|
||||
"MessageTaskCanceledByUser": "Tarea cancelada por el usuario",
|
||||
"MessageTaskDownloadingEpisodeDescription": "Descargando el episodio «{0}»",
|
||||
"MessageTaskEmbeddingMetadata": "Inserción de metadatos",
|
||||
"MessageTaskEmbeddingMetadataDescription": "Inserción de metadatos en el audiolibro \"{0}\"",
|
||||
"MessageTaskEmbeddingMetadataDescription": "Incrustando metadatos en el audiolibro «{0}»",
|
||||
"MessageTaskEncodingM4b": "Codificación M4B",
|
||||
"MessageTaskEncodingM4bDescription": "Codificación del audiolibro \"{0}\" en un único archivo m4b",
|
||||
"MessageTaskEncodingM4bDescription": "Codificando el audiolibro «{0}» en un único archivo m4b",
|
||||
"MessageTaskFailed": "Fallida",
|
||||
"MessageTaskFailedToBackupAudioFile": "Error en la copia de seguridad del archivo de audio \"{0}\"",
|
||||
"MessageTaskFailedToBackupAudioFile": "No se pudo respaldar el archivo de audio «{0}»",
|
||||
"MessageTaskFailedToCreateCacheDirectory": "Error al crear el directorio de la caché",
|
||||
"MessageTaskFailedToEmbedMetadataInFile": "Error al incrustar metadatos en el archivo \"{0}\"",
|
||||
"MessageTaskFailedToEmbedMetadataInFile": "No se pudieron incrustar metadatos en el archivo «{0}»",
|
||||
"MessageTaskFailedToMergeAudioFiles": "Error al fusionar archivos de audio",
|
||||
"MessageTaskFailedToMoveM4bFile": "Error al mover el archivo m4b",
|
||||
"MessageTaskFailedToWriteMetadataFile": "Error al escribir el archivo de metadatos",
|
||||
@@ -870,9 +883,9 @@
|
||||
"MessageTaskOpmlImportDescription": "Creando pódcast a partir de {0} suministros RSS",
|
||||
"MessageTaskOpmlImportFeed": "Feed de importación OPML",
|
||||
"MessageTaskOpmlImportFeedDescription": "Importando el suministro RSS «{0}»",
|
||||
"MessageTaskOpmlImportFeedFailed": "No se puede obtener el podcast",
|
||||
"MessageTaskOpmlImportFeedFailed": "No se pudo obtener el suministro del pódcast",
|
||||
"MessageTaskOpmlImportFeedPodcastDescription": "Creando pódcast «{0}»",
|
||||
"MessageTaskOpmlImportFeedPodcastExists": "Podcast ya existe en la ruta",
|
||||
"MessageTaskOpmlImportFeedPodcastExists": "El pódcast ya existe en la ruta",
|
||||
"MessageTaskOpmlImportFeedPodcastFailed": "No se pudo crear el pódcast",
|
||||
"MessageTaskOpmlImportFinished": "Añadido {0} podcasts",
|
||||
"MessageTaskOpmlParseFailed": "No se pudo analizar el archivo OPML",
|
||||
@@ -882,8 +895,8 @@
|
||||
"MessageTaskScanItemsMissing": "Falta {0}",
|
||||
"MessageTaskScanItemsUpdated": "{0} actualizado",
|
||||
"MessageTaskScanNoChangesNeeded": "No se necesitan cambios",
|
||||
"MessageTaskScanningFileChanges": "Escaneando cambios en el archivo en \"{0}\"",
|
||||
"MessageTaskScanningLibrary": "Escaneando la biblioteca \"{0}\"",
|
||||
"MessageTaskScanningFileChanges": "Escaneando cambios en archivos en «{0}»",
|
||||
"MessageTaskScanningLibrary": "Escaneando la biblioteca «{0}»",
|
||||
"MessageTaskTargetDirectoryNotWritable": "El directorio de destino no se puede escribir",
|
||||
"MessageThinking": "Pensando...",
|
||||
"MessageUploaderItemFailed": "Error al Subir",
|
||||
@@ -933,7 +946,7 @@
|
||||
"ToastAppriseUrlRequired": "Debes ingresar una URL de Apprise",
|
||||
"ToastAsinRequired": "Se requiere ASIN",
|
||||
"ToastAuthorImageRemoveSuccess": "Se eliminó la imagen del autor",
|
||||
"ToastAuthorNotFound": "No se encontró el autor \"{0}\"",
|
||||
"ToastAuthorNotFound": "No se encontró el autor «{0}»",
|
||||
"ToastAuthorRemoveSuccess": "Autor eliminado",
|
||||
"ToastAuthorSearchNotFound": "No se encontró al autor",
|
||||
"ToastAuthorUpdateMerged": "Autor combinado",
|
||||
@@ -944,33 +957,35 @@
|
||||
"ToastBackupCreateSuccess": "Respaldo creado",
|
||||
"ToastBackupDeleteFailed": "Error al eliminar respaldo",
|
||||
"ToastBackupDeleteSuccess": "Respaldo eliminado",
|
||||
"ToastBackupInvalidMaxKeep": "Número no válido de copias de seguridad a conservar",
|
||||
"ToastBackupInvalidMaxSize": "Tamaño máximo de copia de seguridad no válido",
|
||||
"ToastBackupInvalidMaxKeep": "Número no válido de copias de respaldo para conservar",
|
||||
"ToastBackupInvalidMaxSize": "Tamaño máximo de copia de respaldo no válido",
|
||||
"ToastBackupRestoreFailed": "Error al restaurar el respaldo",
|
||||
"ToastBackupUploadFailed": "Error al subir el respaldo",
|
||||
"ToastBackupUploadFailed": "Error al cargar la copia de respaldo",
|
||||
"ToastBackupUploadSuccess": "Respaldo cargado",
|
||||
"ToastBatchApplyDetailsToItemsSuccess": "Detalles aplicados a los elementos",
|
||||
"ToastBatchDeleteFailed": "Error al eliminar por lotes",
|
||||
"ToastBatchDeleteSuccess": "Borrado por lotes correcto",
|
||||
"ToastBatchDeleteFailed": "Falló la eliminación por lotes",
|
||||
"ToastBatchDeleteSuccess": "Se eliminó por lotes correctamente",
|
||||
"ToastBatchQuickMatchFailed": "¡Error en la sincronización rápida por lotes!",
|
||||
"ToastBatchQuickMatchStarted": "¡Se inició el lote de búsqueda rápida de {0} libros!",
|
||||
"ToastBatchUpdateFailed": "Subida masiva fallida",
|
||||
"ToastBatchUpdateSuccess": "Subida masiva exitosa",
|
||||
"ToastBookmarkCreateFailed": "Error al crear marcador",
|
||||
"ToastBookmarkCreateSuccess": "Marcador Agregado",
|
||||
"ToastBatchUpdateFailed": "Falló la actualización por lotes",
|
||||
"ToastBatchUpdateSuccess": "Se actualizó por lotes correctamente",
|
||||
"ToastBookmarkCreateFailed": "No se pudo crear el marcador",
|
||||
"ToastBookmarkCreateSuccess": "Marcador añadido",
|
||||
"ToastBookmarkRemoveSuccess": "Marcador eliminado",
|
||||
"ToastCachePurgeFailed": "Error al purgar el caché",
|
||||
"ToastCachePurgeSuccess": "Caché purgado de manera exitosa",
|
||||
"ToastCachePurgeFailed": "No se pudo purgar la antememoria",
|
||||
"ToastCachePurgeSuccess": "Se purgó la antememoria correctamente",
|
||||
"ToastChaptersHaveErrors": "Los capítulos tienen errores",
|
||||
"ToastChaptersMustHaveTitles": "Los capítulos tienen que tener un título",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Cantidad de desplazamiento no válida. La hora de inicio del último capítulo se extendería más allá de la duración de este audiolibro.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Cantidad de desplazamiento no válida. El primer capítulo tendría una duración cero o negativa y lo sobrescribiría el segundo capítulo. Aumente la duración inicial del segundo capítulo.",
|
||||
"ToastChaptersMustHaveTitles": "Los capítulos deben tener título",
|
||||
"ToastChaptersRemoved": "Capítulos eliminados",
|
||||
"ToastChaptersUpdated": "Capítulos actualizados",
|
||||
"ToastCollectionItemsAddFailed": "Artículo(s) añadido(s) a la colección fallido(s)",
|
||||
"ToastCollectionRemoveSuccess": "Colección quitada",
|
||||
"ToastCollectionUpdateSuccess": "Colección actualizada",
|
||||
"ToastCoverUpdateFailed": "Error al actualizar la cubierta",
|
||||
"ToastDateTimeInvalidOrIncomplete": "Fecha y hora inválidas o incompletas",
|
||||
"ToastDeleteFileFailed": "Error el eliminar archivo",
|
||||
"ToastDateTimeInvalidOrIncomplete": "Fecha y hora no válidas o incompletas",
|
||||
"ToastDeleteFileFailed": "Falló la eliminación del archivo",
|
||||
"ToastDeleteFileSuccess": "Archivo eliminado",
|
||||
"ToastDeviceAddFailed": "Error al añadir dispositivo",
|
||||
"ToastDeviceNameAlreadyExists": "Un libro electrónico ya existe con ese nombre",
|
||||
@@ -1040,7 +1055,7 @@
|
||||
"ToastPodcastCreateSuccess": "Se creó el pódcast correctamente",
|
||||
"ToastPodcastGetFeedFailed": "No se puede obtener el podcast",
|
||||
"ToastPodcastNoEpisodesInFeed": "No se han encontrado episodios en el feed del RSS",
|
||||
"ToastPodcastNoRssFeed": "El podcast no tiene feed RSS",
|
||||
"ToastPodcastNoRssFeed": "El pódcast no tiene suministro RSS",
|
||||
"ToastProgressIsNotBeingSynced": "El progreso no se sincroniza, reinicia la reproducción",
|
||||
"ToastProviderCreatedFailed": "Error al añadir el proveedor",
|
||||
"ToastProviderCreatedSuccess": "Nuevo proveedor añadido",
|
||||
|
||||
@@ -375,7 +375,7 @@
|
||||
"LabelHardDeleteFile": "Kova tiedostojen poisto",
|
||||
"LabelHasEbook": "Sillä on s-kirja",
|
||||
"LabelHasSupplementaryEbook": "Sillä on täydentävän s-kirjan",
|
||||
"LabelHideSubtitles": "Piilota tekstitykset",
|
||||
"LabelHideSubtitles": "Piilota alaotsikot",
|
||||
"LabelHighestPriority": "Korkein etusija",
|
||||
"LabelHost": "Isäntä",
|
||||
"LabelHour": "Tunti",
|
||||
@@ -530,6 +530,7 @@
|
||||
"LabelReleaseDate": "Julkaisupäivä",
|
||||
"LabelRemoveAllMetadataAbs": "Poista kaikki metadata.abs-tiedostot",
|
||||
"LabelRemoveAllMetadataJson": "Poista kaikki metadata.json-tiedostot",
|
||||
"LabelRemoveAudibleBranding": "Poista Audiblen intro ja outro kappaleista",
|
||||
"LabelRemoveCover": "Poista kansikuva",
|
||||
"LabelRemoveMetadataFile": "Poista metatietotiedostot kirjaston kohdekansioista",
|
||||
"LabelRemoveMetadataFileHelp": "Poista kaikki metadata.json- ja metadata.abs-tiedostot {0} kansioistasi.",
|
||||
@@ -577,8 +578,8 @@
|
||||
"LabelSettingsLibraryMarkAsFinishedWhen": "Merkitse mediakohde valmiiksi, kun",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeries": "Ohita aiemmat kirjat Jatka sarjassa",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeriesHelp": "Jatka sarja -kotisivun hyllyssä näkyy ensimmäinen kirja, jota ei ole aloitettu sarjoissa, joissa on vähintään yksi kirja valmiina eikä yhtään kirjaa kesken. Tämän asetuksen ottaminen käyttöön jatkaa sarjaa kauimpana valmistuneesta kirjasta ensimmäisen aloittamattoman kirjan sijaan.",
|
||||
"LabelSettingsParseSubtitles": "Jäsennä tekstitykset",
|
||||
"LabelSettingsParseSubtitlesHelp": "Pura tekstitykset äänikirjojen kansioiden nimistä.<br>Tekstitys on erotettava toisistaan merkillä \"-\"<br>ts. \"Kirjan otsikko - Tekstitys täällä\" on alaotsikko \"Tekstitys täällä\"",
|
||||
"LabelSettingsParseSubtitles": "Jäsennä alaotsikot",
|
||||
"LabelSettingsParseSubtitlesHelp": "Pura alaotsikot äänikirjojen kansioiden nimistä.<br>Tekstitys on erotettava toisistaan merkillä \"-\"<br>ts. \"Kirjan otsikko - Tekstitys täällä\" on alaotsikko \"Tekstitys täällä\"",
|
||||
"LabelSettingsPreferMatchedMetadata": "Pidä mieluummin täsmäävät metatiedot",
|
||||
"LabelSettingsPreferMatchedMetadataHelp": "Täsmäävät tiedot ohittavat kohteen tiedot käytettäessä Pikatäsmäystä. Oletuksena Pikatäsmäys täyttää vain puuttuvat tiedot.",
|
||||
"LabelSettingsSkipMatchingBooksWithASIN": "Ohita täsmäävät kirjat, joilla on jo ASIN",
|
||||
@@ -598,12 +599,13 @@
|
||||
"LabelShareURL": "Jaa URL-osoite",
|
||||
"LabelShowAll": "Näytä kaikki",
|
||||
"LabelShowSeconds": "Näytä sekunnit",
|
||||
"LabelShowSubtitles": "Näytä tekstitykset",
|
||||
"LabelShowSubtitles": "Näytä alaotsikot",
|
||||
"LabelSize": "Koko",
|
||||
"LabelSleepTimer": "Uniajastin",
|
||||
"LabelSlug": "Slug",
|
||||
"LabelSortAscending": "Nouseva",
|
||||
"LabelSortDescending": "Laskeva",
|
||||
"LabelSortPubDate": "Järjestä julkaisupäivän mukaan",
|
||||
"LabelStart": "Aloita",
|
||||
"LabelStartTime": "Aloitusaika",
|
||||
"LabelStarted": "Aloitettu",
|
||||
@@ -623,7 +625,7 @@
|
||||
"LabelStatsOverallDays": "Päivät kokonaisuudessaan",
|
||||
"LabelStatsOverallHours": "Tunnit kokonaisuudessaan",
|
||||
"LabelStatsWeekListening": "Viikon aikana kuunneltu",
|
||||
"LabelSubtitle": "Tekstitys",
|
||||
"LabelSubtitle": "Alaotsikko",
|
||||
"LabelSupportedFileTypes": "Tuetut tiedostotyypit",
|
||||
"LabelTag": "Tägi",
|
||||
"LabelTags": "Tägit",
|
||||
@@ -722,6 +724,7 @@
|
||||
"MessageChapterErrorStartGteDuration": "Epäkelvollinen aloitusaika; on oltava lyhyempi kuin äänikirjan kesto",
|
||||
"MessageChapterErrorStartLtPrev": "Epäkelvollinen aloitusaika; on oltava suurempi tai yhtä suuri kuin edellisen luvun aloitusaika",
|
||||
"MessageChapterStartIsAfter": "Luku alkaa äänikirjan lopun jälkeen",
|
||||
"MessageChaptersNotFound": "Kappaleita ei löydy",
|
||||
"MessageCheckingCron": "Tarkistetaan cronia...",
|
||||
"MessageConfirmCloseFeed": "Oletko varma, että haluat sulkea tämän syötteen?",
|
||||
"MessageConfirmDeleteBackup": "Oletko varma, että haluat poistaa varmuuskopion {0}:lle?",
|
||||
@@ -778,6 +781,7 @@
|
||||
"MessageForceReScanDescription": "skannaa kaikki tiedostot uudelleen kuten uusi tarkistus. Äänitiedoston ID3-tunnisteet, OPF-tiedostot ja tekstitiedostot skannataan uusina.",
|
||||
"MessageImportantNotice": "Tärkeä huomautus!",
|
||||
"MessageInsertChapterBelow": "Syötä luku alle",
|
||||
"MessageInvalidAsin": "Virheellinen ASIN",
|
||||
"MessageItemsSelected": "{0} kohdetta valittu",
|
||||
"MessageItemsUpdated": "{0} kohdetta päivitetty",
|
||||
"MessageJoinUsOn": "Liity meihin",
|
||||
|
||||
@@ -280,7 +280,7 @@
|
||||
"LabelCollections": "Collections",
|
||||
"LabelComplete": "Complet",
|
||||
"LabelConfirmPassword": "Confirmer le mot de passe",
|
||||
"LabelContinueListening": "Continuer la lecture",
|
||||
"LabelContinueListening": "Continuer l'écoute",
|
||||
"LabelContinueReading": "Continuer la lecture",
|
||||
"LabelContinueSeries": "Continuer les séries",
|
||||
"LabelCover": "Couverture",
|
||||
@@ -303,7 +303,7 @@
|
||||
"LabelDiscFromFilename": "Depuis le fichier",
|
||||
"LabelDiscFromMetadata": "Depuis les métadonnées",
|
||||
"LabelDiscover": "Découvrir",
|
||||
"LabelDownload": "Téléchargement",
|
||||
"LabelDownload": "Télécharger",
|
||||
"LabelDownloadNEpisodes": "Télécharger {0} épisode(s)",
|
||||
"LabelDownloadable": "Téléchargeable",
|
||||
"LabelDuration": "Durée",
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Popis za izvođenje",
|
||||
"HeaderPlaylistItems": "Stavke popisa za izvođenje",
|
||||
"HeaderPodcastsToAdd": "Podcasti za dodavanje",
|
||||
"HeaderPresets": "Predlošci postavki",
|
||||
"HeaderPreviewCover": "Pretpregled naslovnice",
|
||||
"HeaderRSSFeedGeneral": "RSS pojedinosti",
|
||||
"HeaderRSSFeedIsOpen": "RSS izvor je otvoren",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Datum izlaska",
|
||||
"LabelRemoveAllMetadataAbs": "Ukloni sve datoteke metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Ukloni sve datoteke metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Ukloni Audibleove najave i odjave iz poglavlja",
|
||||
"LabelRemoveCover": "Ukloni naslovnicu",
|
||||
"LabelRemoveMetadataFile": "Ukloni datoteke s meta-podatcima iz mapa knjižničkih stavki",
|
||||
"LabelRemoveMetadataFileHelp": "Uklanjanje svih datoteka metadata.json i metadata.abs u vaših {0} mapa.",
|
||||
@@ -706,6 +708,7 @@
|
||||
"MessageAddToPlayerQueue": "Dodaj u redoslijed izvođenja",
|
||||
"MessageAppriseDescription": "Da biste se koristili ovom značajkom, treba vam instanca <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API-ja</a> ili API koji može rukovati istom vrstom zahtjeva.<br />The Adresa Apprise API-ja treba biti puna URL putanja za slanje obavijesti, npr. ako vam se API instanca poslužuje na adresi <code>http://192.168.1.1:8337</code> trebate upisati <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Upišite ASIN iz odgovarajuće Audibleove regije, ne s Amazonov.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Ponovno pokrenite poslužitelj da biste primijenili OIDC promjene.",
|
||||
"MessageBackupsDescription": "Sigurnosne kopije sadrže korisnike, korisnikov napredak medija, pojedinosti knjižničke građe, postavke poslužitelja i slike koje se spremaju u <code>/metadata/items</code> & <code>/metadata/authors</code>. Sigurnosne kopije ne sadrže niti jednu datoteku iz mapa knjižnice.",
|
||||
"MessageBackupsLocationEditNote": "Napomena: Uređivanje lokacije za sigurnosne kopije ne premješta ili mijenja postojeće sigurnosne kopije",
|
||||
"MessageBackupsLocationNoEditNote": "Napomena: Lokacija za sigurnosne kopije zadana je kroz varijablu okoline i ovdje se ne može izmijeniti.",
|
||||
@@ -853,6 +856,7 @@
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Pokreni svaki {0} u {1}",
|
||||
"MessageSearchResultsFor": "Rezultati pretrage za",
|
||||
"MessageSelected": "{0} odabrano",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "Slijed serijala ne može sadržavati praznine",
|
||||
"MessageServerCouldNotBeReached": "Nije moguće pristupiti poslužitelju",
|
||||
"MessageSetChaptersFromTracksDescription": "Postavi poglavlja koristeći se zvučnom datotekom kao poglavljem i nazivom datoteke kao naslovom poglavlja",
|
||||
"MessageShareExpirationWillBe": "Vrijeme isteka će biti <strong>{0}</strong>",
|
||||
@@ -971,6 +975,8 @@
|
||||
"ToastCachePurgeFailed": "Čišćenje predmemorije nije uspjelo",
|
||||
"ToastCachePurgeSuccess": "Predmemorija uspješno očišćena",
|
||||
"ToastChaptersHaveErrors": "Poglavlja imaju pogreške",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Neispravna vrijednost pomaka. Početak zadnjeg poglavlja bio bi nakon duljine trajanja ove zvučne knjige.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Neispravna vrijednost pomaka. Trajanje prvog poglavlja bilo bi nula ili negativno i drugo poglavlje bi ga prepisalo. Povećajte vrijeme početka drugog poglavlja.",
|
||||
"ToastChaptersMustHaveTitles": "Poglavlja moraju imati naslove",
|
||||
"ToastChaptersRemoved": "Poglavlja uklonjena",
|
||||
"ToastChaptersUpdated": "Poglavlja su ažurirana",
|
||||
|
||||
@@ -13,7 +13,7 @@
|
||||
"ButtonBatchEditPopulateFromExisting": "Popola da esistente",
|
||||
"ButtonBatchEditPopulateMapDetails": "Inserisci i dettagli della mappa",
|
||||
"ButtonBrowseForFolder": "Per Cartella",
|
||||
"ButtonCancel": "Cancella",
|
||||
"ButtonCancel": "Annulla",
|
||||
"ButtonCancelEncode": "Ferma la codifica",
|
||||
"ButtonChangeRootPassword": "Cambia la Password di root",
|
||||
"ButtonCheckAndDownloadNewEpisodes": "Controlla & scarica i nuovi episodi",
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Playlist",
|
||||
"HeaderPlaylistItems": "Elementi della playlist",
|
||||
"HeaderPodcastsToAdd": "Podcasts da Aggiungere",
|
||||
"HeaderPresets": "Presets",
|
||||
"HeaderPreviewCover": "Anteprima Cover",
|
||||
"HeaderRSSFeedGeneral": "Dettagli RSS",
|
||||
"HeaderRSSFeedIsOpen": "RSS Feed è aperto",
|
||||
@@ -229,6 +230,7 @@
|
||||
"LabelAddedDate": "Aggiunti {0}",
|
||||
"LabelAdminUsersOnly": "Solo utenti Amministratori",
|
||||
"LabelAll": "Tutti",
|
||||
"LabelAllEpisodesDownloaded": "Tutti gli Episodi Scaricati",
|
||||
"LabelAllUsers": "Tutti gli Utenti",
|
||||
"LabelAllUsersExcludingGuests": "Tutti gli Utenti Esclusi gli ospiti",
|
||||
"LabelAllUsersIncludingGuests": "Tutti gli Utenti Inclusi gli ospiti",
|
||||
@@ -252,7 +254,7 @@
|
||||
"LabelBackToUser": "Torna a Utenti",
|
||||
"LabelBackupAudioFiles": "Backup file Audio",
|
||||
"LabelBackupLocation": "Percorso del Backup",
|
||||
"LabelBackupsEnableAutomaticBackups": "Abilita backup Automatico",
|
||||
"LabelBackupsEnableAutomaticBackups": "Backup Automatico",
|
||||
"LabelBackupsEnableAutomaticBackupsHelp": "I Backup saranno salvati in /metadata/backups",
|
||||
"LabelBackupsMaxBackupSize": "Dimensione massima backup (in GB) (0 Illimitato)",
|
||||
"LabelBackupsMaxBackupSizeHelp": "Come protezione contro gli errori di config, i backup falliranno se superano la dimensione configurata.",
|
||||
@@ -279,12 +281,12 @@
|
||||
"LabelCollections": "Raccolte",
|
||||
"LabelComplete": "Completo",
|
||||
"LabelConfirmPassword": "Conferma Password",
|
||||
"LabelContinueListening": "Continua ad Ascoltare",
|
||||
"LabelContinueReading": "Continua la Lettura",
|
||||
"LabelContinueListening": "Continua l'ascolto",
|
||||
"LabelContinueReading": "Continua la lettura",
|
||||
"LabelContinueSeries": "Continua serie",
|
||||
"LabelCover": "Copertina",
|
||||
"LabelCoverImageURL": "Indirizzo della cover URL",
|
||||
"LabelCoverProvider": "Cover Provider",
|
||||
"LabelCoverProvider": "Cover Sorgente",
|
||||
"LabelCreatedAt": "Creato A",
|
||||
"LabelCronExpression": "Espressione Cron",
|
||||
"LabelCurrent": "Attuale",
|
||||
@@ -529,6 +531,7 @@
|
||||
"LabelReleaseDate": "Data Release",
|
||||
"LabelRemoveAllMetadataAbs": "Remuovi tutti i metadata.abs files",
|
||||
"LabelRemoveAllMetadataJson": "Rimuovi tutti i metadata.json files",
|
||||
"LabelRemoveAudibleBranding": "Rimuovi l'intro e il termine Audible dai capitoli",
|
||||
"LabelRemoveCover": "Rimuovi cover",
|
||||
"LabelRemoveMetadataFile": "Rimuovi i file metadata nella cartella della libreria",
|
||||
"LabelRemoveMetadataFileHelp": "Rimuovi tutti i file metadata.json e i file metadata.abs nelle tue {0} cartelle.",
|
||||
@@ -558,6 +561,8 @@
|
||||
"LabelSettingsBookshelfViewHelp": "Design con scaffali in legno",
|
||||
"LabelSettingsChromecastSupport": "Supporto a Chromecast",
|
||||
"LabelSettingsDateFormat": "Formato Data",
|
||||
"LabelSettingsEnableWatcher": "Scansiona le librerie Automaticamente per trovare modifiche",
|
||||
"LabelSettingsEnableWatcherForLibrary": "Scansiona la libreria Automaticamente per trovare modifiche",
|
||||
"LabelSettingsEnableWatcherHelp": "Abilita l'aggiunta/aggiornamento automatico degli elementi quando vengono rilevate modifiche ai file. *Richiede il riavvio del Server",
|
||||
"LabelSettingsEpubsAllowScriptedContent": "Consenti contenuti con script negli epub",
|
||||
"LabelSettingsEpubsAllowScriptedContentHelp": "Consenti ai file epub di eseguire script. Si consiglia di mantenere questa impostazione disabilitata a meno che non si ritenga attendibile l'origine dei file epub.",
|
||||
@@ -601,6 +606,7 @@
|
||||
"LabelSlug": "Lento",
|
||||
"LabelSortAscending": "Crescente",
|
||||
"LabelSortDescending": "Discendente",
|
||||
"LabelSortPubDate": "Ordina per data di pubblicazione",
|
||||
"LabelStart": "Inizo",
|
||||
"LabelStartTime": "Tempo di inizio",
|
||||
"LabelStarted": "Iniziato",
|
||||
@@ -672,7 +678,7 @@
|
||||
"LabelUpdateDetailsHelp": "Consenti la sovrascrittura dei dettagli esistenti per i libri selezionati quando viene individuata una corrispondenza",
|
||||
"LabelUpdatedAt": "Aggiornato alle",
|
||||
"LabelUploaderDragAndDrop": "Drag & drop file o Cartelle",
|
||||
"LabelUploaderDragAndDropFilesOnly": "Drag & drop files",
|
||||
"LabelUploaderDragAndDropFilesOnly": "Drag & drop file",
|
||||
"LabelUploaderDropFiles": "Elimina file",
|
||||
"LabelUploaderItemFetchMetadataHelp": "Recupera automaticamente titolo, autore e serie",
|
||||
"LabelUseAdvancedOptions": "Usa le opzioni avanzate",
|
||||
@@ -701,6 +707,7 @@
|
||||
"LabelYourProgress": "Completato al",
|
||||
"MessageAddToPlayerQueue": "Aggiungi alla coda di riproduzione",
|
||||
"MessageAppriseDescription": "Per utilizzare questa funzione è necessario disporre di un'istanza di <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> in esecuzione o un'API che gestirà quelle stesse richieste. <br />L'API Url dovrebbe essere il percorso URL completo per inviare la notifica, ad esempio se la tua istanza API è servita cosi .<code>http://192.168.1.1:8337</code> Allora dovrai mettere <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Assicurati di utilizzare l'ASIN della regione Audible corretta, non di Amazon.",
|
||||
"MessageBackupsDescription": "I backup includono utenti, progressi degli utenti, dettagli sugli elementi della libreria, impostazioni del server e immagini archiviate in <code>/metadata/items</code> & <code>/metadata/authors</code>. I backup non includono i file archiviati nelle cartelle della libreria.",
|
||||
"MessageBackupsLocationEditNote": "Nota: l'aggiornamento della posizione di backup non sposterà o modificherà i backup esistenti",
|
||||
"MessageBackupsLocationNoEditNote": "Nota: la posizione del backup viene impostata tramite una variabile di ambiente e non può essere modificata qui.",
|
||||
@@ -719,6 +726,7 @@
|
||||
"MessageChapterErrorStartGteDuration": "L'ora di inizio non valida deve essere inferiore alla durata dell'audiolibro",
|
||||
"MessageChapterErrorStartLtPrev": "L'ora di inizio non valida deve essere maggiore o uguale all'ora di inizio del capitolo precedente",
|
||||
"MessageChapterStartIsAfter": "L'inizio del capitolo è dopo la fine del tuo audiolibro",
|
||||
"MessageChaptersNotFound": "Capitoli non trovati",
|
||||
"MessageCheckingCron": "Controllo cron...",
|
||||
"MessageConfirmCloseFeed": "Sei sicuro di voler chiudere questo feed?",
|
||||
"MessageConfirmDeleteBackup": "Sei sicuro di voler eliminare il backup {0}?",
|
||||
@@ -775,8 +783,9 @@
|
||||
"MessageForceReScanDescription": "eseguirà nuovamente la scansione di tutti i file come una nuova scansione. I tag ID3 dei file audio, i file OPF e i file di testo verranno scansionati come nuovi.",
|
||||
"MessageImportantNotice": "Avviso Importante!",
|
||||
"MessageInsertChapterBelow": "Inserisci capitolo sotto",
|
||||
"MessageItemsSelected": "{0} oggetti Selezionati",
|
||||
"MessageItemsUpdated": "{0} Oggetti aggiornati",
|
||||
"MessageInvalidAsin": "ASIN non Valido",
|
||||
"MessageItemsSelected": "{0} oggetti selezionati",
|
||||
"MessageItemsUpdated": "{0} oggetti aggiornati",
|
||||
"MessageJoinUsOn": "Unisciti a noi su",
|
||||
"MessageLoading": "Caricamento…",
|
||||
"MessageLoadingFolders": "Caricamento Cartelle...",
|
||||
@@ -808,7 +817,7 @@
|
||||
"MessageNoItems": "Nessun oggetto",
|
||||
"MessageNoItemsFound": "Nessun oggetto trovato",
|
||||
"MessageNoListeningSessions": "Nessuna sessione di ascolto",
|
||||
"MessageNoLogs": "Nessun Logs",
|
||||
"MessageNoLogs": "Nessun Log",
|
||||
"MessageNoMediaProgress": "Nessun progresso multimediale",
|
||||
"MessageNoNotifications": "Nessuna notifica",
|
||||
"MessageNoPodcastFeed": "Podcast non valido: nessun feed",
|
||||
@@ -843,6 +852,7 @@
|
||||
"MessageRestoreBackupConfirm": "Sei sicuro di voler ripristinare il backup creato su",
|
||||
"MessageRestoreBackupWarning": "Il ripristino di un backup sovrascriverà l'intero database situato in /config e sovrascrive le immagini in /metadata/items & /metadata/authors.<br /><br />I backup non modificano alcun file nelle cartelle della libreria. Se hai abilitato le impostazioni del server per archiviare copertine e metadati nelle cartelle della libreria, questi non vengono sottoposti a backup o sovrascritti.<br /><br />Tutti i client che utilizzano il tuo server verranno aggiornati automaticamente.",
|
||||
"MessageScheduleLibraryScanNote": "Per la maggior parte degli utenti, si consiglia di lasciare questa funzionalità disabilitata e di mantenere abilitata l'impostazione di folder watcher. Il folder watcher rileverà automaticamente le modifiche nelle cartelle della libreria. Il folder watcher non funziona per ogni file system (come NFS), quindi è possibile utilizzare le scansioni pianificate della libreria.",
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Esegui ogni {0} alle {1}",
|
||||
"MessageSearchResultsFor": "cerca risultati per",
|
||||
"MessageSelected": "{0} selezionati",
|
||||
"MessageServerCouldNotBeReached": "Impossibile raggiungere il server",
|
||||
@@ -950,6 +960,7 @@
|
||||
"ToastBackupRestoreFailed": "Ripristino fallito",
|
||||
"ToastBackupUploadFailed": "Caricamento backup fallito",
|
||||
"ToastBackupUploadSuccess": "Backup caricato",
|
||||
"ToastBatchApplyDetailsToItemsSuccess": "Dettagli applicati agli articoli",
|
||||
"ToastBatchDeleteFailed": "Eliminazione batch non riuscita",
|
||||
"ToastBatchDeleteSuccess": "Eliminazione batch riuscita",
|
||||
"ToastBatchQuickMatchFailed": "Batch Quick Match non riuscito!",
|
||||
@@ -962,6 +973,8 @@
|
||||
"ToastCachePurgeFailed": "Impossibile eliminare la cache",
|
||||
"ToastCachePurgeSuccess": "Cache eliminata correttamente",
|
||||
"ToastChaptersHaveErrors": "I capitoli contengono errori",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Quantità di spostamento non valida. L'orario di inizio dell'ultimo capitolo si estenderebbe oltre la durata di questo audiolibro.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Quantità di spostamento non valida. Il primo capitolo avrebbe una lunghezza pari a zero o negativa e verrebbe sovrascritto dal secondo capitolo. Aumentare la durata iniziale del secondo capitolo.",
|
||||
"ToastChaptersMustHaveTitles": "I capitoli devono avere titoli",
|
||||
"ToastChaptersRemoved": "Capitoli rimossi",
|
||||
"ToastChaptersUpdated": "Capitoli aggiornati",
|
||||
@@ -1062,6 +1075,7 @@
|
||||
"ToastSelectAtLeastOneUser": "Seleziona almeno un utente",
|
||||
"ToastSendEbookToDeviceFailed": "Impossibile inviare il libro al dispositivo",
|
||||
"ToastSendEbookToDeviceSuccess": "Libro inviato al dispositivo «{0}»",
|
||||
"ToastSeriesSubmitFailedSameName": "Non è possibile aggiungere due serie con lo stesso nome",
|
||||
"ToastSeriesUpdateFailed": "Aggiornamento Serie Fallito",
|
||||
"ToastSeriesUpdateSuccess": "Serie Aggiornate",
|
||||
"ToastServerSettingsUpdateSuccess": "Impostazioni del server aggiornate",
|
||||
@@ -1080,6 +1094,8 @@
|
||||
"ToastUnknownError": "Errore sconosciuto",
|
||||
"ToastUnlinkOpenIdFailed": "Impossibile scollegare l'utente da OpenID",
|
||||
"ToastUnlinkOpenIdSuccess": "Utente scollegato da OpenID",
|
||||
"ToastUploaderFilepathExistsError": "Il percorso file \"{0}\" esiste già sul server",
|
||||
"ToastUploaderItemExistsInSubdirectoryError": "L'elemento \"{0}\" utilizza una sottodirectory del percorso di caricamento.",
|
||||
"ToastUserDeleteFailed": "Errore eliminazione utente",
|
||||
"ToastUserDeleteSuccess": "Utente eliminato",
|
||||
"ToastUserPasswordChangeSuccess": "Password modificata con successo",
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Playlista",
|
||||
"HeaderPlaylistItems": "Pozycje listy odtwarzania",
|
||||
"HeaderPodcastsToAdd": "Podcasty do dodania",
|
||||
"HeaderPresets": "Ustawienia wstępne",
|
||||
"HeaderPreviewCover": "Podgląd okładki",
|
||||
"HeaderRSSFeedGeneral": "Szczegóły RSS",
|
||||
"HeaderRSSFeedIsOpen": "Kanał RSS jest otwarty",
|
||||
@@ -219,6 +220,7 @@
|
||||
"LabelAccountTypeAdmin": "Administrator",
|
||||
"LabelAccountTypeGuest": "Gość",
|
||||
"LabelAccountTypeUser": "Użytkownik",
|
||||
"LabelActivities": "Aktywności",
|
||||
"LabelActivity": "Aktywność",
|
||||
"LabelAddToCollection": "Dodaj do kolekcji",
|
||||
"LabelAddToCollectionBatch": "Dodaj {0} książki do kolekcji",
|
||||
@@ -228,6 +230,7 @@
|
||||
"LabelAddedDate": "Dodano {0}",
|
||||
"LabelAdminUsersOnly": "Tylko użytkownicy administracyjni",
|
||||
"LabelAll": "Wszystkie",
|
||||
"LabelAllEpisodesDownloaded": "Wszystkie odcinki pobrane",
|
||||
"LabelAllUsers": "Wszyscy użytkownicy",
|
||||
"LabelAllUsersExcludingGuests": "Wszyscy użytkownicy z wyłączeniem gości",
|
||||
"LabelAllUsersIncludingGuests": "Wszyscy użytkownicy, łącznie z gośćmi",
|
||||
@@ -245,6 +248,7 @@
|
||||
"LabelAutoFetchMetadata": "Automatycznie pobierz metadane",
|
||||
"LabelAutoFetchMetadataHelp": "Pobiera metadane dotyczące tytułu, autora i serii, aby usprawnić przesyłanie. Po przesłaniu może być konieczne dopasowanie dodatkowych metadanych.",
|
||||
"LabelAutoLaunch": "Uruchom automatycznie",
|
||||
"LabelAutoLaunchDescription": "Automatyczne przekierowanie do dostawcy uwierzytelniania podczas przechodzenia na stronę logowania (ręczna zamiana ścieżki <code>/login?autoLaunch=0</code>)",
|
||||
"LabelAutoRegister": "Automatyczna rejestracja",
|
||||
"LabelAutoRegisterDescription": "Automatycznie utwórz nowych użytkowników po zalogowaniu",
|
||||
"LabelBackToUser": "Powrót",
|
||||
@@ -282,6 +286,7 @@
|
||||
"LabelContinueSeries": "Kontynuuj serię",
|
||||
"LabelCover": "Okładka",
|
||||
"LabelCoverImageURL": "URL okładki",
|
||||
"LabelCoverProvider": "Dostawca okładki",
|
||||
"LabelCreatedAt": "Utworzone",
|
||||
"LabelCronExpression": "Wyrażenie CRON",
|
||||
"LabelCurrent": "Aktualny",
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Плейлист",
|
||||
"HeaderPlaylistItems": "Элементы списка воспроизведения",
|
||||
"HeaderPodcastsToAdd": "Подкасты для добавления",
|
||||
"HeaderPresets": "Пресеты",
|
||||
"HeaderPreviewCover": "Предпросмотр обложки",
|
||||
"HeaderRSSFeedGeneral": "Сведения о RSS",
|
||||
"HeaderRSSFeedIsOpen": "RSS-канал открыт",
|
||||
@@ -219,7 +220,7 @@
|
||||
"LabelAccountTypeAdmin": "Администратор",
|
||||
"LabelAccountTypeGuest": "Гость",
|
||||
"LabelAccountTypeUser": "Пользователь",
|
||||
"LabelActivities": "Мероприятия",
|
||||
"LabelActivities": "События",
|
||||
"LabelActivity": "Активность",
|
||||
"LabelAddToCollection": "Добавить в коллекцию",
|
||||
"LabelAddToCollectionBatch": "Добавить {0} книг в коллекцию",
|
||||
@@ -306,7 +307,7 @@
|
||||
"LabelDownload": "Скачать",
|
||||
"LabelDownloadNEpisodes": "Скачать {0} эпизодов",
|
||||
"LabelDownloadable": "Загружаемый",
|
||||
"LabelDuration": "Продолжительность",
|
||||
"LabelDuration": "Длительность",
|
||||
"LabelDurationComparisonExactMatch": "(точное совпадение)",
|
||||
"LabelDurationComparisonLonger": "({0} дольше)",
|
||||
"LabelDurationComparisonShorter": "({0} короче)",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Дата выхода",
|
||||
"LabelRemoveAllMetadataAbs": "Удалите все файлы metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Удалите все файлы metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Удалить вступление и концовку Audible из глав",
|
||||
"LabelRemoveCover": "Удалить обложку",
|
||||
"LabelRemoveMetadataFile": "Удаление файлов метаданных в папках элементов библиотеки",
|
||||
"LabelRemoveMetadataFileHelp": "Удалите все файлы metadata.json и metadata.abs из ваших папок {0}.",
|
||||
@@ -706,6 +708,7 @@
|
||||
"MessageAddToPlayerQueue": "Добавить в очередь проигрывателя",
|
||||
"MessageAppriseDescription": "Для использования этой функции необходимо иметь запущенный экземпляр <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> или api которое обрабатывает те же самые запросы. <br />URL-адрес API Apprise должен быть полным URL-адресом для отправки уведомления, т.е., если API запущено по адресу <code>http://192.168.1.1:8337</code> тогда нужно указать <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Убедитесь, что вы используете ASIN из правильной региональной зоны Audible, а не из Amazon.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Перезапустите ваш сервер после сохранения для применения изменений в OIDC.",
|
||||
"MessageBackupsDescription": "Бэкап включает пользователей, прогресс пользователей, данные элементов библиотеки, настройки сервера и изображения хранящиеся в <code>/metadata/items</code> и <code>/metadata/authors</code>. Бэкапы <strong>НЕ</strong> сохраняют файлы из папок библиотек.",
|
||||
"MessageBackupsLocationEditNote": "Примечание: Обновление местоположения резервной копии не приведет к перемещению или изменению существующих резервных копий",
|
||||
"MessageBackupsLocationNoEditNote": "Примечание: Местоположение резервного копирования задается с помощью переменной среды и не может быть изменено здесь.",
|
||||
@@ -971,6 +974,8 @@
|
||||
"ToastCachePurgeFailed": "Не удалось очистить кэш",
|
||||
"ToastCachePurgeSuccess": "Кэш успешно очищен",
|
||||
"ToastChaptersHaveErrors": "Главы имеют ошибки",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Некорректное значение сдвига. Начало последней главы будет превышать продолжительность этой аудиокниги.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Некорректное значение сдвига. Первая глава будет иметь нулевую или отрицательную длину и будет перезаписана второй главой. Увеличьте начальную продолжительность второй главы.",
|
||||
"ToastChaptersMustHaveTitles": "Главы должны содержать названия",
|
||||
"ToastChaptersRemoved": "Удалены главы",
|
||||
"ToastChaptersUpdated": "Обновленные главы",
|
||||
|
||||
@@ -22,7 +22,7 @@
|
||||
"ButtonClearFilter": "Zrušiť filter",
|
||||
"ButtonCloseFeed": "Zatvoriť zdroj",
|
||||
"ButtonCloseSession": "Ukončiť otvorené pripojenie",
|
||||
"ButtonCollections": "Zbierky",
|
||||
"ButtonCollections": "Kolekcie",
|
||||
"ButtonConfigureScanner": "Nastaviť skener",
|
||||
"ButtonCreate": "Vytvoriť",
|
||||
"ButtonCreateBackup": "Vytvoriť zálohu",
|
||||
@@ -56,7 +56,7 @@
|
||||
"ButtonOk": "OK",
|
||||
"ButtonOpenFeed": "Otvoriť zdroj",
|
||||
"ButtonOpenManager": "Otvoriť správcu",
|
||||
"ButtonPause": "Zastaviť",
|
||||
"ButtonPause": "Pozastaviť",
|
||||
"ButtonPlay": "Prehrať",
|
||||
"ButtonPlayAll": "Prehrať všetko",
|
||||
"ButtonPlaying": "Prehráva sa",
|
||||
@@ -92,7 +92,7 @@
|
||||
"ButtonScanLibrary": "Skenovať knižnicu",
|
||||
"ButtonScrollLeft": "Doľava",
|
||||
"ButtonScrollRight": "Doprava",
|
||||
"ButtonSearch": "Vyhľadávanie",
|
||||
"ButtonSearch": "Vyhľadať",
|
||||
"ButtonSelectFolderPath": "Vybrať umiestnenie priečinku",
|
||||
"ButtonSeries": "Série",
|
||||
"ButtonSetChaptersFromTracks": "Nastaviť kapitoly podľa stôp",
|
||||
@@ -104,7 +104,7 @@
|
||||
"ButtonStats": "Štatistiky",
|
||||
"ButtonSubmit": "Odoslať",
|
||||
"ButtonTest": "Test",
|
||||
"ButtonUnlinkOpenId": "Odhlásiť OpenID",
|
||||
"ButtonUnlinkOpenId": "Zrušiť prepojenie OpenID",
|
||||
"ButtonUpload": "Nahrať",
|
||||
"ButtonUploadBackup": "Nahrať zálohu",
|
||||
"ButtonUploadCover": "Nahrať prebal",
|
||||
@@ -127,7 +127,7 @@
|
||||
"HeaderChangePassword": "Zmeniť heslo",
|
||||
"HeaderChapters": "Kapitoly",
|
||||
"HeaderChooseAFolder": "Vybrať priečinok",
|
||||
"HeaderCollection": "Zbierka",
|
||||
"HeaderCollection": "Zbierky",
|
||||
"HeaderCollectionItems": "Položky zbierky",
|
||||
"HeaderCover": "Prebal",
|
||||
"HeaderCurrentDownloads": "Aktuálne sťahovanie",
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Playlist",
|
||||
"HeaderPlaylistItems": "Položky playlistu",
|
||||
"HeaderPodcastsToAdd": "Podcasty na pridanie",
|
||||
"HeaderPresets": "Predvolené",
|
||||
"HeaderPreviewCover": "Ukážka prebalu",
|
||||
"HeaderRSSFeedGeneral": "Detaily RSS",
|
||||
"HeaderRSSFeedIsOpen": "RSS zdroj je otvorený",
|
||||
@@ -210,7 +211,7 @@
|
||||
"HeaderUpdateLibrary": "Aktualizovať knižnicu",
|
||||
"HeaderUsers": "Užívatelia",
|
||||
"HeaderYearReview": "Prehľad roka {0}",
|
||||
"HeaderYourStats": "Tvoje štatistiky",
|
||||
"HeaderYourStats": "Vaše štatistiky",
|
||||
"LabelAbridged": "Skrátená verzia",
|
||||
"LabelAbridgedChecked": "Skrátená verzia (zaškrtnuté)",
|
||||
"LabelAbridgedUnchecked": "Neskrátená verzia (nezaškrtnuté)",
|
||||
@@ -221,7 +222,7 @@
|
||||
"LabelAccountTypeUser": "Užívateľ",
|
||||
"LabelActivities": "Aktivity",
|
||||
"LabelActivity": "Aktivita",
|
||||
"LabelAddToCollection": "Pridať do kolekcie",
|
||||
"LabelAddToCollection": "Pridať do zbierky",
|
||||
"LabelAddToCollectionBatch": "Pridať {0} kníh do kolekcie",
|
||||
"LabelAddToPlaylist": "Pridať do playlistu",
|
||||
"LabelAddToPlaylistBatch": "Pridať {0} položie do playlistu",
|
||||
@@ -240,7 +241,7 @@
|
||||
"LabelAudioChannels": "Počet kanálov audio stopy (1 alebo 2)",
|
||||
"LabelAudioCodec": "Kodek audio stopy",
|
||||
"LabelAuthor": "Autor",
|
||||
"LabelAuthorFirstLast": "Autor (Meno, Priezvisko)",
|
||||
"LabelAuthorFirstLast": "Autor (Meno Priezvisko)",
|
||||
"LabelAuthorLastFirst": "Autor (Priezvisko, Meno)",
|
||||
"LabelAuthors": "Autori",
|
||||
"LabelAutoDownloadEpisodes": "Automaticky sťahovať epizódy",
|
||||
@@ -469,7 +470,7 @@
|
||||
"LabelNotificationsMaxQueueSize": "Maximálna dĺžka fronty oznámení",
|
||||
"LabelNotificationsMaxQueueSizeHelp": "Odosielanie udalostí je ohraničené na jedno oznámenie za sekundu. Novovzniknuté udalosti budú ignorované, ak bude fronta oznámení naplnená. Toto nastavenie zabraňuje nevyžiadanému zahlteniu oznámeniami.",
|
||||
"LabelNumberOfBooks": "Počet kníh",
|
||||
"LabelNumberOfEpisodes": "# epizód",
|
||||
"LabelNumberOfEpisodes": "# z epizód",
|
||||
"LabelOpenIDAdvancedPermsClaimDescription": "Názov OpenID predpokladá prítomnosť pokročilých povolení pre užívateľské akcie v rámci aplikácie, ktoré sú aplikovateľné na ne-administrátorské role (<b>ak sú nakonfigurované</b>). Ak potvrdenie takýchto pokročilých povolení nie je v odozve prítomné, prístup do ABS bude automaticky zamietnutý. Ak v odozve chýba len niektoré z očakávaných nastavení, tak bude jeho hodnota automaticky nastavená na <code>false</code>. Uistite sa prosím, že forma odozvy poskytovateľa identity má nasledovnú štruktúru:",
|
||||
"LabelOpenIDClaims": "Ak ponecháte nasledujúce nastavenia prázdne, pokročilé nastavenia skupín a povolení nebudú aktivované a automaticky bude nastavená skupina 'Užívateľ'.",
|
||||
"LabelOpenIDGroupClaimDescription": "Pri názve požiadavky OpenID sa predpokladá, že obsahuje zoznam užívateľských skupín. Bežne označovaný ako <code>groups</code>. <b>Ak je správne nakonfigurovaný</b>, aplikácia automaticky pridelí role podľa príslušnosti k užívateľským skupinám pod podmienkou, že sú tieto skupiny v požiadavke nazvané (bez ohľadu na veľkosť písmen) ako 'admin', 'user' alebo 'guest'. Požiadavka musí obsahovať zoznam skupín a ak užívateľ patrí do viacerých skupín, aplikácia mu priradí rolu, ktorá zodpovedá skupine s najvyššími prístupovými právami. Ak sa žiadna z poskytnutých skupín nezhoduje, prístup bude zamietnutý.",
|
||||
@@ -499,12 +500,12 @@
|
||||
"LabelPodcasts": "Podcasty",
|
||||
"LabelPort": "Prístav",
|
||||
"LabelPrefixesToIgnore": "Ignorované predpony (bez ohľadu na veľkosť písmen)",
|
||||
"LabelPreventIndexing": "Zabráňte indexovaniu Vášho zdroja službami iTunes a Google podcasts directories",
|
||||
"LabelPreventIndexing": "Zabráni indexácii vašich zdrojov službami iTunes a Google podcast directories",
|
||||
"LabelPrimaryEbook": "Primárny e-book",
|
||||
"LabelProgress": "Pokrok",
|
||||
"LabelProgress": "Stav",
|
||||
"LabelProvider": "Poskytovateľ",
|
||||
"LabelProviderAuthorizationValue": "Obsah hlavičky autorizácie",
|
||||
"LabelPubDate": "Dátum vydania",
|
||||
"LabelPubDate": "Dátum publikovania",
|
||||
"LabelPublishYear": "Rok vydania",
|
||||
"LabelPublishedDate": "Vydané {0}",
|
||||
"LabelPublishedDecade": "Dekáda vydania",
|
||||
@@ -519,7 +520,7 @@
|
||||
"LabelRSSFeedURL": "URL RSS zdroja",
|
||||
"LabelRandomly": "Náhodne",
|
||||
"LabelReAddSeriesToContinueListening": "Znova pridať série do pokračujúceho počúvania",
|
||||
"LabelRead": "Čítať",
|
||||
"LabelRead": "Načítať",
|
||||
"LabelReadAgain": "Čítať znova",
|
||||
"LabelReadEbookWithoutProgress": "Čítať e-knihu bez sledovania pokroku",
|
||||
"LabelRecentSeries": "Posledné série",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Dátum vydania",
|
||||
"LabelRemoveAllMetadataAbs": "Odstrániť všetky súbory metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Odstrániť všetky súbory metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Odstrániť z kapitol Audible intro a outro",
|
||||
"LabelRemoveCover": "Odstrániť prebal",
|
||||
"LabelRemoveMetadataFile": "Odstrániť súbory metadát z priečinkov položiek v knižnici",
|
||||
"LabelRemoveMetadataFileHelp": "Odstrániť všetky súbory metadata.json a metadata.abs vo Vašich {0} priečinkoch.",
|
||||
@@ -546,7 +548,7 @@
|
||||
"LabelSendEbookToDevice": "Poslať e-knihu do...",
|
||||
"LabelSequence": "Postupnosť",
|
||||
"LabelSerial": "Na pokračovanie",
|
||||
"LabelSeries": "Séria",
|
||||
"LabelSeries": "Série",
|
||||
"LabelSeriesName": "Názov série",
|
||||
"LabelSeriesProgress": "Pokrok série",
|
||||
"LabelServerLogLevel": "Úroveň logovania servera",
|
||||
@@ -576,7 +578,7 @@
|
||||
"LabelSettingsLibraryMarkAsFinishedTimeRemaining": "Zostávajúci čas je menší ako (sekúnd)",
|
||||
"LabelSettingsLibraryMarkAsFinishedWhen": "Označiť položku média ako dokončenú",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeries": "Preskočiť predchádzajúce knihy v pokračujúcej sérii",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeriesHelp": "Polička pokračujúcich sérií na domácej stránke zobrazuje prvú nezačatú knihu série, ktorá má ukončenú aspoň jednu z kníh série a žiadne započaté knihy. Povolením tohto nastavenia bude pokračujúca séria začínať poslednou ukončenou knihou miesto prvej nepočúvanej.",
|
||||
"LabelSettingsOnlyShowLaterBooksInContinueSeriesHelp": "Polička pokračujúcich sérií na domácej stránke zobrazuje prvú nezačatú knihu série, ktorá má dokončenú aspoň jednu z kníh série a žiadne započaté knihy. Povolením tohto nastavenia bude pokračujúca séria začínať poslednou ukončenou knihou miesto prvej nepočúvanej.",
|
||||
"LabelSettingsParseSubtitles": "Parsovať podtituly",
|
||||
"LabelSettingsParseSubtitlesHelp": "Extrahovať podtituly z názvov priečinkov audiokníh.<br>Podtitul musí byť oddelený \" - \"<br>tj. \"Názov knihy - Podtitul knihy\" má podtitul \"Podtitul knihy\"",
|
||||
"LabelSettingsPreferMatchedMetadata": "Preferovať vyhľadané metadáta",
|
||||
@@ -605,7 +607,7 @@
|
||||
"LabelSortAscending": "Vzostupne",
|
||||
"LabelSortDescending": "Zostupne",
|
||||
"LabelSortPubDate": "Zoradiť podľa dátumu vydania",
|
||||
"LabelStart": "Začiatok",
|
||||
"LabelStart": "Spustiť",
|
||||
"LabelStartTime": "Čas spustenia",
|
||||
"LabelStarted": "Začaté",
|
||||
"LabelStartedAt": "Začaté v",
|
||||
@@ -613,7 +615,7 @@
|
||||
"LabelStatsAuthors": "Autori",
|
||||
"LabelStatsBestDay": "Najlepší deň",
|
||||
"LabelStatsDailyAverage": "Denný priemer",
|
||||
"LabelStatsDays": "Dni",
|
||||
"LabelStatsDays": "Dní",
|
||||
"LabelStatsDaysListened": "Dní počúvania",
|
||||
"LabelStatsHours": "Hodiny",
|
||||
"LabelStatsInARow": "v rade",
|
||||
@@ -632,9 +634,9 @@
|
||||
"LabelTagsNotAccessibleToUser": "Štítky nedostupné užívateľovi",
|
||||
"LabelTasks": "Bežiace úlohy",
|
||||
"LabelTextEditorBulletedList": "Zoznam s odrážkami",
|
||||
"LabelTextEditorLink": "Pripojiť",
|
||||
"LabelTextEditorLink": "Prepojiť",
|
||||
"LabelTextEditorNumberedList": "Očíslovaný zoznam",
|
||||
"LabelTextEditorUnlink": "Odpojiť",
|
||||
"LabelTextEditorUnlink": "Zrušiť prepojenie",
|
||||
"LabelTheme": "Téma",
|
||||
"LabelThemeDark": "Tmavá",
|
||||
"LabelThemeLight": "Svetlá",
|
||||
@@ -646,7 +648,7 @@
|
||||
"LabelTimeLeft": "{0} ponechaných",
|
||||
"LabelTimeListened": "Čas počúvania",
|
||||
"LabelTimeListenedToday": "Dnešný čas počúvania",
|
||||
"LabelTimeRemaining": "{0} zostávajúcich",
|
||||
"LabelTimeRemaining": "Zostáva {0}",
|
||||
"LabelTimeToShift": "Čas posunutia v sekundách",
|
||||
"LabelTitle": "Názov",
|
||||
"LabelToolsEmbedMetadata": "Vlož metadáta",
|
||||
@@ -683,7 +685,7 @@
|
||||
"LabelUseChapterTrack": "Použiť stopu kapitoly",
|
||||
"LabelUseFullTrack": "Použiť celú stopu",
|
||||
"LabelUseZeroForUnlimited": "Použito 0 pre neobmedzené",
|
||||
"LabelUser": "Užívateľ",
|
||||
"LabelUser": "Používateľ",
|
||||
"LabelUsername": "Prihlasovacie meno",
|
||||
"LabelValue": "Hodnota",
|
||||
"LabelVersion": "Verzia",
|
||||
@@ -702,9 +704,11 @@
|
||||
"LabelYourAudiobookDuration": "Dĺžka Vašej audioknihy",
|
||||
"LabelYourBookmarks": "Vaše záložky",
|
||||
"LabelYourPlaylists": "Vaše playlisty",
|
||||
"LabelYourProgress": "Váš pokrok",
|
||||
"LabelYourProgress": "Váš aktuálny stav",
|
||||
"MessageAddToPlayerQueue": "Pridať do zoznamu prehrávania",
|
||||
"MessageAppriseDescription": "Aby ste mohli používať túto funkciumusíte mať k dispozícii inštanciu <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> alebo inú, ktorá dokáže spracovávať rovnaké požiadavky/requesty.<br/>Apprise URL musí byť úplná URL určená na zasielanie notifikácií, tj. ak napr. vaša APi beží na <code>http://192.168.1.1:8337</code>, vložte do daného poľa <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Uistite sa, že používate ASIN zo správneho regiónu Audible, nie Amazonu.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Reštartujte svoj server po uložení, aby mohli byť použité zmeny OIDC.",
|
||||
"MessageBackupsDescription": "Zálohy pokrývajú používateľov, ich aktuálne stavy počúvania, detaily položiek knižnice, nastavenia servera a obrázky uložené v <code>/metadata/items</code> a <code>/metadata/authors</code>. Zálohy <strong>neobsahujú</strong> súbory v priečinkoch vašich knižníc.",
|
||||
"MessageBackupsLocationEditNote": "Poznámka: Zmena umiestnenia záloh nepresunie ani nezmení existujúce zálohy",
|
||||
"MessageBackupsLocationNoEditNote": "Poznámka: Umietnenie záloh je nastavené prostredníctvom premennej prostredia a nie je ho možné zmeniť z tohto miesta.",
|
||||
@@ -723,8 +727,381 @@
|
||||
"MessageChapterErrorStartGteDuration": "Neplatný čas začiatku musí byť menší ako celkové trvanie audioknihy",
|
||||
"MessageChapterErrorStartLtPrev": "Neplatný čas začiatku musí byť väčší alebo rovný času začiatku predchádzajúcej kapitoly",
|
||||
"MessageChapterStartIsAfter": "Začiatok kapitoly je až za koncom vašej audioknihy",
|
||||
"MessageChaptersNotFound": "Kapitoly nenájdené",
|
||||
"MessageCheckingCron": "Kontrola cron-u...",
|
||||
"MessageConfirmCloseFeed": "Ste si istý, že chcete zavrieť tento zdroj?",
|
||||
"MessageConfirmDeleteBackup": "Ste si istý, že chcete zmazať zálohu {0}?",
|
||||
"MessageConfirmDeleteDevice": "Ste si istý, že chcete zmazať zariadenie čítačky e-kníh \"{0}\"?"
|
||||
"MessageConfirmDeleteDevice": "Ste si istý, že chcete zmazať zariadenie čítačky e-kníh \"{0}\"?",
|
||||
"MessageConfirmDeleteFile": "Týmto odstránite súbor z vášho súborového systému. Ste si istý?",
|
||||
"MessageConfirmDeleteLibrary": "Ste si istý, že chcete natrvalo odstrániť knižnicu \"{0}\"?",
|
||||
"MessageConfirmDeleteLibraryItem": "Týmto odstránite položku knižnice z databázy aj vášho súborového systému. Ste si istý?",
|
||||
"MessageConfirmDeleteLibraryItems": "Týmto odstránite {0} položiek knižnice z databázy a vášho súborového systému. Ste si istý?",
|
||||
"MessageConfirmDeleteMetadataProvider": "Ste si istý, že chcete odstrániť poskytovateľa metadát \"{0}\"?",
|
||||
"MessageConfirmDeleteNotification": "Ste si istý, že chcete odstrániť túto notifikáciu?",
|
||||
"MessageConfirmDeleteSession": "Ste si istý, že chcete zmazať túto reláciu?",
|
||||
"MessageConfirmEmbedMetadataInAudioFiles": "Ste si istý, že chcete vložiť metadáta do {0} zvukových súborov?",
|
||||
"MessageConfirmForceReScan": "Ste si istý, že chcete spustiť vynútený re-sken?",
|
||||
"MessageConfirmMarkAllEpisodesFinished": "Chcete označiť všetky epizódy ako dokončené?",
|
||||
"MessageConfirmMarkAllEpisodesNotFinished": "Checte označiť všetky epizódy ako nedokončené?",
|
||||
"MessageConfirmMarkItemFinished": "Chcete označiť \"{0}\" ako dokončené?",
|
||||
"MessageConfirmMarkItemNotFinished": "Chcete označiť \"{0}\" ako nedokončené?",
|
||||
"MessageConfirmMarkSeriesFinished": "Ste si istý, že chcete označiť všetky knihy v tejto sérii ako dokončené?",
|
||||
"MessageConfirmMarkSeriesNotFinished": "Ste si istý, že chcete označiť všetky knihy v tejto sérii ako nedokončené?",
|
||||
"MessageConfirmNotificationTestTrigger": "Zapnúť túto notifikáciu pre testovacie dáta?",
|
||||
"MessageConfirmPurgeCache": "Vymazanie vyrovnávacej pamäte kompletne odstráni priečinok <code>/metadata/cache</code>.<br /><br />Ste si istý, že chcete vymazať priečinok vyrovnávacej pamäte?",
|
||||
"MessageConfirmPurgeItemsCache": "Vymazanie položiek vyrovnávacej pamäte kompletne odstráni priečinok <code>/metadata/cache/items</code>.<br />Ste si istý?",
|
||||
"MessageConfirmQuickEmbed": "Varovanie! Pri rýchlom vložení metadát nie sú vaše zvukové súbory zálohované. Uistite sa, že máte vaše zvukové súbory zálohované. <br><br>Chcete pokračovať?",
|
||||
"MessageConfirmQuickMatchEpisodes": "Pri rýchlom vyhľadávaní epizód sú v prípade zhody prepísané podrobnosti nájdených epizód. Nespárované epizódy budú len aktualizované. Ste si istý?",
|
||||
"MessageConfirmReScanLibraryItems": "Ste si istý, že chcete spustiť opakovaný sken {0} položiek?",
|
||||
"MessageConfirmRemoveAllChapters": "Ste si istý, že chcete odstrániť všetky kapitoly?",
|
||||
"MessageConfirmRemoveAuthor": "Ste si istý, že chcete odstrániť autora \"{0}\"?",
|
||||
"MessageConfirmRemoveCollection": "Ste si istý, že chcete odstrániť zbierku \"{0}\"?",
|
||||
"MessageConfirmRemoveEpisode": "Ste si istý, že chcete odstrániť epizódu \"{0}\"?",
|
||||
"MessageConfirmRemoveEpisodes": "Ste si istý, že chcete odstrániť {0} epizód?",
|
||||
"MessageConfirmRemoveListeningSessions": "Ste si istý, že chcete odstrániť týchto {0} relácií?",
|
||||
"MessageConfirmRemoveMetadataFiles": "Ste si istý, že chcete odstrániť všetky súbory metadata.{0} z priečinkov položiek vašej knižnice?",
|
||||
"MessageConfirmRemoveNarrator": "Ste si istý, že chcete odstrániť interpreta \"{0}\"?",
|
||||
"MessageConfirmRemovePlaylist": "Ste si istý, že chcete odstrániť váš playlist \"{0}\"?",
|
||||
"MessageConfirmRenameGenre": "Ste si istý, že chcete nahradiť žáner \"{0}\" vo všetkých položkách žánrom \"{0}\"?",
|
||||
"MessageConfirmRenameGenreMergeNote": "Poznámka: Tento žáner už existuje a preto bude zlúčený.",
|
||||
"MessageConfirmRenameGenreWarning": "Varovanie! Podobný žáner len s odlišným zápisom už existuje \"{0}\".",
|
||||
"MessageConfirmRenameTag": "Ste si istý, že chcete zmeniť štítok \"{0}\" za \"{1}\" vo všetkých položkách?",
|
||||
"MessageConfirmRenameTagMergeNote": "Poznámka: Tento štítok už existuje a preto bude zlúčený.",
|
||||
"MessageConfirmRenameTagWarning": "Varovanie! Štítok s podobným zápisom už existuje \"{0}\".",
|
||||
"MessageConfirmResetProgress": "Ste si istý, že chcete resetnúť váš aktuálny stav počúvania?",
|
||||
"MessageConfirmSendEbookToDevice": "Ste si istý, že chcete poslať {0} e-knihu \"{1}\" do zariadenia \"{2}\"?",
|
||||
"MessageConfirmUnlinkOpenId": "Ste si istý, že chcete odstrániť prepojenie tohto používateľa na OpenID?",
|
||||
"MessageDaysListenedInTheLastYear": "{0} dní počúvania počas posledného roku",
|
||||
"MessageDownloadingEpisode": "Sťahovanie epizódy",
|
||||
"MessageDragFilesIntoTrackOrder": "Presuňte súbory do správneho poradia prehrávania",
|
||||
"MessageEmbedFailed": "Vloženie zlyhalo!",
|
||||
"MessageEmbedFinished": "Vloženie skončené!",
|
||||
"MessageEmbedQueue": "Zaradené do fronty na vloženie metadát ({0} v zozname)",
|
||||
"MessageEpisodesQueuedForDownload": "{0} epizód(a) v zozname na sťahovanie",
|
||||
"MessageEreaderDevices": "Na zaistenie dodania e-kníh môže byť nutné zadanie vyššie uvedenej e-mailovej adresy ako overeného odosielateľa v každom z nižšie vypísaných zariadení.",
|
||||
"MessageFeedURLWillBe": "URL zdroja bude {0}",
|
||||
"MessageFetching": "Získavam...",
|
||||
"MessageForceReScanDescription": "preskenuje všetky súbory ako pri prvom skenovaní. ID3 štítky zvukových súborov, OPF súbory a textové súbory budú nanovo naskenované.",
|
||||
"MessageImportantNotice": "Dôležité upozornenie!",
|
||||
"MessageInsertChapterBelow": "Vložte kapitolu nižšie",
|
||||
"MessageInvalidAsin": "Neplatné ASIN",
|
||||
"MessageItemsSelected": "{0} vybraných položiek",
|
||||
"MessageItemsUpdated": "{0} aktualizovaných položiek",
|
||||
"MessageJoinUsOn": "Pridajte sa k nám",
|
||||
"MessageLoading": "Načítavam...",
|
||||
"MessageLoadingFolders": "Načítanie priečinkov...",
|
||||
"MessageLogsDescription": "Záznamy logovania sú uložené v <code>/metadata/logs</code> vo forme JSON súborov. Záznamy kritických chýb sú uložené v <code>/metadata/logs/crash_logs.txt</code>.",
|
||||
"MessageM4BFailed": "M4B zlyhalo!",
|
||||
"MessageM4BFinished": "M4B skončené!",
|
||||
"MessageMapChapterTitles": "Prepojte názvy kapitol s vašimi už existujúcimi kapitolami audioknihy bez vplyvu na časové značky",
|
||||
"MessageMarkAllEpisodesFinished": "Označiť všetky epizódy ako dokončené",
|
||||
"MessageMarkAllEpisodesNotFinished": "Označiť všetky epizódy ako nedokončené",
|
||||
"MessageMarkAsFinished": "Označiť ako dokončené",
|
||||
"MessageMarkAsNotFinished": "Označiť ako nedokončené",
|
||||
"MessageMatchBooksDescription": "sa pokúsi spárovať knihy vo vašej knižnici s knihami nájdenými zvoleným poskytovateľom a doplniť chýbajúce údaje a prebal. aktuálne vyplnené údaje nebudú prepísané.",
|
||||
"MessageNoAudioTracks": "Žiadne zvukové stopy",
|
||||
"MessageNoAuthors": "Žiadni autori",
|
||||
"MessageNoBackups": "Žiadne zálohy",
|
||||
"MessageNoBookmarks": "Žiadne záložky",
|
||||
"MessageNoChapters": "Žiadne kapitoly",
|
||||
"MessageNoCollections": "Žiadne zbierky",
|
||||
"MessageNoCoversFound": "Neboli nájdené žiadne prebaly",
|
||||
"MessageNoDescription": "Žiadny popis",
|
||||
"MessageNoDevices": "Žiadne zariadenia",
|
||||
"MessageNoDownloadsInProgress": "Aktuálne neprebieha žiadne sťahovanie",
|
||||
"MessageNoDownloadsQueued": "Žiadne sťahovania v poradí",
|
||||
"MessageNoEpisodeMatchesFound": "Neboli spárované žiadne epizódy",
|
||||
"MessageNoEpisodes": "Žiadne eizódy",
|
||||
"MessageNoFoldersAvailable": "Žiadne priečinky nie sú dostupné",
|
||||
"MessageNoGenres": "Žiadne žánre",
|
||||
"MessageNoIssues": "Žiadne problémy",
|
||||
"MessageNoItems": "Žiadne položky",
|
||||
"MessageNoItemsFound": "Žiadne položky neboli nájdené",
|
||||
"MessageNoListeningSessions": "Žiadne relácie",
|
||||
"MessageNoLogs": "Žiadne záznamy udalostí",
|
||||
"MessageNoMediaProgress": "Žiadny stav médií",
|
||||
"MessageNoNotifications": "Žiadne upozornenia",
|
||||
"MessageNoPodcastFeed": "Chybný podcast: Žiadny zdroj",
|
||||
"MessageNoPodcastsFound": "Žiadne podcasty neboli nájdené",
|
||||
"MessageNoResults": "Žiadne výsledky",
|
||||
"MessageNoSearchResultsFor": "Žiadne výsledky vyhľadávania \"{0}\"",
|
||||
"MessageNoSeries": "Žiadne série",
|
||||
"MessageNoTags": "Žiadne štítky",
|
||||
"MessageNoTasksRunning": "Žiadne prebiehajúce úlohy",
|
||||
"MessageNoUpdatesWereNecessary": "Neboli potrebné žiadne aktualizácie",
|
||||
"MessageNoUserPlaylists": "Nemáte žiadny playlist",
|
||||
"MessageNoUserPlaylistsHelp": "Playlisty sú súkromné. Každý playlist môže vidieť iba používateľ, ktorý ho vytvoril.",
|
||||
"MessageNotYetImplemented": "Ešte neimplementované",
|
||||
"MessageOpmlPreviewNote": "Poznámka: Toto je náhľad pársovaného OPML súboru. Skutočný názov podcastu bude prevzatý zo RSS zdroja.",
|
||||
"MessageOr": "alebo",
|
||||
"MessagePauseChapter": "Pozastaviť prehrávanie kapitoly",
|
||||
"MessagePlayChapter": "Počúvať začiatok kapitoly",
|
||||
"MessagePlaylistCreateFromCollection": "Vytvoriť playlist zo zbierky",
|
||||
"MessagePleaseWait": "Prosím, počkajte...",
|
||||
"MessagePodcastHasNoRSSFeedForMatching": "Podcast nemá žiadnu URL RSS zdroja na spárovanie",
|
||||
"MessagePodcastSearchField": "Zadajte hľadaný výraz alebo URL RSS zdroja",
|
||||
"MessageQuickEmbedInProgress": "Prebieha rýchle vkladanie",
|
||||
"MessageQuickEmbedQueue": "Zadané na rýchle vloženie ({0} v rade)",
|
||||
"MessageQuickMatchAllEpisodes": "Rýchle vyhľadanie všetkých epizód",
|
||||
"MessageQuickMatchDescription": "Doplní všetky nevyplnené údaje a chýbajúci prebal z prvého výsledku vyhľdávania '{0}'. Neprepíše už existujúce údaje, vykoná tak iba v prípade, ak je zaškrtnutá voľba 'Preferovať vyhľadané metadáta'.",
|
||||
"MessageRemoveChapter": "Odstrániť kapitolu",
|
||||
"MessageRemoveEpisodes": "Odstrániť {0} epizódu(-y)",
|
||||
"MessageRemoveFromPlayerQueue": "Odstrániť zo zoznamu prehrávania",
|
||||
"MessageRemoveUserWarning": "Ste si istí, že chcete trvalo odstrániť užívateľa \"{0}\"?",
|
||||
"MessageReportBugsAndContribute": "Nahlásiť chyby, požiadavky na funkcie, a prispievať na",
|
||||
"MessageResetChaptersConfirm": "Ste si istý, že chcete resetnúť kapitoly a zahodiť zmeny, ktoré ste vykonali?",
|
||||
"MessageRestoreBackupConfirm": "Ste si istí, že chcete obnoviť zálohu vytvorenú",
|
||||
"MessageRestoreBackupWarning": "Obnovenie zálohy spôsobí kompletný prepis databázy umiestnenej v /config a obrázkov prebalov a autorov v /metadata/items a /metadata/authors.<br /><br />Zálohy nemenia žiadne súbory v priečinkoch vašej knižnice. Ak ste povolili v nastaveniach servera ukladanie obrázkov prebalov a metadát v priečinkoch knižnice, tieto nie sú zálohované a teda ani prepisované.<br /><br />Všetky klienti používajúci váš server budú automaticky obnovené.",
|
||||
"MessageScheduleLibraryScanNote": "Pre väčšinu používateľov sa odporúča ponechať túto funkciu vypnutú a povoliť nastavenia funkcie sledovania obsahu priečinku. Funkcia sledovania priečinku bude automaticky detekovať zmeny v priečinkoch knižnice. Táto funkcia však nefunguje pre všetky súborové systémy (ako napr. NFS), v tom prípade využite funkciu plánovaného skenovania knižnice.",
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Spustiť každú {0} o {1}",
|
||||
"MessageSearchResultsFor": "Výsledky vyhľadávania pre",
|
||||
"MessageSelected": "{0} vybrané",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "Poradie série nemôže obsahovať medzery",
|
||||
"MessageServerCouldNotBeReached": "Nepodarilo sa pripojiť na server",
|
||||
"MessageSetChaptersFromTracksDescription": "Nastaviť jednotlivé zvukové súbory ako kapitoly a názvy zvukových súborov ako názvy týchto kapitol",
|
||||
"MessageShareExpirationWillBe": "Expiruje <strong>{0}</strong>",
|
||||
"MessageShareExpiresIn": "Uplynie za {0}",
|
||||
"MessageShareURLWillBe": "URL na zdielanie bude <strong>{0}</strong>",
|
||||
"MessageStartPlaybackAtTime": "Spustiť prehrávanie \"{0}\" o {1}?",
|
||||
"MessageTaskAudioFileNotWritable": "Do súboru \"{0}\" sa nedá zapisovať",
|
||||
"MessageTaskCanceledByUser": "Úloha zrušená používateľom",
|
||||
"MessageTaskDownloadingEpisodeDescription": "Sťahuje sa diel \"{0}\"",
|
||||
"MessageTaskEmbeddingMetadata": "Vkladanie metadát",
|
||||
"MessageTaskEmbeddingMetadataDescription": "Vkladanie metadát v audioknihe \"{0}\"",
|
||||
"MessageTaskEncodingM4b": "Kódovanie M4B",
|
||||
"MessageTaskEncodingM4bDescription": "Konverzia audioknihy \"{0}\" do jedného m4b súboru",
|
||||
"MessageTaskFailed": "Zlyhané",
|
||||
"MessageTaskFailedToBackupAudioFile": "Nepodarilo sa zazálohovať audio súbor \"{0}\"",
|
||||
"MessageTaskFailedToCreateCacheDirectory": "Nepodarilo sa vytvoriť adresár na cache",
|
||||
"MessageTaskFailedToEmbedMetadataInFile": "Vloženie metadát do \"{0}\" zlyhalo",
|
||||
"MessageTaskFailedToMergeAudioFiles": "Nepodarilo sa pospájať audio súbory",
|
||||
"MessageTaskFailedToMoveM4bFile": "Nepodarilo sa presunúť m4b súbor",
|
||||
"MessageTaskFailedToWriteMetadataFile": "Nepodarilo sa zapísať súbor s metadátami",
|
||||
"MessageTaskMatchingBooksInLibrary": "Spárované knihy v knižnici \"{0}\"",
|
||||
"MessageTaskNoFilesToScan": "Žiadne súbory k skenu",
|
||||
"MessageTaskOpmlImport": "OMPL import",
|
||||
"MessageTaskOpmlImportDescription": "Vytvorenie podcastov z {0} RSS zdrojov",
|
||||
"MessageTaskOpmlImportFeed": "Importný zdroj OPML",
|
||||
"MessageTaskOpmlImportFeedDescription": "Import RSS zdroja \"{0}\"",
|
||||
"MessageTaskOpmlImportFeedFailed": "Získanie zdroja podcastu zlyhalo",
|
||||
"MessageTaskOpmlImportFeedPodcastDescription": "Vytvorenie podcastu \"{0}\"",
|
||||
"MessageTaskOpmlImportFeedPodcastExists": "Na zadanom umiestnení už podcast existuje",
|
||||
"MessageTaskOpmlImportFeedPodcastFailed": "Vytvorenie podcastu zlyhalo",
|
||||
"MessageTaskOpmlImportFinished": "Pridaných {0} podcastov",
|
||||
"MessageTaskOpmlParseFailed": "Pársovanie OPML súboru zlyhalo",
|
||||
"MessageTaskOpmlParseFastFail": "Neplatný OPML súbor. <opml> tag ALEBO <outline> tag nebol nájdený",
|
||||
"MessageTaskOpmlParseNoneFound": "V OPML súbore neboli nájdené žiadne zdroje",
|
||||
"MessageTaskScanItemsAdded": "{0} pridaných",
|
||||
"MessageTaskScanItemsMissing": "{0} chýbajúcich",
|
||||
"MessageTaskScanItemsUpdated": "{0} aktualizovaných",
|
||||
"MessageTaskScanNoChangesNeeded": "Neboli potrebné žiadne zmeny",
|
||||
"MessageTaskScanningFileChanges": "Skenovanie zmien súborov v \"{0}\"",
|
||||
"MessageTaskScanningLibrary": "Skenovanie knižnice \"{0}\"",
|
||||
"MessageTaskTargetDirectoryNotWritable": "Cieľový priečinok nemá oprávnenie na zápis",
|
||||
"MessageThinking": "Premýšľam...",
|
||||
"MessageUploaderItemFailed": "Nahratie zlyhalo",
|
||||
"MessageUploaderItemSuccess": "Úspešne nahrané!",
|
||||
"MessageUploading": "Nahrávanie zmien...",
|
||||
"MessageValidCronExpression": "Platný cron výraz",
|
||||
"MessageWatcherIsDisabledGlobally": "Funkcia sledovania zmien je globálne vypnutá v nastaveniach servera",
|
||||
"MessageXLibraryIsEmpty": "Knižnica {0} je prázdna!",
|
||||
"MessageYourAudiobookDurationIsLonger": "Dĺžka vašej audioknihy je väčšia ako dĺžka nájdených súborov",
|
||||
"MessageYourAudiobookDurationIsShorter": "Dĺžka vašej audioknihy je menšia ako dĺžka nájdených súborov",
|
||||
"NoteChangeRootPassword": "Root používateľ je jediný používateľ, ktorý môže mať prázdne heslo",
|
||||
"NoteChapterEditorTimes": "Poznámka: Prvá kapitola musí vždy začínať v 0:00 a začiatok poslednej kapitoly nemôže prekročiť trvanie tejto audioknihy.",
|
||||
"NoteFolderPicker": "Poznámka: Priečinky, ktoré už boli priradené, sa ďalej nezobrazujú",
|
||||
"NoteRSSFeedPodcastAppsHttps": "Varovanie: Väčšina podcastových aplikácií požaduje URL RSS zdroja s HTTPS",
|
||||
"NoteRSSFeedPodcastAppsPubDate": "Varovanie: 1 alebo viac vašich epizód neobsahuje infomáciu o dátum vydania. Niektoré podcastové ju vyžadujú.",
|
||||
"NoteUploaderFoldersWithMediaFiles": "Priečinky obsahujúce súbory médií budú považované za samostatné položky knižnice.",
|
||||
"NoteUploaderOnlyAudioFiles": "Ak budú nahraté iba zvukové súbory, každý zvukový súbor bude považovaný za samostatnú audioknihu.",
|
||||
"NoteUploaderUnsupportedFiles": "Nepodporované súbory budú ignorované. Pri výbere alebo prenesení priečinka, budú všetky súbory, ktoré nie sú v priečinku niektorej z položiek, ignorované.",
|
||||
"NotificationOnBackupCompletedDescription": "Spustené po dokončení zálohovania",
|
||||
"NotificationOnBackupFailedDescription": "Spustené pri zlyhaní zálohovania",
|
||||
"NotificationOnEpisodeDownloadedDescription": "Spustené po automatickom stiahnutí epizódy podcastu",
|
||||
"NotificationOnTestDescription": "Udalosť určená na testovanie systému notifikácií",
|
||||
"PlaceholderNewCollection": "Názov novej zbierky",
|
||||
"PlaceholderNewFolderPath": "Umiestnenie nového priečinka",
|
||||
"PlaceholderNewPlaylist": "Názov nového playlistu",
|
||||
"PlaceholderSearch": "Vyhľadávanie..",
|
||||
"PlaceholderSearchEpisode": "Vyhľadať epizódu..",
|
||||
"StatsAuthorsAdded": "autorov pridaných",
|
||||
"StatsBooksAdded": "kníh pridaných",
|
||||
"StatsBooksAdditional": "Niektoré prílohy zahŕňajú…",
|
||||
"StatsBooksFinished": "dokončených kníh",
|
||||
"StatsBooksFinishedThisYear": "Niektoré knihy dokončené tento rok…",
|
||||
"StatsBooksListenedTo": "vypočutých kníh",
|
||||
"StatsCollectionGrewTo": "Vaša knižná zbierka sa blíži k…",
|
||||
"StatsSessions": "relácie",
|
||||
"StatsSpentListening": "strávených počúvaním",
|
||||
"StatsTopAuthor": "TOP AUTOR",
|
||||
"StatsTopAuthors": "TOP AUTORI",
|
||||
"StatsTopGenre": "TOP ŽÁNER",
|
||||
"StatsTopGenres": "TOP ŽÁNRE",
|
||||
"StatsTopMonth": "TOP MESIAC",
|
||||
"StatsTopNarrator": "TOP INTERPRET",
|
||||
"StatsTopNarrators": "TOP INTERPRETI",
|
||||
"StatsTotalDuration": "S celkovou dĺžkou…",
|
||||
"StatsYearInReview": "ROK V PREHĽADE",
|
||||
"ToastAccountUpdateSuccess": "Účet bol aktualizovaný",
|
||||
"ToastAppriseUrlRequired": "Musíte zadať Apprise URL",
|
||||
"ToastAsinRequired": "Vyžaduje sa ASIN",
|
||||
"ToastAuthorImageRemoveSuccess": "Obrázok autora bol odstránený",
|
||||
"ToastAuthorNotFound": "Autor \"{0}\" nenájdený",
|
||||
"ToastAuthorRemoveSuccess": "Autor bol odstránený",
|
||||
"ToastAuthorSearchNotFound": "Autor nebol nájdený",
|
||||
"ToastAuthorUpdateMerged": "Autor bol zlúčený",
|
||||
"ToastAuthorUpdateSuccess": "Autor bol aktualizovaný",
|
||||
"ToastAuthorUpdateSuccessNoImageFound": "Autor bol aktualizovaný (obrázok nebol nájdený)",
|
||||
"ToastBackupAppliedSuccess": "Záloha bola aplikovaná",
|
||||
"ToastBackupCreateFailed": "Vytvorenie zálohy zlyhalo",
|
||||
"ToastBackupCreateSuccess": "Záloha bola vytvorená",
|
||||
"ToastBackupDeleteFailed": "Odstránenie zálohy zlyhalo",
|
||||
"ToastBackupDeleteSuccess": "Záloha bola odstránená",
|
||||
"ToastBackupInvalidMaxKeep": "Neplatný počet sledovaných záloh",
|
||||
"ToastBackupInvalidMaxSize": "Neplatná maximálna veľkosť zálohy",
|
||||
"ToastBackupRestoreFailed": "Obnovenie zo zálohy zlyhalo",
|
||||
"ToastBackupUploadFailed": "Nahratie zálohy zlyhalo",
|
||||
"ToastBackupUploadSuccess": "Záloha bola nahratá",
|
||||
"ToastBatchApplyDetailsToItemsSuccess": "Dotknuté údaje položiek",
|
||||
"ToastBatchDeleteFailed": "Hromadné odstránenie zlyhalo",
|
||||
"ToastBatchDeleteSuccess": "Hromadné odstránenie bolo úspešné",
|
||||
"ToastBatchQuickMatchFailed": "Hromadné rýchle vyhľadanie zlyhalo!",
|
||||
"ToastBatchQuickMatchStarted": "Hromadné rýchle vyhľadanie {0} kníh začalo!",
|
||||
"ToastBatchUpdateFailed": "Hromadná aktualizácia zlyhala",
|
||||
"ToastBatchUpdateSuccess": "Hromadná aktualizácia prebehla úspešne",
|
||||
"ToastBookmarkCreateFailed": "Vytvorenie záložky zlyhalo",
|
||||
"ToastBookmarkCreateSuccess": "Záložka pridaná",
|
||||
"ToastBookmarkRemoveSuccess": "Záložka odstránená",
|
||||
"ToastCachePurgeFailed": "Vyčistenie vyrovnávacej pamäte zlyhalo",
|
||||
"ToastCachePurgeSuccess": "Vyrovnávacia pamäť vyčistená",
|
||||
"ToastChaptersHaveErrors": "Kapitoly obsahujú chyby",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Neplatná hodnota veľkosti posunutia. Začiatok poslednej kapitoly by ležal za koncom audioknihy.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Nesprávna hodnota posunutia. Prvá kapitola by mala nulovú alebo zápornú dĺžku a bola by nahradená nasledujúcou kapitolou. Navýšte čas začiatku druhej kapitoly.",
|
||||
"ToastChaptersMustHaveTitles": "Kapitoly musia mať názvy",
|
||||
"ToastChaptersRemoved": "Kapitoly boli odstránené",
|
||||
"ToastChaptersUpdated": "Kapitoly boli aktualizované",
|
||||
"ToastCollectionItemsAddFailed": "Pridanie položky/-iek do zbierky zlyhalo",
|
||||
"ToastCollectionRemoveSuccess": "Zbierka odstránená",
|
||||
"ToastCollectionUpdateSuccess": "Zbierka aktualizovaná",
|
||||
"ToastCoverUpdateFailed": "Aktualizácia prebalu zlyhala",
|
||||
"ToastDateTimeInvalidOrIncomplete": "Dátum a čas sú neplatné alebo neúplné",
|
||||
"ToastDeleteFileFailed": "Odstránenie súboru zlyhalo",
|
||||
"ToastDeleteFileSuccess": "Súbor bol odstránený",
|
||||
"ToastDeviceAddFailed": "Pridanie zariadenia zlyhalo",
|
||||
"ToastDeviceNameAlreadyExists": "Čítačka e-kníh s rovnakým názvom už existuje",
|
||||
"ToastDeviceTestEmailFailed": "Odoslanie testovacieho e-mailu zlyhalo",
|
||||
"ToastDeviceTestEmailSuccess": "Testovací e-mail bol odoslaný",
|
||||
"ToastEmailSettingsUpdateSuccess": "Nastavenia e-mailu boli aktualizované",
|
||||
"ToastEncodeCancelFailed": "Zrušenie znakového kódovania zlyhalo",
|
||||
"ToastEncodeCancelSucces": "Znakové kódovanie bolo zrušené",
|
||||
"ToastEpisodeDownloadQueueClearFailed": "Vyčistenie poradia zlyhalo",
|
||||
"ToastEpisodeDownloadQueueClearSuccess": "Poradie sťahovania bolo vyčistené",
|
||||
"ToastEpisodeUpdateSuccess": "{0} epizód bolo aktualizovaných",
|
||||
"ToastErrorCannotShare": "Na tomto zariadení nie je možné zdielať vybraným spôsobom",
|
||||
"ToastFailedToLoadData": "Načítanie údajov zlyhalo",
|
||||
"ToastFailedToMatch": "Spárovanie zlyhalo",
|
||||
"ToastFailedToShare": "Zdieľanie zlyhalo",
|
||||
"ToastFailedToUpdate": "Aktualizácia zlyhala",
|
||||
"ToastInvalidImageUrl": "Neplatná URL obrázku",
|
||||
"ToastInvalidMaxEpisodesToDownload": "Neplatný maximálny počet epizód na stiahnutie",
|
||||
"ToastInvalidUrl": "Neplatná URL",
|
||||
"ToastItemCoverUpdateSuccess": "Prebal položky bol aktualizovaný",
|
||||
"ToastItemDeletedFailed": "Odstránenie položky zlyhalo",
|
||||
"ToastItemDeletedSuccess": "Položka bola odstránená",
|
||||
"ToastItemDetailsUpdateSuccess": "Údaje položky boli aktualizované",
|
||||
"ToastItemMarkedAsFinishedFailed": "Označenie za Dokončené zlyhalo",
|
||||
"ToastItemMarkedAsFinishedSuccess": "Položka bola označená ako Dokončená",
|
||||
"ToastItemMarkedAsNotFinishedFailed": "Označenie za Nedokončené zlyhalo",
|
||||
"ToastItemMarkedAsNotFinishedSuccess": "Položka bola označená ako Nedokončená",
|
||||
"ToastItemUpdateSuccess": "Položka bola aktualizovaná",
|
||||
"ToastLibraryCreateFailed": "Vytvorenie knižnice zlyhalo",
|
||||
"ToastLibraryCreateSuccess": "Knižnica \"{0}\" bola vytvorená",
|
||||
"ToastLibraryDeleteFailed": "Odstránenie knižnice zlyhalo",
|
||||
"ToastLibraryDeleteSuccess": "Knižnica bola odstránená",
|
||||
"ToastLibraryScanFailedToStart": "Spustenie skenovania zlyhalo",
|
||||
"ToastLibraryScanStarted": "Skenovanie knižnice sa začalo",
|
||||
"ToastLibraryUpdateSuccess": "Knižnica \"{0}\" bola aktualizovaná",
|
||||
"ToastMatchAllAuthorsFailed": "Spárovanie všetkých autorov zlyhalo",
|
||||
"ToastMetadataFilesRemovedError": "Chyba pri odstraňovaní súborov metadata.{0}",
|
||||
"ToastMetadataFilesRemovedNoneFound": "Žiadne súbory metadata.{0} neboli v knižnici nájdené",
|
||||
"ToastMetadataFilesRemovedNoneRemoved": "Žiadne súbory metadata.{0} neboli odstránené",
|
||||
"ToastMetadataFilesRemovedSuccess": "{0} súborov metadata.{1} bolo odstránených",
|
||||
"ToastMustHaveAtLeastOnePath": "Musí mať aspoň jednu cestu umiestnenia",
|
||||
"ToastNameEmailRequired": "Meno a e-mail sú povinné",
|
||||
"ToastNameRequired": "Meno je povinné",
|
||||
"ToastNewEpisodesFound": "Bolo nájdených {0} nových epizód",
|
||||
"ToastNewUserCreatedFailed": "Vytvorenie účtu zlyhalo: \"{0}\"",
|
||||
"ToastNewUserCreatedSuccess": "Nový účet bol vytvorený",
|
||||
"ToastNewUserLibraryError": "Musíte vybrať aspoň jednu knižnicu",
|
||||
"ToastNewUserPasswordError": "Musí mať heslo, len root používateľ môže mať prázdne heslo",
|
||||
"ToastNewUserTagError": "Musíte vybrať aspoň jeden štítok",
|
||||
"ToastNewUserUsernameError": "Zadajte používateľské meno",
|
||||
"ToastNoNewEpisodesFound": "Žiadne nové epizódy neboli nájdené",
|
||||
"ToastNoRSSFeed": "Podcast nemá RSS zdroj",
|
||||
"ToastNoUpdatesNecessary": "Žiadne aktualizácie nie nutné",
|
||||
"ToastNotificationCreateFailed": "Vytvorenie notifikácie zlyhalo",
|
||||
"ToastNotificationDeleteFailed": "Odstránenie notifikácie zlyhalo",
|
||||
"ToastNotificationFailedMaximum": "Maximálny počet chybných pokusov musí byť >= 0",
|
||||
"ToastNotificationQueueMaximum": "Maximálny počet notifikácií v poradí musí byť >= 0",
|
||||
"ToastNotificationSettingsUpdateSuccess": "Nastavenia notifikácií boli aktualizované",
|
||||
"ToastNotificationTestTriggerFailed": "Spustenie testovacej notifikácie zlyhalo",
|
||||
"ToastNotificationTestTriggerSuccess": "Testovacia notifikácia bola spustená",
|
||||
"ToastNotificationUpdateSuccess": "Notifikácia bola aktualizovaná",
|
||||
"ToastPlaylistCreateFailed": "Vytvorenie playlistu zlyhalo",
|
||||
"ToastPlaylistCreateSuccess": "Playlist bol vytvorený",
|
||||
"ToastPlaylistRemoveSuccess": "Playlist bol odstránený",
|
||||
"ToastPlaylistUpdateSuccess": "Playlist bol aktualizovaný",
|
||||
"ToastPodcastCreateFailed": "Vytvorenie podcastu zlyhalo",
|
||||
"ToastPodcastCreateSuccess": "Podcast bol vytvorený",
|
||||
"ToastPodcastGetFeedFailed": "Získanie zdroja podcastu zlyhalo",
|
||||
"ToastPodcastNoEpisodesInFeed": "Na RSS zdroji neboli nájdené žiadne epizódy",
|
||||
"ToastPodcastNoRssFeed": "Podcast nemá RSS zdroj",
|
||||
"ToastProgressIsNotBeingSynced": "Stav počúvania nie je synchronizovaný, reštartujte prehrávanie",
|
||||
"ToastProviderCreatedFailed": "Pridanie poskytovateľa zlyhalo",
|
||||
"ToastProviderCreatedSuccess": "Nový poskytovateľ bol pridaný",
|
||||
"ToastProviderNameAndUrlRequired": "Meno a URL sú povinné",
|
||||
"ToastProviderRemoveSuccess": "Poskytovateľ bol odstránený",
|
||||
"ToastRSSFeedCloseFailed": "Odstránenie RSS zdroja zlyhalo",
|
||||
"ToastRSSFeedCloseSuccess": "RSS zdroj bol odstránený",
|
||||
"ToastRemoveFailed": "Odstránenie zlyhalo",
|
||||
"ToastRemoveItemFromCollectionFailed": "Odstránenie položky zo zbierky zlyhalo",
|
||||
"ToastRemoveItemFromCollectionSuccess": "Položka bola zo zbierky odstránená",
|
||||
"ToastRemoveItemsWithIssuesFailed": "Odstránenie položiek s problémami zlyhalo",
|
||||
"ToastRemoveItemsWithIssuesSuccess": "Položky knižnice s problémami boli odstránené",
|
||||
"ToastRenameFailed": "Premenovanie zlyhalo",
|
||||
"ToastRescanFailed": "Opakované skenovanie {0} zlyhalo",
|
||||
"ToastRescanRemoved": "Opakovane skenovaná položka bola odstránená",
|
||||
"ToastRescanUpToDate": "Opakovane skenovaná položka bola aktuálna",
|
||||
"ToastRescanUpdated": "Opakovane skenovaná položka bola aktualizovaná",
|
||||
"ToastScanFailed": "Skenovanie položky knižnice zlyhalo",
|
||||
"ToastSelectAtLeastOneUser": "Vyberte aspoň jedného používateľa",
|
||||
"ToastSendEbookToDeviceFailed": "Odoslanie e-knihy do zariadenia zlyhalo",
|
||||
"ToastSendEbookToDeviceSuccess": "E-kniha bola odoslaná do zariadenia \"{0}\"",
|
||||
"ToastSeriesSubmitFailedSameName": "Nie je možné pridať dve série s rovnakým názvom",
|
||||
"ToastSeriesUpdateFailed": "Aktualizácia série zlyhala",
|
||||
"ToastSeriesUpdateSuccess": "Séria bola úspešne aktualizovaná",
|
||||
"ToastServerSettingsUpdateSuccess": "Nastavenia servera boli aktualizované",
|
||||
"ToastSessionCloseFailed": "Ukončenie relácie zlyhalo",
|
||||
"ToastSessionDeleteFailed": "Odstránenie relácie zlyhalo",
|
||||
"ToastSessionDeleteSuccess": "Relácia odstránená",
|
||||
"ToastSleepTimerDone": "Časovač spánku bol spustený... zZzzZz",
|
||||
"ToastSlugMustChange": "Slug obsahuje naplatné znaky",
|
||||
"ToastSlugRequired": "Slug je povinný",
|
||||
"ToastSocketConnected": "Socket je pripojený",
|
||||
"ToastSocketDisconnected": "Socket je odpojený",
|
||||
"ToastSocketFailedToConnect": "Pripojenie socketu zlyhalo",
|
||||
"ToastSortingPrefixesEmptyError": "Musí mať aspoň jednu triediacu predponu",
|
||||
"ToastSortingPrefixesUpdateSuccess": "Triediace predpony aktualizované ({0} položiek)",
|
||||
"ToastTitleRequired": "Názov je povinný",
|
||||
"ToastUnknownError": "Neznáma chyba",
|
||||
"ToastUnlinkOpenIdFailed": "Odstránenie prepojenia na OpenID zlyhalo",
|
||||
"ToastUnlinkOpenIdSuccess": "Prepojenie používateľa na OpenID bolo odstránené",
|
||||
"ToastUploaderFilepathExistsError": "Umiestnenie \"{0}\" na serveri už existuje",
|
||||
"ToastUploaderItemExistsInSubdirectoryError": "Položka \"{0}\" používa podpriečinok umiestnenia pre nahrávanie.",
|
||||
"ToastUserDeleteFailed": "Odstránenie používateľa zlyhalo",
|
||||
"ToastUserDeleteSuccess": "Používateľ bol odstránený",
|
||||
"ToastUserPasswordChangeSuccess": "Zmena hesla prebehla úspešne",
|
||||
"ToastUserPasswordMismatch": "Heslá sa nezhodujú",
|
||||
"ToastUserPasswordMustChange": "Nové heslo sa nesmie zhodovať so starým",
|
||||
"ToastUserRootRequireName": "Musíte zadať používateľské meno root používateľa"
|
||||
}
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Seznam predvajanja",
|
||||
"HeaderPlaylistItems": "Elementi seznama predvajanja",
|
||||
"HeaderPodcastsToAdd": "Podcasti za dodajanje",
|
||||
"HeaderPresets": "Prednastavitve",
|
||||
"HeaderPreviewCover": "Naslovnica za predogled",
|
||||
"HeaderRSSFeedGeneral": "RSS podrobnosti",
|
||||
"HeaderRSSFeedIsOpen": "Vir RSS je odprt",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Datum izdaje",
|
||||
"LabelRemoveAllMetadataAbs": "Odstrani vse datoteke metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Odstrani vse datoteke metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Odstrani Audible uvod in zaključek iz poglavij",
|
||||
"LabelRemoveCover": "Odstrani naslovnico",
|
||||
"LabelRemoveMetadataFile": "Odstrani datoteke z metapodatki v mapah elementov knjižnice",
|
||||
"LabelRemoveMetadataFileHelp": "Odstrani vse datoteke metadata.json in metadata.abs v svojih mapah {0}.",
|
||||
@@ -705,6 +707,8 @@
|
||||
"LabelYourProgress": "Tvoj napredek",
|
||||
"MessageAddToPlayerQueue": "Dodaj v čakalno vrsto predvajalnika",
|
||||
"MessageAppriseDescription": "Če želite uporabljati to funkcijo, morate imeti zagnano namestitev <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">API Apprise</a> ali API, ki bo obravnavala te iste zahteve. <br />Url API-ja Apprise mora biti celotna pot URL-ja za pošiljanje obvestila, npr. če je vaša namestitev API-ja postrežena na <code>http://192.168.1.1:8337</code>, bi morali vnesti <code >http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Prepričajte se, da uporabljate ASIN iz pravilne zvočne regije, ne iz Amazona.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Za uveljavitev OIDC sprememb, po shranjevanju znova zaženite strežnik.",
|
||||
"MessageBackupsDescription": "Varnostne kopije vključujejo uporabnike, napredek uporabnikov, podrobnosti elementov knjižnice, nastavitve strežnika in slike, shranjene v <code>/metadata/items</code> & <code>/metadata/authors</code>. Varnostne kopije <strong>ne</strong> vključujejo datotek, shranjenih v mapah vaše knjižnice.",
|
||||
"MessageBackupsLocationEditNote": "Opomba: Posodabljanje lokacije varnostne kopije ne bo premaknilo ali spremenilo obstoječih varnostnih kopij",
|
||||
"MessageBackupsLocationNoEditNote": "Opomba: Lokacija varnostne kopije je nastavljena s spremenljivko okolja in je tu ni mogoče spremeniti.",
|
||||
@@ -723,6 +727,7 @@
|
||||
"MessageChapterErrorStartGteDuration": "Neveljaven začetni čas, mora biti krajši od trajanja zvočne knjige",
|
||||
"MessageChapterErrorStartLtPrev": "Neveljaven začetni čas mora biti večji od ali enak začetnemu času prejšnjega poglavja",
|
||||
"MessageChapterStartIsAfter": "Začetek poglavja je po koncu vaše zvočne knjige",
|
||||
"MessageChaptersNotFound": "Poglavij ni bilo najdenih",
|
||||
"MessageCheckingCron": "Preverjam cron...",
|
||||
"MessageConfirmCloseFeed": "Ali ste prepričani, da želite zapreti ta vir?",
|
||||
"MessageConfirmDeleteBackup": "Ali ste prepričani, da želite izbrisati varnostno kopijo za {0}?",
|
||||
@@ -779,6 +784,7 @@
|
||||
"MessageForceReScanDescription": "bo znova pregledal vse datoteke kot pregled od začetka. Oznake ID3 zvočnih datotek, datoteke OPF in besedilne datoteke bodo pregledane kot nove.",
|
||||
"MessageImportantNotice": "Pomembno obvestilo!",
|
||||
"MessageInsertChapterBelow": "Spodaj vstavite poglavje",
|
||||
"MessageInvalidAsin": "Neveljaven ASIN",
|
||||
"MessageItemsSelected": "{0} izbranih elementov",
|
||||
"MessageItemsUpdated": "Št. posodobljenih elementov: {0}",
|
||||
"MessageJoinUsOn": "Pridružite se nam",
|
||||
@@ -850,6 +856,7 @@
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Zaženi vsakih {0} ob {1}",
|
||||
"MessageSearchResultsFor": "Rezultati iskanja za",
|
||||
"MessageSelected": "{0} izbrano",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "Zaporedje serij ne sme vsebovati presledkov",
|
||||
"MessageServerCouldNotBeReached": "Strežnika ni bilo mogoče doseči",
|
||||
"MessageSetChaptersFromTracksDescription": "Nastavi poglavja z uporabo vsake zvočne datoteke kot poglavja in naslova poglavja kot imena zvočne datoteke",
|
||||
"MessageShareExpirationWillBe": "Potečeno bo <strong>{0}</strong>",
|
||||
@@ -968,6 +975,8 @@
|
||||
"ToastCachePurgeFailed": "Čiščenje predpomnilnika ni uspelo",
|
||||
"ToastCachePurgeSuccess": "Predpomnilnik je bil uspešno očiščen",
|
||||
"ToastChaptersHaveErrors": "Poglavja imajo napake",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Neveljavna vrednost zamika. Začetni čas zadnjega poglavja bi presegel trajanje te zvočne knjige.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Neveljavna vrednost zamika. Prvo poglavje bi imelo ničelno ali negativno dolžino in bi ga prepisalo drugo poglavje. Povečajte začetno trajanje drugega poglavja.",
|
||||
"ToastChaptersMustHaveTitles": "Poglavja morajo imeti naslove",
|
||||
"ToastChaptersRemoved": "Poglavja so odstranjena",
|
||||
"ToastChaptersUpdated": "Poglavja so posodobljena",
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "Список відтворення",
|
||||
"HeaderPlaylistItems": "Елементи списку відтворення",
|
||||
"HeaderPodcastsToAdd": "Додати подкасти",
|
||||
"HeaderPresets": "Пресети",
|
||||
"HeaderPreviewCover": "Попередній перегляд",
|
||||
"HeaderRSSFeedGeneral": "Подробиці RSS",
|
||||
"HeaderRSSFeedIsOpen": "RSS-канал відкрито",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "Дата публікації",
|
||||
"LabelRemoveAllMetadataAbs": "Видалити всі файли metadata.abs",
|
||||
"LabelRemoveAllMetadataJson": "Видалити всі файли metadata.json",
|
||||
"LabelRemoveAudibleBranding": "Видалити звуковий вступ та завершення з розділів",
|
||||
"LabelRemoveCover": "Видалити обкладинку",
|
||||
"LabelRemoveMetadataFile": "Видалити файли метаданих у папках елементів бібліотеки",
|
||||
"LabelRemoveMetadataFileHelp": "Видалити всі файли metadata.json та metadata.abs у ваших папках {0}.",
|
||||
@@ -607,8 +609,8 @@
|
||||
"LabelSortPubDate": "Сортувати дату публікації",
|
||||
"LabelStart": "Початок",
|
||||
"LabelStartTime": "Час початку",
|
||||
"LabelStarted": "Почато",
|
||||
"LabelStartedAt": "Почато",
|
||||
"LabelStarted": "Стартував",
|
||||
"LabelStartedAt": "Почато з",
|
||||
"LabelStatsAudioTracks": "Аудіодоріжки",
|
||||
"LabelStatsAuthors": "Автори",
|
||||
"LabelStatsBestDay": "Найкращий день",
|
||||
@@ -706,6 +708,7 @@
|
||||
"MessageAddToPlayerQueue": "Додати до черги відтворення",
|
||||
"MessageAppriseDescription": "Щоб скористатися цією функцією, вам потрібно мати запущену <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> або API, що оброблятиме ті ж запити. <br />Аби надсилати сповіщення, URL-адреса API Apprise мусить бути повною, наприклад, якщо ваш API розміщено за адресою <code>http://192.168.1.1:8337</code>, то необхідно вказати адресу <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "Переконайтесь, що ви використовуєте ASIN з правильної регіональної Audible зони, а не з Amazon.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "Перезавантажте сервер після збереження, щоб застосувати зміни OIDC.",
|
||||
"MessageBackupsDescription": "Резервні копії містять користувачів, прогрес, подробиці елементів бібліотеки, налаштування сервера та зображення з <code>/metadata/items</code> та <code>/metadata/authors</code>. Резервні копії <strong>не</strong> містять жодних файлів з тек бібліотеки.",
|
||||
"MessageBackupsLocationEditNote": "Примітка: оновлення розташування резервної копії не переносить та не змінює існуючих копій",
|
||||
"MessageBackupsLocationNoEditNote": "Примітка: розташування резервної копії встановлюється за допомогою змінної середовища та не може бути змінене тут.",
|
||||
@@ -815,7 +818,7 @@
|
||||
"MessageNoItems": "Елементи відсутні",
|
||||
"MessageNoItemsFound": "Елементів не знайдено",
|
||||
"MessageNoListeningSessions": "Сеанси прослуховування відсутні",
|
||||
"MessageNoLogs": "Немає журналів",
|
||||
"MessageNoLogs": "Немає журнали",
|
||||
"MessageNoMediaProgress": "Прогрес відсутній",
|
||||
"MessageNoNotifications": "Сповіщення відсутні",
|
||||
"MessageNoPodcastFeed": "Невірний подкаст: Немає каналу",
|
||||
@@ -853,6 +856,7 @@
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "Запуск кожні {0} о {1}",
|
||||
"MessageSearchResultsFor": "Результати пошуку для",
|
||||
"MessageSelected": "Вибрано: {0}",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "Послідовність серій не може містити пробілів",
|
||||
"MessageServerCouldNotBeReached": "Не вдалося підключитися до сервера",
|
||||
"MessageSetChaptersFromTracksDescription": "Створити глави з аудіодоріжок, встановивши назви файлів за заголовки",
|
||||
"MessageShareExpirationWillBe": "Термін сплине за <strong>{0}</strong>",
|
||||
@@ -971,6 +975,8 @@
|
||||
"ToastCachePurgeFailed": "Не вдалося очистити кеш",
|
||||
"ToastCachePurgeSuccess": "Кеш очищено",
|
||||
"ToastChaptersHaveErrors": "Глави містять помилки",
|
||||
"ToastChaptersInvalidShiftAmountLast": "Недійсна тривалість зсуву. Час початку останнього розділу перевищує тривалість цієї аудіокниги.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "Недійсна величина зсуву. Перший розділ матиме нульову або від’ємну тривалість і буде перезаписаний другим розділом. Збільште початкову тривалість другого розділу.",
|
||||
"ToastChaptersMustHaveTitles": "Глави повинні мати назви",
|
||||
"ToastChaptersRemoved": "Розділи видалені",
|
||||
"ToastChaptersUpdated": "Розділи оновлені",
|
||||
|
||||
@@ -177,6 +177,7 @@
|
||||
"HeaderPlaylist": "播放列表",
|
||||
"HeaderPlaylistItems": "播放列表项目",
|
||||
"HeaderPodcastsToAdd": "要添加的播客",
|
||||
"HeaderPresets": "预设",
|
||||
"HeaderPreviewCover": "预览封面",
|
||||
"HeaderRSSFeedGeneral": "RSS 详细信息",
|
||||
"HeaderRSSFeedIsOpen": "RSS 源已打开",
|
||||
@@ -530,6 +531,7 @@
|
||||
"LabelReleaseDate": "发布日期",
|
||||
"LabelRemoveAllMetadataAbs": "删除所有 metadata.abs 文件",
|
||||
"LabelRemoveAllMetadataJson": "删除所有 metadata.json 文件",
|
||||
"LabelRemoveAudibleBranding": "删除章节中的 Audible 简介和结尾",
|
||||
"LabelRemoveCover": "移除封面",
|
||||
"LabelRemoveMetadataFile": "删除库项目文件夹中的元数据文件",
|
||||
"LabelRemoveMetadataFileHelp": "删除 {0} 文件夹中的所有 metadata.json 和 metadata.abs 文件.",
|
||||
@@ -706,6 +708,7 @@
|
||||
"MessageAddToPlayerQueue": "添加到播放队列",
|
||||
"MessageAppriseDescription": "要使用此功能,你需要运行一个 <a href=\"https://github.com/caronc/apprise-api\" target=\"_blank\">Apprise API</a> 实例或一个可以处理这些相同请求的 API. <br />Apprise API Url 应该是发送通知的完整 URL 路径, 例如: 如果你的 API 实例运行在 <code>http://192.168.1.1:8337</code>, 那么你可以输入 <code>http://192.168.1.1:8337/notify</code>.",
|
||||
"MessageAsinCheck": "确保你使用的 ASIN 来自正确的 Audible 地区, 而不是亚马逊.",
|
||||
"MessageAuthenticationOIDCChangesRestart": "保存后重新启动服务器以应用 OIDC 更改.",
|
||||
"MessageBackupsDescription": "备份包括用户, 用户进度, 媒体库项目详细信息, 服务器设置和图像, 存储在 <code>/metadata/items</code> & <code>/metadata/authors</code>. 备份不包括存储在你的媒体库文件夹中的任何文件.",
|
||||
"MessageBackupsLocationEditNote": "注意: 更新备份位置不会移动或修改现有备份",
|
||||
"MessageBackupsLocationNoEditNote": "注意: 备份位置是通过环境变量设置的, 不能在此处更改.",
|
||||
@@ -853,6 +856,7 @@
|
||||
"MessageScheduleRunEveryWeekdayAtTime": "每隔 {0} 在 {1} 运行一次",
|
||||
"MessageSearchResultsFor": "搜索结果",
|
||||
"MessageSelected": "{0} 已选择",
|
||||
"MessageSeriesSequenceCannotContainSpaces": "系列序列不能包含空格",
|
||||
"MessageServerCouldNotBeReached": "无法访问服务器",
|
||||
"MessageSetChaptersFromTracksDescription": "把每个音频文件设置为章节并将章节标题设置为音频文件名",
|
||||
"MessageShareExpirationWillBe": "到期日期为 <strong>{0}</strong>",
|
||||
@@ -971,6 +975,8 @@
|
||||
"ToastCachePurgeFailed": "清除缓存失败",
|
||||
"ToastCachePurgeSuccess": "缓存清除成功",
|
||||
"ToastChaptersHaveErrors": "章节有错误",
|
||||
"ToastChaptersInvalidShiftAmountLast": "偏移量无效. 最后一章的开始时间将超过这本有声读物的持续时间.",
|
||||
"ToastChaptersInvalidShiftAmountStart": "偏移量无效. 第一章的长度将为零或负数, 并会被第二章覆盖. 请增加第二章的起始时长.",
|
||||
"ToastChaptersMustHaveTitles": "章节必须有标题",
|
||||
"ToastChaptersRemoved": "已删除章节",
|
||||
"ToastChaptersUpdated": "章节已更新",
|
||||
|
||||
4
package-lock.json
generated
4
package-lock.json
generated
@@ -1,12 +1,12 @@
|
||||
{
|
||||
"name": "audiobookshelf",
|
||||
"version": "2.21.0",
|
||||
"version": "2.23.0",
|
||||
"lockfileVersion": 3,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "audiobookshelf",
|
||||
"version": "2.21.0",
|
||||
"version": "2.23.0",
|
||||
"license": "GPL-3.0",
|
||||
"dependencies": {
|
||||
"axios": "^0.27.2",
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "audiobookshelf",
|
||||
"version": "2.21.0",
|
||||
"version": "2.23.0",
|
||||
"buildNumber": 1,
|
||||
"description": "Self-hosted audiobook and podcast server",
|
||||
"main": "index.js",
|
||||
|
||||
@@ -20,10 +20,7 @@ class Auth {
|
||||
// Map of openId sessions indexed by oauth2 state-variable
|
||||
this.openIdAuthSession = new Map()
|
||||
const escapedRouterBasePath = escapeRegExp(global.RouterBasePath)
|
||||
this.ignorePatterns = [
|
||||
new RegExp(`^(${escapedRouterBasePath}/api)?/items/[^/]+/cover$`),
|
||||
new RegExp(`^(${escapedRouterBasePath}/api)?/authors/[^/]+/image$`)
|
||||
]
|
||||
this.ignorePatterns = [new RegExp(`^(${escapedRouterBasePath}/api)?/items/[^/]+/cover$`), new RegExp(`^(${escapedRouterBasePath}/api)?/authors/[^/]+/image$`)]
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -310,10 +310,10 @@ class Server {
|
||||
})
|
||||
)
|
||||
router.use(express.urlencoded({ extended: true, limit: '5mb' }))
|
||||
router.use(express.json({ limit: '5mb' }))
|
||||
router.use(express.json({ limit: '10mb' }))
|
||||
|
||||
router.use('/api', this.auth.ifAuthNeeded(this.authMiddleware.bind(this)), this.apiRouter.router)
|
||||
router.use('/hls', this.authMiddleware.bind(this), this.hlsRouter.router)
|
||||
router.use('/hls', this.hlsRouter.router)
|
||||
router.use('/public', this.publicRouter.router)
|
||||
|
||||
// Static path to generated nuxt
|
||||
@@ -395,10 +395,19 @@ class Server {
|
||||
})
|
||||
router.get('/healthcheck', (req, res) => res.sendStatus(200))
|
||||
|
||||
this.server.listen(this.Port, this.Host, () => {
|
||||
if (this.Host) Logger.info(`Listening on http://${this.Host}:${this.Port}`)
|
||||
else Logger.info(`Listening on port :${this.Port}`)
|
||||
})
|
||||
const unixSocketPrefix = 'unix/'
|
||||
if (this.Host?.startsWith(unixSocketPrefix)) {
|
||||
const sockPath = this.Host.slice(unixSocketPrefix.length)
|
||||
this.server.listen(sockPath, async () => {
|
||||
await fs.chmod(sockPath, 0o666)
|
||||
Logger.info(`Listening on unix socket ${sockPath}`)
|
||||
})
|
||||
} else {
|
||||
this.server.listen(this.Port, this.Host, () => {
|
||||
if (this.Host) Logger.info(`Listening on http://${this.Host}:${this.Port}`)
|
||||
else Logger.info(`Listening on port :${this.Port}`)
|
||||
})
|
||||
}
|
||||
|
||||
// Start listening for socket connections
|
||||
SocketAuthority.initialize(this)
|
||||
|
||||
@@ -84,49 +84,67 @@ class FileSystemController {
|
||||
*/
|
||||
async checkPathExists(req, res) {
|
||||
if (!req.user.canUpload) {
|
||||
Logger.error(`[FileSystemController] Non-admin user "${req.user.username}" attempting to check path exists`)
|
||||
Logger.error(`[FileSystemController] User "${req.user.username}" without upload permissions attempting to check path exists`)
|
||||
return res.sendStatus(403)
|
||||
}
|
||||
|
||||
const { filepath, directory, folderPath } = req.body
|
||||
const { directory, folderPath } = req.body
|
||||
|
||||
if (!filepath?.length || typeof filepath !== 'string') {
|
||||
if (!directory?.length || typeof directory !== 'string' || !folderPath?.length || typeof folderPath !== 'string') {
|
||||
Logger.error(`[FileSystemController] Invalid request body: ${JSON.stringify(req.body)}`)
|
||||
return res.status(400).json({
|
||||
error: 'Invalid request body'
|
||||
})
|
||||
}
|
||||
|
||||
// Check that library folder exists
|
||||
const libraryFolder = await Database.libraryFolderModel.findOne({
|
||||
where: {
|
||||
path: folderPath
|
||||
}
|
||||
})
|
||||
|
||||
if (!libraryFolder) {
|
||||
Logger.error(`[FileSystemController] Library folder not found: ${folderPath}`)
|
||||
return res.sendStatus(404)
|
||||
}
|
||||
|
||||
const filepath = Path.posix.join(libraryFolder.path, directory)
|
||||
// Ensure filepath is inside library folder (prevents directory traversal)
|
||||
if (!filepath.startsWith(libraryFolder.path)) {
|
||||
Logger.error(`[FileSystemController] Filepath is not inside library folder: ${filepath}`)
|
||||
return res.sendStatus(400)
|
||||
}
|
||||
|
||||
const exists = await fs.pathExists(filepath)
|
||||
|
||||
if (exists) {
|
||||
if (await fs.pathExists(filepath)) {
|
||||
return res.json({
|
||||
exists: true
|
||||
})
|
||||
}
|
||||
|
||||
// If directory and folderPath are passed in, check if a library item exists in a subdirectory
|
||||
// Check if a library item exists in a subdirectory
|
||||
// See: https://github.com/advplyr/audiobookshelf/issues/4146
|
||||
if (typeof directory === 'string' && typeof folderPath === 'string' && directory.length > 0 && folderPath.length > 0) {
|
||||
const cleanedDirectory = directory.split('/').filter(Boolean).join('/')
|
||||
if (cleanedDirectory.includes('/')) {
|
||||
// Can only be 2 levels deep
|
||||
const possiblePaths = []
|
||||
const subdir = Path.dirname(directory)
|
||||
possiblePaths.push(fileUtils.filePathToPOSIX(Path.join(folderPath, subdir)))
|
||||
if (subdir.includes('/')) {
|
||||
possiblePaths.push(fileUtils.filePathToPOSIX(Path.join(folderPath, Path.dirname(subdir))))
|
||||
}
|
||||
const cleanedDirectory = directory.split('/').filter(Boolean).join('/')
|
||||
if (cleanedDirectory.includes('/')) {
|
||||
// Can only be 2 levels deep
|
||||
const possiblePaths = []
|
||||
const subdir = Path.dirname(directory)
|
||||
possiblePaths.push(fileUtils.filePathToPOSIX(Path.join(folderPath, subdir)))
|
||||
if (subdir.includes('/')) {
|
||||
possiblePaths.push(fileUtils.filePathToPOSIX(Path.join(folderPath, Path.dirname(subdir))))
|
||||
}
|
||||
|
||||
const libraryItem = await Database.libraryItemModel.findOne({
|
||||
where: {
|
||||
path: possiblePaths
|
||||
}
|
||||
const libraryItem = await Database.libraryItemModel.findOne({
|
||||
where: {
|
||||
path: possiblePaths
|
||||
}
|
||||
})
|
||||
|
||||
if (libraryItem) {
|
||||
return res.json({
|
||||
exists: true,
|
||||
libraryItemTitle: libraryItem.title
|
||||
})
|
||||
|
||||
if (libraryItem) {
|
||||
return res.json({
|
||||
exists: true,
|
||||
libraryItemTitle: libraryItem.title
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -834,8 +834,8 @@ class LibraryItemController {
|
||||
}
|
||||
|
||||
if (req.libraryItem.isMissing || !req.libraryItem.isBook || !req.libraryItem.media.includedAudioFiles.length) {
|
||||
Logger.error(`[LibraryItemController] Invalid library item`)
|
||||
return res.sendStatus(500)
|
||||
Logger.error(`[LibraryItemController] getMetadataObject: Invalid library item "${req.libraryItem.media.title}"`)
|
||||
return res.sendStatus(400)
|
||||
}
|
||||
|
||||
res.json(this.audioMetadataManager.getMetadataObjectForApi(req.libraryItem))
|
||||
|
||||
@@ -222,7 +222,7 @@ class MiscController {
|
||||
|
||||
// Update nameIgnorePrefix column on series
|
||||
const allSeries = await Database.seriesModel.findAll({
|
||||
attributes: ['id', 'name', 'nameIgnorePrefix']
|
||||
attributes: ['id', 'name', 'nameIgnorePrefix', 'libraryId']
|
||||
})
|
||||
const bulkUpdateSeries = []
|
||||
allSeries.forEach((series) => {
|
||||
@@ -230,6 +230,8 @@ class MiscController {
|
||||
if (nameIgnorePrefix !== series.nameIgnorePrefix) {
|
||||
bulkUpdateSeries.push({
|
||||
id: series.id,
|
||||
name: series.name,
|
||||
libraryId: series.libraryId,
|
||||
nameIgnorePrefix
|
||||
})
|
||||
}
|
||||
|
||||
@@ -9,6 +9,7 @@ const fs = require('../libs/fsExtra')
|
||||
const { getPodcastFeed, findMatchingEpisodes } = require('../utils/podcastUtils')
|
||||
const { getFileTimestampsWithIno, filePathToPOSIX } = require('../utils/fileUtils')
|
||||
const { validateUrl } = require('../utils/index')
|
||||
const htmlSanitizer = require('../utils/htmlSanitizer')
|
||||
|
||||
const Scanner = require('../scanner/Scanner')
|
||||
const CoverManager = require('../managers/CoverManager')
|
||||
@@ -404,6 +405,15 @@ class PodcastController {
|
||||
const supportedStringKeys = ['title', 'subtitle', 'description', 'pubDate', 'episode', 'season', 'episodeType']
|
||||
for (const key in req.body) {
|
||||
if (supportedStringKeys.includes(key) && typeof req.body[key] === 'string') {
|
||||
// Sanitize description HTML
|
||||
if (key === 'description' && req.body[key]) {
|
||||
const sanitizedDescription = htmlSanitizer.sanitize(req.body[key])
|
||||
if (sanitizedDescription !== req.body[key]) {
|
||||
Logger.debug(`[PodcastController] Sanitized description from "${req.body[key]}" to "${sanitizedDescription}"`)
|
||||
req.body[key] = sanitizedDescription
|
||||
}
|
||||
}
|
||||
|
||||
updatePayload[key] = req.body[key]
|
||||
} else if (key === 'chapters' && Array.isArray(req.body[key]) && req.body[key].every((ch) => typeof ch === 'object' && ch.title && ch.start)) {
|
||||
updatePayload[key] = req.body[key]
|
||||
|
||||
@@ -1,7 +1,9 @@
|
||||
const Path = require('path')
|
||||
const { Request, Response, NextFunction } = require('express')
|
||||
const Logger = require('../Logger')
|
||||
const Database = require('../Database')
|
||||
const { toNumber, isUUID } = require('../utils/index')
|
||||
const { getAudioMimeTypeFromExtname, encodeUriPath } = require('../utils/fileUtils')
|
||||
|
||||
const ShareManager = require('../managers/ShareManager')
|
||||
|
||||
@@ -266,6 +268,51 @@ class SessionController {
|
||||
this.playbackSessionManager.syncLocalSessionsRequest(req, res)
|
||||
}
|
||||
|
||||
/**
|
||||
* GET: /public/session/:id/track/:index
|
||||
* While a session is open, this endpoint can be used to stream the audio track
|
||||
*
|
||||
* @this {import('../routers/PublicRouter')}
|
||||
*
|
||||
* @param {Request} req
|
||||
* @param {Response} res
|
||||
*/
|
||||
async getTrack(req, res) {
|
||||
const audioTrackIndex = toNumber(req.params.index, null)
|
||||
if (audioTrackIndex === null) {
|
||||
Logger.error(`[SessionController] Invalid audio track index "${req.params.index}"`)
|
||||
return res.sendStatus(400)
|
||||
}
|
||||
|
||||
const playbackSession = this.playbackSessionManager.getSession(req.params.id)
|
||||
if (!playbackSession) {
|
||||
Logger.error(`[SessionController] Unable to find playback session with id=${req.params.id}`)
|
||||
return res.sendStatus(404)
|
||||
}
|
||||
|
||||
const audioTrack = playbackSession.audioTracks.find((t) => t.index === audioTrackIndex)
|
||||
if (!audioTrack) {
|
||||
Logger.error(`[SessionController] Unable to find audio track with index=${audioTrackIndex}`)
|
||||
return res.sendStatus(404)
|
||||
}
|
||||
|
||||
const user = await Database.userModel.getUserById(playbackSession.userId)
|
||||
Logger.debug(`[SessionController] Serving audio track ${audioTrack.index} for session "${req.params.id}" belonging to user "${user.username}"`)
|
||||
|
||||
if (global.XAccel) {
|
||||
const encodedURI = encodeUriPath(global.XAccel + audioTrack.metadata.path)
|
||||
Logger.debug(`Use X-Accel to serve static file ${encodedURI}`)
|
||||
return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send()
|
||||
}
|
||||
|
||||
// Express does not set the correct mimetype for m4b files so use our defined mimetypes if available
|
||||
const audioMimeType = getAudioMimeTypeFromExtname(Path.extname(audioTrack.metadata.path))
|
||||
if (audioMimeType) {
|
||||
res.setHeader('Content-Type', audioMimeType)
|
||||
}
|
||||
res.sendFile(audioTrack.metadata.path)
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {RequestWithUser} req
|
||||
|
||||
@@ -48,6 +48,7 @@ class ToolsController {
|
||||
}
|
||||
|
||||
const options = req.query || {}
|
||||
Logger.info(`[ToolsController] encodeM4b: Starting audiobook merge for "${req.libraryItem.media.title}" with options: ${JSON.stringify(options)}`)
|
||||
this.abMergeManager.startAudiobookMerge(req.user.id, req.libraryItem, options)
|
||||
|
||||
res.sendStatus(200)
|
||||
|
||||
@@ -26,6 +26,12 @@ class PlaybackSessionManager {
|
||||
this.sessions = []
|
||||
}
|
||||
|
||||
/**
|
||||
* Get open session by id
|
||||
*
|
||||
* @param {string} sessionId
|
||||
* @returns {PlaybackSession}
|
||||
*/
|
||||
getSession(sessionId) {
|
||||
return this.sessions.find((s) => s.id === sessionId)
|
||||
}
|
||||
|
||||
@@ -377,8 +377,17 @@ class Book extends Model {
|
||||
if (typeof payload.metadata[key] == 'number') {
|
||||
payload.metadata[key] = String(payload.metadata[key])
|
||||
}
|
||||
|
||||
|
||||
if ((typeof payload.metadata[key] === 'string' || payload.metadata[key] === null) && this[key] !== payload.metadata[key]) {
|
||||
// Sanitize description HTML
|
||||
if (key === 'description' && payload.metadata[key]) {
|
||||
const sanitizedDescription = htmlSanitizer.sanitize(payload.metadata[key])
|
||||
if (sanitizedDescription !== payload.metadata[key]) {
|
||||
Logger.debug(`[Book] "${this.title}" Sanitized description from "${payload.metadata[key]}" to "${sanitizedDescription}"`)
|
||||
payload.metadata[key] = sanitizedDescription
|
||||
}
|
||||
}
|
||||
|
||||
this[key] = payload.metadata[key] || null
|
||||
|
||||
if (key === 'title') {
|
||||
|
||||
@@ -246,7 +246,6 @@ class LibraryItem extends Model {
|
||||
include
|
||||
})
|
||||
if (!libraryItem) {
|
||||
Logger.error(`[LibraryItem] Library item not found`)
|
||||
return null
|
||||
}
|
||||
|
||||
|
||||
@@ -246,9 +246,10 @@ class MediaProgress extends Model {
|
||||
// For local sync
|
||||
if (progressPayload.lastUpdate) {
|
||||
this.updatedAt = progressPayload.lastUpdate
|
||||
this.changed('updatedAt', true)
|
||||
}
|
||||
|
||||
return this.save()
|
||||
return this.save({ silent: !!progressPayload.lastUpdate })
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -2,6 +2,7 @@ const { DataTypes, Model } = require('sequelize')
|
||||
const { getTitlePrefixAtEnd, getTitleIgnorePrefix } = require('../utils')
|
||||
const Logger = require('../Logger')
|
||||
const libraryItemsPodcastFilters = require('../utils/queries/libraryItemsPodcastFilters')
|
||||
const htmlSanitizer = require('../utils/htmlSanitizer')
|
||||
|
||||
/**
|
||||
* @typedef PodcastExpandedProperties
|
||||
@@ -215,6 +216,15 @@ class Podcast extends Model {
|
||||
newKey = 'itunesPageURL'
|
||||
}
|
||||
if ((typeof payload.metadata[key] === 'string' || payload.metadata[key] === null) && payload.metadata[key] !== this[newKey]) {
|
||||
// Sanitize description HTML
|
||||
if (key === 'description' && payload.metadata[key]) {
|
||||
const sanitizedDescription = htmlSanitizer.sanitize(payload.metadata[key])
|
||||
if (sanitizedDescription !== payload.metadata[key]) {
|
||||
Logger.debug(`[Podcast] "${this.title}" Sanitized description from "${payload.metadata[key]}" to "${sanitizedDescription}"`)
|
||||
payload.metadata[key] = sanitizedDescription
|
||||
}
|
||||
}
|
||||
|
||||
this[newKey] = payload.metadata[key] || null
|
||||
|
||||
if (key === 'title') {
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
const express = require('express')
|
||||
const ShareController = require('../controllers/ShareController')
|
||||
const SessionController = require('../controllers/SessionController')
|
||||
|
||||
class PublicRouter {
|
||||
constructor(playbackSessionManager) {
|
||||
@@ -17,6 +18,7 @@ class PublicRouter {
|
||||
this.router.get('/share/:slug/cover', ShareController.getMediaItemShareCoverImage.bind(this))
|
||||
this.router.get('/share/:slug/download', ShareController.downloadMediaItemShare.bind(this))
|
||||
this.router.patch('/share/:slug/progress', ShareController.updateMediaItemShareProgress.bind(this))
|
||||
this.router.get('/session/:id/track/:index', SessionController.getTrack.bind(this))
|
||||
}
|
||||
}
|
||||
module.exports = PublicRouter
|
||||
|
||||
@@ -407,7 +407,7 @@ class LibraryScanner {
|
||||
const folder = library.libraryFolders[0]
|
||||
|
||||
const filePathItems = folderGroups[folderId].fileUpdates.map((fileUpdate) => fileUtils.getFilePathItemFromFileUpdate(fileUpdate))
|
||||
const fileUpdateGroup = scanUtils.groupFileItemsIntoLibraryItemDirs(library.mediaType, filePathItems, !!library.settings?.audiobooksOnly)
|
||||
const fileUpdateGroup = scanUtils.groupFileItemsIntoLibraryItemDirs(library.mediaType, filePathItems, !!library.settings?.audiobooksOnly, true)
|
||||
|
||||
if (!Object.keys(fileUpdateGroup).length) {
|
||||
Logger.info(`[LibraryScanner] No important changes to scan for in folder "${folderId}"`)
|
||||
|
||||
@@ -242,7 +242,7 @@ module.exports.recurseFiles = async (path, relPathToReplace = null) => {
|
||||
})
|
||||
.filter((item) => {
|
||||
// Filter out items in ignore directories
|
||||
if (directoriesToIgnore.some((dir) => item.fullname.startsWith(dir))) {
|
||||
if (directoriesToIgnore.some((dir) => item.fullname.startsWith(dir + '/'))) {
|
||||
Logger.debug(`[fileUtils] Ignoring path in dir with .ignore "${item.fullname}"`)
|
||||
return false
|
||||
}
|
||||
|
||||
@@ -205,7 +205,7 @@ function extractEpisodeData(item) {
|
||||
} else if (typeof guidItem?._ === 'string') {
|
||||
episode.guid = guidItem._
|
||||
} else {
|
||||
Logger.error(`[podcastUtils] Invalid guid ${item['guid']} for ${episode.enclosure.url}`)
|
||||
Logger.error(`[podcastUtils] Invalid guid for ${episode.enclosure.url}`, item['guid'])
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1195,10 +1195,12 @@ module.exports = {
|
||||
libraryItem.media.series = []
|
||||
return libraryItem.toOldJSON()
|
||||
})
|
||||
seriesMatches.push({
|
||||
series: series.toOldJSON(),
|
||||
books
|
||||
})
|
||||
if (books.length) {
|
||||
seriesMatches.push({
|
||||
series: series.toOldJSON(),
|
||||
books
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// Search authors
|
||||
@@ -1247,7 +1249,12 @@ module.exports = {
|
||||
libraryId
|
||||
}
|
||||
})
|
||||
return statResults[0]
|
||||
return {
|
||||
totalSize: statResults?.[0]?.totalSize || 0,
|
||||
totalDuration: statResults?.[0]?.totalDuration || 0,
|
||||
numAudioFiles: statResults?.[0]?.numAudioFiles || 0,
|
||||
totalItems: statResults?.[0]?.totalItems || 0
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
|
||||
@@ -149,11 +149,12 @@ module.exports = {
|
||||
libraryId
|
||||
}
|
||||
const libraryItemIncludes = []
|
||||
if (includeRSSFeed) {
|
||||
if (filterGroup === 'feed-open' || includeRSSFeed) {
|
||||
const rssFeedRequired = filterGroup === 'feed-open'
|
||||
libraryItemIncludes.push({
|
||||
model: Database.feedModel,
|
||||
required: filterGroup === 'feed-open',
|
||||
separate: true
|
||||
required: rssFeedRequired,
|
||||
separate: !rssFeedRequired
|
||||
})
|
||||
}
|
||||
if (filterGroup === 'issues') {
|
||||
@@ -411,6 +412,43 @@ module.exports = {
|
||||
})
|
||||
}
|
||||
|
||||
// Search podcast episode title
|
||||
const podcastEpisodes = await Database.podcastEpisodeModel.findAll({
|
||||
where: [
|
||||
Sequelize.literal(textSearchQuery.matchExpression('podcastEpisode.title')),
|
||||
{
|
||||
'$podcast.libraryItem.libraryId$': library.id
|
||||
}
|
||||
],
|
||||
replacements: userPermissionPodcastWhere.replacements,
|
||||
include: [
|
||||
{
|
||||
model: Database.podcastModel,
|
||||
where: [...userPermissionPodcastWhere.podcastWhere],
|
||||
include: [
|
||||
{
|
||||
model: Database.libraryItemModel
|
||||
}
|
||||
]
|
||||
}
|
||||
],
|
||||
distinct: true,
|
||||
offset,
|
||||
limit
|
||||
})
|
||||
const episodeMatches = []
|
||||
for (const episode of podcastEpisodes) {
|
||||
const libraryItem = episode.podcast.libraryItem
|
||||
libraryItem.media = episode.podcast
|
||||
libraryItem.media.podcastEpisodes = []
|
||||
const oldPodcastEpisodeJson = episode.toOldJSONExpanded(libraryItem.id)
|
||||
const libraryItemJson = libraryItem.toOldJSONExpanded()
|
||||
libraryItemJson.recentEpisode = oldPodcastEpisodeJson
|
||||
episodeMatches.push({
|
||||
libraryItem: libraryItemJson
|
||||
})
|
||||
}
|
||||
|
||||
const matchJsonValue = textSearchQuery.matchExpression('json_each.value')
|
||||
|
||||
// Search tags
|
||||
@@ -450,7 +488,8 @@ module.exports = {
|
||||
return {
|
||||
podcast: itemMatches,
|
||||
tags: tagMatches,
|
||||
genres: genreMatches
|
||||
genres: genreMatches,
|
||||
episodes: episodeMatches
|
||||
}
|
||||
},
|
||||
|
||||
@@ -533,8 +572,10 @@ module.exports = {
|
||||
}
|
||||
})
|
||||
return {
|
||||
...statResults[0],
|
||||
totalSize: sizeResults[0].totalSize || 0
|
||||
totalDuration: statResults?.[0]?.totalDuration || 0,
|
||||
numAudioFiles: statResults?.[0]?.numAudioFiles || 0,
|
||||
totalItems: statResults?.[0]?.totalItems || 0,
|
||||
totalSize: sizeResults?.[0]?.totalSize || 0
|
||||
}
|
||||
},
|
||||
|
||||
|
||||
@@ -24,6 +24,12 @@ function isMediaFile(mediaType, ext, audiobooksOnly = false) {
|
||||
return globals.SupportedAudioTypes.includes(extclean) || globals.SupportedEbookTypes.includes(extclean)
|
||||
}
|
||||
|
||||
function isScannableNonMediaFile(ext) {
|
||||
if (!ext) return false
|
||||
const extclean = ext.slice(1).toLowerCase()
|
||||
return globals.TextFileTypes.includes(extclean) || globals.MetadataFileTypes.includes(extclean) || globals.SupportedImageTypes.includes(extclean)
|
||||
}
|
||||
|
||||
function checkFilepathIsAudioFile(filepath) {
|
||||
const ext = Path.extname(filepath)
|
||||
if (!ext) return false
|
||||
@@ -35,27 +41,31 @@ module.exports.checkFilepathIsAudioFile = checkFilepathIsAudioFile
|
||||
/**
|
||||
* @param {string} mediaType
|
||||
* @param {import('./fileUtils').FilePathItem[]} fileItems
|
||||
* @param {boolean} [audiobooksOnly=false]
|
||||
* @param {boolean} audiobooksOnly
|
||||
* @param {boolean} [includeNonMediaFiles=false] - Used by the watcher to re-scan when covers/metadata files are added/removed
|
||||
* @returns {Record<string,string[]>} map of files grouped into potential libarary item dirs
|
||||
*/
|
||||
function groupFileItemsIntoLibraryItemDirs(mediaType, fileItems, audiobooksOnly = false) {
|
||||
function groupFileItemsIntoLibraryItemDirs(mediaType, fileItems, audiobooksOnly, includeNonMediaFiles = false) {
|
||||
// Step 1: Filter out non-book-media files in root dir (with depth of 0)
|
||||
const itemsFiltered = fileItems.filter((i) => {
|
||||
return i.deep > 0 || (mediaType === 'book' && isMediaFile(mediaType, i.extension, audiobooksOnly))
|
||||
})
|
||||
|
||||
// Step 2: Separate media files and other files
|
||||
// - Directories without a media file will not be included
|
||||
// - Directories without a media file will not be included (unless includeNonMediaFiles is true)
|
||||
/** @type {import('./fileUtils').FilePathItem[]} */
|
||||
const mediaFileItems = []
|
||||
/** @type {import('./fileUtils').FilePathItem[]} */
|
||||
const otherFileItems = []
|
||||
itemsFiltered.forEach((item) => {
|
||||
if (isMediaFile(mediaType, item.extension, audiobooksOnly)) mediaFileItems.push(item)
|
||||
else otherFileItems.push(item)
|
||||
if (isMediaFile(mediaType, item.extension, audiobooksOnly) || (includeNonMediaFiles && isScannableNonMediaFile(item.extension))) {
|
||||
mediaFileItems.push(item)
|
||||
} else {
|
||||
otherFileItems.push(item)
|
||||
}
|
||||
})
|
||||
|
||||
// Step 3: Group audio files in library items
|
||||
// Step 3: Group media files (or non-media files if includeNonMediaFiles is true) in library items
|
||||
const libraryItemGroup = {}
|
||||
mediaFileItems.forEach((item) => {
|
||||
const dirparts = item.reldirpath.split('/').filter((p) => !!p)
|
||||
|
||||
0
tailwind.config.js
Normal file
0
tailwind.config.js
Normal file
@@ -47,7 +47,7 @@ describe('fileUtils', () => {
|
||||
|
||||
// Mock file structure with normalized paths
|
||||
const mockDirContents = new Map([
|
||||
['/test', ['file1.mp3', 'subfolder', 'ignoreme', 'temp.mp3.tmp']],
|
||||
['/test', ['file1.mp3', 'subfolder', 'ignoreme', 'ignoremenot.mp3', 'temp.mp3.tmp']],
|
||||
['/test/subfolder', ['file2.m4b']],
|
||||
['/test/ignoreme', ['.ignore', 'ignored.mp3']]
|
||||
])
|
||||
@@ -59,7 +59,8 @@ describe('fileUtils', () => {
|
||||
['/test/ignoreme', { isDirectory: () => true, size: 0, mtimeMs: Date.now(), ino: '4' }],
|
||||
['/test/ignoreme/.ignore', { isDirectory: () => false, size: 0, mtimeMs: Date.now(), ino: '5' }],
|
||||
['/test/ignoreme/ignored.mp3', { isDirectory: () => false, size: 1024, mtimeMs: Date.now(), ino: '6' }],
|
||||
['/test/temp.mp3.tmp', { isDirectory: () => false, size: 1024, mtimeMs: Date.now(), ino: '7' }]
|
||||
['/test/ignoremenot.mp3', { isDirectory: () => false, size: 1024, mtimeMs: Date.now(), ino: '7' }],
|
||||
['/test/temp.mp3.tmp', { isDirectory: () => false, size: 1024, mtimeMs: Date.now(), ino: '8' }]
|
||||
])
|
||||
|
||||
// Stub fs.readdir
|
||||
@@ -103,7 +104,7 @@ describe('fileUtils', () => {
|
||||
it('should return filtered file list', async () => {
|
||||
const files = await fileUtils.recurseFiles('/test')
|
||||
expect(files).to.be.an('array')
|
||||
expect(files).to.have.lengthOf(2)
|
||||
expect(files).to.have.lengthOf(3)
|
||||
|
||||
expect(files[0]).to.deep.equal({
|
||||
name: 'file1.mp3',
|
||||
@@ -115,6 +116,15 @@ describe('fileUtils', () => {
|
||||
})
|
||||
|
||||
expect(files[1]).to.deep.equal({
|
||||
name: 'ignoremenot.mp3',
|
||||
path: 'ignoremenot.mp3',
|
||||
reldirpath: '',
|
||||
fullpath: '/test/ignoremenot.mp3',
|
||||
extension: '.mp3',
|
||||
deep: 0
|
||||
})
|
||||
|
||||
expect(files[2]).to.deep.equal({
|
||||
name: 'file2.m4b',
|
||||
path: 'subfolder/file2.m4b',
|
||||
reldirpath: 'subfolder',
|
||||
|
||||
Reference in New Issue
Block a user