mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-09 15:38:37 +01:00
fdfb07ff2c
Starting to use audiobookshelf, the function of some buttons weren't very clear to me and while some buttons have tooltips, others have not. This patch adds some additional tooltips to the user interface, further explaining some of the functionality.
320 lines
12 KiB
Vue
320 lines
12 KiB
Vue
<template>
|
|
<div class="w-full h-full overflow-hidden overflow-y-auto px-2 sm:px-4 py-6 relative">
|
|
<div class="flex flex-wrap">
|
|
<div class="relative">
|
|
<covers-book-cover :library-item="libraryItem" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
|
<!-- book cover overlay -->
|
|
<div v-if="media.coverPath" class="absolute top-0 left-0 w-full h-full z-10 opacity-0 hover:opacity-100 transition-opacity duration-100">
|
|
<div class="absolute top-0 left-0 w-full h-16 bg-gradient-to-b from-black-600 to-transparent" />
|
|
<div class="p-1 absolute top-1 right-1 text-red-500 rounded-full w-8 h-8 cursor-pointer hover:text-red-400 shadow-sm" @click="removeCover">
|
|
<ui-tooltip direction="top" :text="$strings.LabelRemoveCover">
|
|
<span class="material-icons">delete</span>
|
|
</ui-tooltip>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="flex-grow sm:pl-2 md:pl-6 sm:pr-2 mt-2 md:mt-0">
|
|
<div class="flex items-center">
|
|
<div v-if="userCanUpload" class="w-10 md:w-40 pr-2 pt-4 md:min-w-32">
|
|
<ui-file-input ref="fileInput" @change="fileUploadSelected"
|
|
><span class="hidden md:inline-block">{{ $strings.ButtonUploadCover }}</span
|
|
><span class="material-icons inline-block md:!hidden">upload</span></ui-file-input
|
|
>
|
|
</div>
|
|
<form @submit.prevent="submitForm" class="flex flex-grow">
|
|
<ui-text-input-with-label v-model="imageUrl" :label="$strings.LabelCoverImageURL" />
|
|
<ui-btn color="success" type="submit" :padding-x="4" class="mt-5 ml-2 sm:ml-3 w-24">{{ $strings.ButtonSave }}</ui-btn>
|
|
</form>
|
|
</div>
|
|
|
|
<div v-if="localCovers.length" class="mb-4 mt-6 border-t border-b border-primary">
|
|
<div class="flex items-center justify-center py-2">
|
|
<p>{{ localCovers.length }} local image{{ localCovers.length !== 1 ? 's' : '' }}</p>
|
|
<div class="flex-grow" />
|
|
<ui-btn small @click="showLocalCovers = !showLocalCovers">{{ showLocalCovers ? $strings.ButtonHide : $strings.ButtonShow }}</ui-btn>
|
|
</div>
|
|
|
|
<div v-if="showLocalCovers" class="flex items-center justify-center">
|
|
<template v-for="cover in localCovers">
|
|
<div :key="cover.path" class="m-0.5 mb-5 border-2 border-transparent hover:border-yellow-300 cursor-pointer" :class="cover.metadata.path === coverPath ? 'border-yellow-300' : ''" @click="setCover(cover)">
|
|
<div class="h-24 bg-primary" :style="{ width: 96 / bookCoverAspectRatio + 'px' }">
|
|
<covers-preview-cover :src="`${cover.localPath}?token=${userToken}`" :width="96 / bookCoverAspectRatio" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<form @submit.prevent="submitSearchForm">
|
|
<div class="flex items-center justify-start -mx-1 h-20">
|
|
<div class="w-40 px-1">
|
|
<ui-dropdown v-model="provider" :items="providers" :label="$strings.LabelProvider" small />
|
|
</div>
|
|
<div class="w-72 px-1">
|
|
<ui-text-input-with-label v-model="searchTitle" :label="searchTitleLabel" :placeholder="$strings.PlaceholderSearch" />
|
|
</div>
|
|
<div v-show="provider != 'itunes'" class="w-72 px-1">
|
|
<ui-text-input-with-label v-model="searchAuthor" :label="$strings.LabelAuthor" />
|
|
</div>
|
|
<ui-btn class="mt-5 ml-1" type="submit">{{ $strings.ButtonSearch }}</ui-btn>
|
|
</div>
|
|
</form>
|
|
<div v-if="hasSearched" class="flex items-center flex-wrap justify-center max-h-80 overflow-y-scroll mt-2 max-w-full">
|
|
<p v-if="!coversFound.length">{{ $strings.MessageNoCoversFound }}</p>
|
|
<template v-for="cover in coversFound">
|
|
<div :key="cover" class="m-0.5 mb-5 border-2 border-transparent hover:border-yellow-300 cursor-pointer" :class="cover === imageUrl ? 'border-yellow-300' : ''" @click="updateCover(cover)">
|
|
<covers-preview-cover :src="cover" :width="80" show-open-new-tab :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
|
</div>
|
|
</template>
|
|
</div>
|
|
|
|
<div v-if="previewUpload" class="absolute top-0 left-0 w-full h-full z-10 bg-bg p-8">
|
|
<p class="text-lg">{{ $strings.HeaderPreviewCover }}</p>
|
|
<span class="absolute top-4 right-4 material-icons text-2xl cursor-pointer" @click="resetCoverPreview">close</span>
|
|
<div class="flex justify-center py-4">
|
|
<covers-preview-cover :src="previewUpload" :width="240" :book-cover-aspect-ratio="bookCoverAspectRatio" />
|
|
</div>
|
|
<div class="absolute bottom-0 right-0 flex py-4 px-5">
|
|
<ui-btn :disabled="processingUpload" class="mx-2" @click="resetCoverPreview">{{ $strings.ButtonReset }}</ui-btn>
|
|
<ui-btn :loading="processingUpload" color="success" @click="submitCoverUpload">{{ $strings.ButtonUpload }}</ui-btn>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
export default {
|
|
props: {
|
|
processing: Boolean,
|
|
libraryItem: {
|
|
type: Object,
|
|
default: () => {}
|
|
}
|
|
},
|
|
data() {
|
|
return {
|
|
processingUpload: false,
|
|
searchTitle: null,
|
|
searchAuthor: null,
|
|
imageUrl: null,
|
|
coversFound: [],
|
|
hasSearched: false,
|
|
showLocalCovers: false,
|
|
previewUpload: null,
|
|
selectedFile: null,
|
|
provider: 'google'
|
|
}
|
|
},
|
|
watch: {
|
|
libraryItem: {
|
|
immediate: true,
|
|
handler(newVal) {
|
|
if (newVal) {
|
|
this.init()
|
|
}
|
|
}
|
|
}
|
|
},
|
|
computed: {
|
|
isProcessing: {
|
|
get() {
|
|
return this.processing
|
|
},
|
|
set(val) {
|
|
this.$emit('update:processing', val)
|
|
}
|
|
},
|
|
providers() {
|
|
if (this.isPodcast) return this.$store.state.scanners.podcastProviders
|
|
return this.$store.state.scanners.providers
|
|
},
|
|
searchTitleLabel() {
|
|
if (this.provider.startsWith('audible')) return this.$strings.LabelSearchTitleOrASIN
|
|
else if (this.provider == 'itunes') return this.$strings.LabelSearchTerm
|
|
return this.$strings.LabelSearchTitle
|
|
},
|
|
bookCoverAspectRatio() {
|
|
return this.$store.getters['libraries/getBookCoverAspectRatio']
|
|
},
|
|
libraryItemId() {
|
|
return this.libraryItem ? this.libraryItem.id : null
|
|
},
|
|
mediaType() {
|
|
return this.libraryItem ? this.libraryItem.mediaType : null
|
|
},
|
|
isPodcast() {
|
|
return this.mediaType == 'podcast'
|
|
},
|
|
media() {
|
|
return this.libraryItem ? this.libraryItem.media || {} : {}
|
|
},
|
|
coverPath() {
|
|
return this.media.coverPath
|
|
},
|
|
mediaMetadata() {
|
|
return this.media.metadata || {}
|
|
},
|
|
libraryFiles() {
|
|
return this.libraryItem ? this.libraryItem.libraryFiles || [] : []
|
|
},
|
|
userCanUpload() {
|
|
return this.$store.getters['user/getUserCanUpload']
|
|
},
|
|
userToken() {
|
|
return this.$store.getters['user/getToken']
|
|
},
|
|
localCovers() {
|
|
return this.libraryFiles
|
|
.filter((f) => f.fileType === 'image')
|
|
.map((file) => {
|
|
var _file = { ...file }
|
|
_file.localPath = `${process.env.serverUrl}/s/item/${this.libraryItemId}/${this.$encodeUriPath(file.metadata.relPath).replace(/^\//, '')}`
|
|
return _file
|
|
})
|
|
}
|
|
},
|
|
methods: {
|
|
submitCoverUpload() {
|
|
this.processingUpload = true
|
|
var form = new FormData()
|
|
form.set('cover', this.selectedFile)
|
|
|
|
this.$axios
|
|
.$post(`/api/items/${this.libraryItemId}/cover`, form)
|
|
.then((data) => {
|
|
if (data.error) {
|
|
this.$toast.error(data.error)
|
|
} else {
|
|
this.$toast.success('Cover Uploaded')
|
|
this.resetCoverPreview()
|
|
}
|
|
this.processingUpload = false
|
|
})
|
|
.catch((error) => {
|
|
console.error('Failed', error)
|
|
if (error.response && error.response.data) {
|
|
this.$toast.error(error.response.data)
|
|
} else {
|
|
this.$toast.error('Oops, something went wrong...')
|
|
}
|
|
this.processingUpload = false
|
|
})
|
|
},
|
|
resetCoverPreview() {
|
|
if (this.$refs.fileInput) {
|
|
this.$refs.fileInput.reset()
|
|
}
|
|
this.previewUpload = null
|
|
this.selectedFile = null
|
|
},
|
|
fileUploadSelected(file) {
|
|
this.previewUpload = URL.createObjectURL(file)
|
|
this.selectedFile = file
|
|
},
|
|
init() {
|
|
this.showLocalCovers = false
|
|
if (this.coversFound.length && (this.searchTitle !== this.mediaMetadata.title || this.searchAuthor !== this.mediaMetadata.authorName)) {
|
|
this.coversFound = []
|
|
this.hasSearched = false
|
|
}
|
|
this.imageUrl = this.media.coverPath || ''
|
|
this.searchTitle = this.mediaMetadata.title || ''
|
|
this.searchAuthor = this.mediaMetadata.authorName || ''
|
|
if (this.isPodcast) this.provider = 'itunes'
|
|
else this.provider = localStorage.getItem('book-provider') || 'google'
|
|
},
|
|
removeCover() {
|
|
if (!this.media.coverPath) {
|
|
this.imageUrl = ''
|
|
return
|
|
}
|
|
this.updateCover('')
|
|
},
|
|
submitForm() {
|
|
this.updateCover(this.imageUrl)
|
|
},
|
|
async updateCover(cover) {
|
|
if (cover === this.coverPath) {
|
|
console.warn('Cover has not changed..', cover)
|
|
return
|
|
}
|
|
|
|
this.isProcessing = true
|
|
var success = false
|
|
|
|
if (!cover) {
|
|
// Remove cover
|
|
success = await this.$axios
|
|
.$delete(`/api/items/${this.libraryItemId}/cover`)
|
|
.then(() => true)
|
|
.catch((error) => {
|
|
console.error('Failed to remove cover', error)
|
|
if (error.response && error.response.data) {
|
|
this.$toast.error(error.response.data)
|
|
}
|
|
return false
|
|
})
|
|
} else if (cover.startsWith('http:') || cover.startsWith('https:')) {
|
|
// Download cover from url and use
|
|
success = await this.$axios.$post(`/api/items/${this.libraryItemId}/cover`, { url: cover }).catch((error) => {
|
|
console.error('Failed to download cover from url', error)
|
|
if (error.response && error.response.data) {
|
|
this.$toast.error(error.response.data)
|
|
}
|
|
return false
|
|
})
|
|
} else {
|
|
// Update local cover url
|
|
const updatePayload = {
|
|
cover
|
|
}
|
|
success = await this.$axios.$patch(`/api/items/${this.libraryItemId}/cover`, updatePayload).catch((error) => {
|
|
console.error('Failed to update', error)
|
|
if (error.response && error.response.data) {
|
|
this.$toast.error(error.response.data)
|
|
}
|
|
return false
|
|
})
|
|
}
|
|
if (success) {
|
|
this.$toast.success('Update Successful')
|
|
// this.$emit('close')
|
|
} else {
|
|
this.imageUrl = this.media.coverPath || ''
|
|
}
|
|
this.isProcessing = false
|
|
},
|
|
getSearchQuery() {
|
|
var searchQuery = `provider=${this.provider}&title=${this.searchTitle}`
|
|
if (this.searchAuthor) searchQuery += `&author=${this.searchAuthor}`
|
|
if (this.isPodcast) searchQuery += '&podcast=1'
|
|
return searchQuery
|
|
},
|
|
persistProvider() {
|
|
try {
|
|
localStorage.setItem('book-provider', this.provider)
|
|
} catch (error) {
|
|
console.error('PersistProvider', error)
|
|
}
|
|
},
|
|
async submitSearchForm() {
|
|
// Store provider in local storage
|
|
this.persistProvider()
|
|
|
|
this.isProcessing = true
|
|
var searchQuery = this.getSearchQuery()
|
|
var results = await this.$axios.$get(`/api/search/covers?${searchQuery}`).catch((error) => {
|
|
console.error('Failed', error)
|
|
return []
|
|
})
|
|
this.coversFound = results
|
|
this.isProcessing = false
|
|
this.hasSearched = true
|
|
},
|
|
setCover(coverFile) {
|
|
this.updateCover(coverFile.metadata.path)
|
|
}
|
|
}
|
|
}
|
|
</script> |