mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-11-04 03:17:00 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			318 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			318 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<template>
 | 
						|
  <div class="w-full h-full overflow-hidden overflow-y-auto px-4 py-6 relative">
 | 
						|
    <div class="flex">
 | 
						|
      <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">
 | 
						|
            <span class="material-icons">delete</span>
 | 
						|
          </div>
 | 
						|
        </div>
 | 
						|
      </div>
 | 
						|
      <div class="flex-grow pl-6 pr-2">
 | 
						|
        <div class="flex items-center">
 | 
						|
          <div v-if="userCanUpload" class="w-40 pr-2 pt-4" style="min-width: 160px">
 | 
						|
            <ui-file-input ref="fileInput" @change="fileUploadSelected">Upload Cover</ui-file-input>
 | 
						|
          </div>
 | 
						|
          <form @submit.prevent="submitForm" class="flex flex-grow">
 | 
						|
            <ui-text-input-with-label v-model="imageUrl" label="Cover Image URL" />
 | 
						|
            <ui-btn color="success" type="submit" :padding-x="4" class="mt-5 ml-3 w-24">Update</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 ? 'Hide' : 'Show' }}</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 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' }">
 | 
						|
                  <img :src="`${cover.localPath}?token=${userToken}`" class="h-full w-full object-contain" />
 | 
						|
                </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="Provider" small />
 | 
						|
        </div>
 | 
						|
        <div class="w-72 px-1">
 | 
						|
          <ui-text-input-with-label v-model="searchTitle" :label="searchTitleLabel" placeholder="Search" />
 | 
						|
        </div>
 | 
						|
        <div v-show="provider != 'itunes'" class="w-72 px-1">
 | 
						|
          <ui-text-input-with-label v-model="searchAuthor" label="Author" />
 | 
						|
        </div>
 | 
						|
        <ui-btn class="mt-5 ml-1" type="submit">Search</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">No Covers Found</p>
 | 
						|
      <template v-for="cover in coversFound">
 | 
						|
        <div :key="cover" class="m-0.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">Preview Cover</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">Clear</ui-btn>
 | 
						|
        <ui-btn :loading="processingUpload" color="success" @click="submitCoverUpload">Upload</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 == 'audible') return 'Search Title or ASIN'
 | 
						|
      else if (this.provider == 'itunes') return 'Search Term'
 | 
						|
      return 'Search Title'
 | 
						|
    },
 | 
						|
    coverAspectRatio() {
 | 
						|
      return this.$store.getters['getServerSetting']('coverAspectRatio')
 | 
						|
    },
 | 
						|
    bookCoverAspectRatio() {
 | 
						|
      return this.coverAspectRatio === this.$constants.BookCoverAspectRatio.SQUARE ? 1 : 1.6
 | 
						|
    },
 | 
						|
    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 = `/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> |